UOJ

思路

显然可以转化一下,变成统计异或起来等于0的集合个数,这样一个集合的贡献是\(2^{|S|}\)。

考虑朴素的\(dp_{i,j}\)表示前\(i\)个数凑出了\(j\)的方案数,发现这其实就是一堆多项式用异或卷积搞起来。第\(i\)个多项式是\(1+2x^{a_i}\)。

对\(1+2x^{a}\)FWT一下,发现结果就只有-13。为什么?根据FWT的理论,\(a_i\)会对\(FWT(a)_j\)产生\(a_i\times (-1)^{\text{bitcnt}[i\&j]}\)的贡献。

我们就是要求出最后这一堆东西乘在一起是什么,也就是对于每一位求出这里有几个-1,有几个3。

这个怎么做?脑洞一下,把所有多项式加在一起FWT,设有\(x\)个-1,那么就有方程\(-x+3(n-x)=f_i\),就可以解出来了。

最后再FWT回去,就做完了。

(这个解方程咋想到的啊qwq)

代码

最后减1不取模你人就没了qwq

  1. #include<bits/stdc++.h>
  2. clock_t t=clock();
  3. namespace my_std{
  4. using namespace std;
  5. #define pii pair<int,int>
  6. #define fir first
  7. #define sec second
  8. #define MP make_pair
  9. #define rep(i,x,y) for (int i=(x);i<=(y);i++)
  10. #define drep(i,x,y) for (int i=(x);i>=(y);i--)
  11. #define go(x) for (int i=head[x];i;i=edge[i].nxt)
  12. #define templ template<typename T>
  13. #define sz 1100000
  14. #define mod 998244353ll
  15. typedef long long ll;
  16. typedef double db;
  17. mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
  18. templ inline T rnd(T l,T r) {return uniform_int_distribution<T>(l,r)(rng);}
  19. templ inline bool chkmax(T &x,T y){return x<y?x=y,1:0;}
  20. templ inline bool chkmin(T &x,T y){return x>y?x=y,1:0;}
  21. templ inline void read(T& t)
  22. {
  23. t=0;char f=0,ch=getchar();double d=0.1;
  24. while(ch>'9'||ch<'0') f|=(ch=='-'),ch=getchar();
  25. while(ch<='9'&&ch>='0') t=t*10+ch-48,ch=getchar();
  26. if(ch=='.'){ch=getchar();while(ch<='9'&&ch>='0') t+=d*(ch^48),d*=0.1,ch=getchar();}
  27. t=(f?-t:t);
  28. }
  29. template<typename T,typename... Args>inline void read(T& t,Args&... args){read(t); read(args...);}
  30. char __sr[1<<21],__z[20];int __C=-1,__zz=0;
  31. inline void Ot(){fwrite(__sr,1,__C+1,stdout),__C=-1;}
  32. inline void print(register int x)
  33. {
  34. if(__C>1<<20)Ot();if(x<0)__sr[++__C]='-',x=-x;
  35. while(__z[++__zz]=x%10+48,x/=10);
  36. while(__sr[++__C]=__z[__zz],--__zz);__sr[++__C]='\n';
  37. }
  38. void file()
  39. {
  40. #ifdef NTFOrz
  41. freopen("a.in","r",stdin);
  42. #endif
  43. }
  44. inline void chktime()
  45. {
  46. #ifndef ONLINE_JUDGE
  47. cout<<(clock()-t)/1000.0<<'\n';
  48. #endif
  49. }
  50. #ifdef mod
  51. ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x%mod) if (y&1) ret=ret*x%mod;return ret;}
  52. ll inv(ll x){return ksm(x,mod-2);}
  53. #else
  54. ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x) if (y&1) ret=ret*x;return ret;}
  55. #endif
  56. // inline ll mul(ll a,ll b){ll d=(ll)(a*(double)b/mod+0.5);ll ret=a*b-d*mod;if (ret<0) ret+=mod;return ret;}
  57. }
  58. using namespace my_std;
  59. int n;
  60. int a[sz];
  61. int f[sz];
  62. ll g[sz];
  63. void FWT(int *a,int n)
  64. {
  65. int N=1<<n,x,y;
  66. rep(i,0,n-1)
  67. for (int mid=1<<i,j=0;j<N;j+=mid<<1)
  68. rep(k,0,mid-1)
  69. x=a[j+k],y=a[j+k+mid],a[j+k]=x+y,a[j+k+mid]=x-y;
  70. }
  71. ll I=inv(2);
  72. void iFWT(ll *a,int n)
  73. {
  74. int N=1<<n;ll x,y;
  75. rep(i,0,n-1)
  76. for (int mid=1<<i,j=0;j<N;j+=mid<<1)
  77. rep(k,0,mid-1)
  78. x=a[j+k],y=a[j+k+mid],a[j+k]=(x+y)*I%mod,a[j+k+mid]=(x-y+mod)*I%mod;
  79. }
  80. int main()
  81. {
  82. file();
  83. read(n);
  84. rep(i,1,n) read(a[i]),++f[0],f[a[i]]+=2;
  85. FWT(f,20);
  86. int x;
  87. rep(i,0,(1<<20)-1) x=(3*n-f[i])/4,g[i]=ksm(mod-1,x)*ksm(3,n-x)%mod;
  88. iFWT(g,20);
  89. printf("%lld\n",(g[0]-1+mod)%mod);
  90. return 0;
  91. }

扩展

updated on 2020.2.1

这个解方程的思路其实不止在这题可以用到,还有luogu5577。

同样是要把一堆多项式乘在一起,但扩展到\(k\)阶循环卷积。

我们还是想要把所有东西加在一起之后求出每个\(\omega\)的系数,但是现在解方程就不那么容易。为了方便起见,下面FWT只对\(\sum x^{a_i}\)做。

假设现在在算\(f_i\)的组成。令\(y=\omega_k\),有\(f_i=\sum_{j=0}^{k-1} x_{i,j} y^j\),其中\(x_{i,j}\)是我们真实想要的东西。

但是,FWT出来的系数不一定是真的\(x_{i,j}\),因为有万恶的折半引理、求和引理、消去引理……

对于\(k\)是奇质数的情况,发现只有求和引理还有效。由于我们知道\(\sum x_{i,j}=n\),所以可以求出到底消去了多少个\(\sum_j y^j\),补上就完事了。

对于\(k\)的性质不那么好的情况呢?我们考虑大力解方程。令\(y=\omega^0,\omega^1,\cdots,\omega^{k-1}\),分别求出一大堆\(x\)。由于范德蒙德矩阵的存在,可以很轻松地解出真正的\(x_{i,j}\),就做完了。

UOJ310. 【UNR #2】黎明前的巧克力 [FWT]的更多相关文章

  1. 【uoj#310】[UNR #2]黎明前的巧克力 FWT

    题目描述 给出 $n$ 个数,从中选出两个互不相交的集合,使得第一个集合与第二个集合内的数的异或和相等.求总方案数. 输入 第一行一个正整数 $n$ ,表示巧克力的个数.第二行 $n$ 个整数 $a_ ...

  2. [UOJ310][UNR #2]黎明前的巧克力

    uoj description 给你\(n\)个数,求从中选出两个交集为空的非空集合异或和相等的方案数模\(998244353\). sol 其实也就是选出一个集合满足异或和为\(0\),然后把它分成 ...

  3. uoj310【UNR #2】黎明前的巧克力(FWT)

    uoj310[UNR #2]黎明前的巧克力(FWT) uoj 题解时间 对非零项极少的FWT的优化. 首先有个十分好想的DP: $ f[i][j] $ 表示考虑了前 $ i $ 个且异或和为 $ j ...

  4. UOJ #310 黎明前的巧克力 FWT dp

    LINK:黎明前的巧克力 我发现 很多难的FWT的题 都和方程有关. 上次那个西行寺无余涅槃 也是各种解方程...(不过这个题至今还未理解. 考虑dp 容易想到f[i][j][k]表示 第一个人得到巧 ...

  5. UOJ#310 【UNR #2】黎明前的巧克力 FWT 多项式

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ310.html 题目传送门 - UOJ#310 题意 给定 $n$ 个数 ,请你选出两个不相交的集合(两个 ...

  6. UOJ#310. 【UNR #2】黎明前的巧克力(FWT)

    题意 题目链接 Sol 挂一个讲的看起来比较好的链接 然鹅我最后一步还是没看懂qwq.. 坐等SovietPower大佬发博客 #include<bits/stdc++.h> using ...

  7. [UOJ UNR#2 黎明前的巧克力]

    来自FallDream的博客,未经允许,请勿转载,谢谢. 传送门 很奇妙的一道题 首先不难发现一个暴力做法,就是f[i]表示异或和为i的答案数,每次FWT上一个F数组,其中F[0]=1,F[ai]=2 ...

  8. UOJ #310 黎明前的巧克力 (FWT)

    题目传送门 题目大意:给你一个序列,定义一个子序列的权值表示子序列中元素的异或和,现在让你选出两个互不相交的子序列,求选出的这两个子序列权值相等的方案数,$n,a_{i}\leq 10^{6}$ 这是 ...

  9. [FWT] UOJ #310. 【UNR #2】黎明前的巧克力

    [uoj#310][UNR #2]黎明前的巧克力 FWT - GXZlegend - 博客园 f[i][xor],考虑优化暴力,暴力就是FWT xor一个多项式 整体处理 (以下FWT代表第一步) F ...

随机推荐

  1. 5_PHP数组_3_数组处理函数及其应用_3_数组指针函数

    以下为学习孔祥盛主编的<PHP编程基础与实例教程>(第二版)所做的笔记. 数组指针函数 1. key() 函数 程序: <?php $interests[2] = "mus ...

  2. Jsp页面显示时间:<fmt>标签问题

    <%@ taglib prefix="fmt" uri="http://java.sun.com/jstl/fmt"%> <%@ taglib ...

  3. Mongodb 学习笔记(三) .net core SDK

    首先添加 Nuget包  MongoDB.Driver 创建一个Model. public class Student { public ObjectId _id { get; set; } publ ...

  4. 从 SimpleIntegerProperty 看 Java属性绑定(property binding) 与 观察者模式(Observable)

    //TODO:ExpressionHelper .bindBidirectional双向绑定.以及IntegerExpression的一系列算术方法和返回的IntegerBinding暂未详细解析(比 ...

  5. Java内存模型JMM简单分析

    参考博文:http://blog.csdn.net/suifeng3051/article/details/52611310 http://www.cnblogs.com/nexiyi/p/java_ ...

  6. MySQL Binlog--基于ROW模式的binlog event大小限制

    参数binlog-row-event-max-size:Specify the maximum size of a row-based binary log event, in bytes. Rows ...

  7. [daily][tmux] tmux常用快捷键

    介绍 什么是tmux? Terminal Multiplexer. 1. 如果你是linux用户,tmux就是screen的alternative. 2. 如果你是windows用户,tmux就是一个 ...

  8. 运维开发笔记整理-django日志配置

    运维开发笔记整理-django日志配置 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Django日志 Django使用python内建的logging模块打印日志,Pytho ...

  9. MySQL事件自动kill运行时间超时的SQL

    delimiter $create event my_long_running_trx_monitoron schedule every 1 minutestarts '2015-09-15 11:0 ...

  10. 采用MySQL-MMM做DB高可用时,遇到的一个小坑

    一.服务器分布   二.MySQL-MMM 配置 (1).公共配置[所有DB节点:Master1.Master2.Slave1.Slave2   Monitor节点] # vim /etc/mysql ...