【uoj#94】【集训队互测2015】胡策的统计(集合幂级数)

  题目传送门:http://uoj.ac/problem/94

  这是一道集合幂级数的入门题目。我们先考虑求出每个点集的连通生成子图个数,记为$g_S$,再记$h_S$为点集$S$的生成子图个数,容易发现,$h_S=2^{size_S}$,其中$size_S$为点集$S$的极大生成子图内的边数。特殊的,$f_{\o}=g_{\o}=0$。

  定义集合幂级数的乘法为子集卷积,考虑集合幂级数$h$和$g$的关系,我们可以得到

    $$h=1+\sum_{k \geq 1}\frac{g^k}{k!}=1+e^h$$

  因此可得

    $$g=\ln{(1+h)}$$

  我们再记$f_S$为点集$S$的生成子图的价值和,可得到

    $$f=1+\sum{k \geq 1}\frac{g^k}{k!}k!=\frac{1}{1-g}$$

  计算集合幂级数的对数和逆时,因为我们将乘法定义为子集卷积,所以可以将其映射成集合占位幂级数$f_{|S|,S}$后,在将每个集合对应的位看作形式幂级数再暴力求解。

  代码:

#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<algorithm>
#define mod 998244353
#define Mod1(x) (x>=mod?x-mod:x)
#define Mod2(x) (x<0?x+mod:x)
#define ll long long
inline ll read()
{
    ll x=0; char c=getchar(),f=1;
    for(;c<‘0‘||‘9‘<c;c=getchar())if(c==‘-‘)f=-1;
    for(;‘0‘<=c&&c<=‘9‘;c=getchar())x=x*10+c-‘0‘;
    return x*f;
}
inline void write(ll x)
{
    static int buf[20],len; len=0;
    if(x<0)x=-x,putchar(‘-‘);
    for(;x;x/=10)buf[len++]=x%10;
    if(!len)putchar(‘0‘);
    else while(len)putchar(buf[--len]+‘0‘);
}
inline void writeln(ll x){write(x); putchar(‘\n‘);}
inline void writesp(ll x){write(x); putchar(‘ ‘);}
int f[25][(1<<20)+5],g[25][(1<<20)+5];
int cnt[(1<<20)+5],inv[25];
int x[410],y[410];
int n,m;
inline ll power(ll a,ll b)
{
    ll ans=1;
    for(;b;b>>=1,a=a*a%mod)
        if(b&1)ans=ans*a%mod;
    return ans;
}
inline void fwt(int* a,int n)
{
    for(int i=1;i<n;i<<=1)
        for(int j=0;j<n;j+=(i<<1))
            for(int k=j;k<j+i;k++)
                a[k+i]=Mod1(a[k+i]+a[k]);
}
inline void ifwt(int* a,int n)
{
    for(int i=1;i<n;i<<=1)
        for(int j=0;j<n;j+=(i<<1))
            for(int k=j;k<j+i;k++)
                a[k+i]=Mod2(a[k+i]-a[k]);
}
int main()
{
    n=read(); m=read();
    for(int i=0;i<m;i++)
        x[i]=read()-1,y[i]=read()-1;
    for(int i=1;i<1<<n;i++)
        cnt[i]=cnt[i>>1]+(i&1);
    for(int i=1;i<1<<n;i++){
        int tot=0;
        for(int j=0;j<m;j++)
            if((i&(1<<x[j]))&&(i&(1<<y[j])))++tot;
        f[cnt[i]][i]=power(2,tot);
    }
    for(int i=1;i<=n;i++)
        fwt(f[i],1<<n);
    for(int i=1;i<=n;i++)
        inv[i]=power(i,mod-2);
    for(int i=1;i<=n;i++){\\求ln
        for(int k=0;k<i;k++)
            for(int j=0;j<1<<n;j++)
                g[i][j]=(g[i][j]+(ll)k*g[k][j]%mod*f[i-k][j])%mod;
        for(int j=0;j<1<<n;j++)
            g[i][j]=(f[i][j]+(ll)(mod-inv[i])*g[i][j])%mod;
    }
    memset(f,0,sizeof(f));
    for(int i=0;i<1<<n;i++)
        f[0][i]=1;
    for(int i=1;i<=n;i++)\\求逆
        for(int k=0;k<i;k++)
            for(int j=0;j<1<<n;j++)
                f[i][j]=(f[i][j]+(ll)f[k][j]*g[i-k][j])%mod;
    ifwt(f[n],1<<n);
    writeln(f[n][(1<<n)-1]);
    return 0;
}

uoj94

原文地址:https://www.cnblogs.com/quzhizhou/p/11311381.html

时间: 2024-07-31 04:26:18

【uoj#94】【集训队互测2015】胡策的统计(集合幂级数)的相关文章

UOJ#191. 【集训队互测2016】Unknown

题意:维护一个数列,每个元素是个二维向量,每次可以在后面加一个元素或者删除一个元素.给定P(x,y),询问对于[l,r]区间内的元素$S_i$,$S_i \times P$的最大值是多少. 首先简单地推出类似斜率优化的式子,那么我们需要在凸包上二分. 学习了一下这份代码http://uoj.ac/submission/69959 使用线段树按下标维护凸包.那么这里有一个问题,如果按照传统的写法,合并一次的复杂度是与$O(区间长度)$的,这样会导致单次插入/删除的时间复杂度变为$O(n)$,是不能

集训队互测2016Unknown(UOJ191)

题目链接 前面部分和lzz的题解是一样的. 首先将输入点(x,y)变为(-y,x)然后,只需找一个向量与(-y,x)的点积最大,即找一个向量在(-y,x)上的投影最长.此时所有的点都是在x轴上方的,容易发现答案一定是在凸包上的,再继续观察,如果有一个点在凸包而不在上凸包上,那么它的右上角及左上角一定有一个点,因此这个点一定不是最优的,所以答案一定在上凸包上,且可以在上凸包上二分. 对于subtask5,使用线段树,每个节点存储这个区间的凸包,合并凸包的话可以将两个凸包上的点归并后线性做凸包. 从

EZ 2018 05 06 NOIP2018 慈溪中学集训队互测(五)

享受爆零的快感 老叶本来是让初三的打的,然后我SB的去凑热闹了 TM的T2写炸了(去你妹的优化),T1连-1的分都忘记判了,T3理所当然的不会 光荣革命啊! T1 思维图论题,CHJ dalao给出了正解但-1输成0了缅怀 而且这题不能用读优玄学 思路也很新奇,先跑一遍MST,判断是否有无解的情况 然后看一下MST中与1相连的边有几条 如果小于k那么我们把所有与1相连的边减上一个值使它们优先被选,然后跑MST 大于k就加上去即可 注意到这个值可以二分,因此不停做MST即可 CODE #inclu

[JZOJ3302] 【集训队互测2013】供电网络

题目 题目大意 给你一个有向图,每个点开始有一定的水量(可能为负数),可以通过边流到其它点. 每条边的流量是有上下界的. 每个点的水量可以增加或减少(从外界补充或泄出到外界),但是需要费用,和增加(减少)流量呈正比例函数关系. 每条边的流量也需要费用,费用和流量呈二次函数关系(常数项为\(0\)). 问将所有水流完的最小花费. 思考历程 这显然是一道上下界最小费用可行流嘛! ==有源有汇的上下界可行流的做法:建立超级源\(ss\)和超级汇\(tt\)(和\(S\).\(T\)),对于\(u\)到

jzoj 2867. 【集训队互测 2012】Contra

Description 偶然间,chnlich 发现了他小时候玩过的一个游戏"魂斗罗",于是决定怀旧.但是这是一个奇怪的魂斗罗 MOD. 有 N 个关卡,初始有 Q 条命. 每通过一个关卡,会得到 u 分和1条命,生命上限为 Q.其中 u=min(最近一次连续通过的关数,R). 若没有通过这个关卡,将会失去1条命,并进入下一个关卡. 当没有生命或没有未挑战过的关卡时,游戏结束,得到的分数为每关得到的分数的总和. 由于 chnlich 好久不玩这个游戏了,每条命通过每个关卡的概率均为p(

[JZOJ2866] 【集训队互测 2012】Bomb

题目 题目大意 给你一个有\(n\)个点的平面. 选择三个点,求两两之间曼哈顿距离和的最大值和最小值. 思考历程&正解 比赛的时候没有想太多,但感觉似乎比较水-- 首先有个很显然的性质,答案为这三个点的最大最小横坐标之差和最大最小纵坐标之差的和. 可以把它看成矩形的周长,容易发现矩形至少一个顶点是三个点之一. 后来才发现水的是求最大值,而不是求最小值. 比赛之后开始和WMY刚-- 最大值是很好求的.我一开始打了个线段树来求,后来发现根本不用-- 求出所有点的\(Xmin,Xmax,Ymin,Ym

jzoj 2866. 【集训队互测 2012】Bomb

Description 给你\(n\)个点,坐标分别为\((xi,yi)\).从中取出三个点,使得其两两间曼哈顿距离和最大和最小,求最大值和最小值. 对于 100% 的数据, N<=100000 , 0<=Xi,Yi<=10^8 Solution 看完题目后感觉要分类讨论,思考1h后果断暴力O(n^3). 但是判断了一下n<=500才跑暴力,得了30分.(?10^9过4s?不判50分) 其实可以O(n^2)暴力的.(听说O(n^2)+优化 = 100) 对于第一个最大值,我们可以\

[集训队互测2012]calc——DP

题面 Bzoj2655 解析 可以强制让$a$数列递增,最后乘以$n!$ 有一个显然的$dp$,$f[i][j]$表示填前$i$个位置,且填的数最大不超过$j$的序列权值和,易有:$f[i][j] = f[i][j-1] + f[i-1][j-1] * j$ $O(AN)$的$dp$显然会$T$ 设$f[i][j]$是关于$j$的$g(i)$次多项式,$g(0)=0$ 注意到$f[i][j] - f[i][j-1]$就是关于$j$的$g(i)-1$次多项式,将$dp$方程移项:$f[i][j]

弱省胡策系列简要题解

现在不是非常爽,感觉智商掉没了,就整理一下最近弱省胡策的题目吧. 其实题目质量还是很高的. 如果实在看不懂官方题解,说不定这里bb的能给您一些帮助呢? [弱省胡策]Round #0 A 20%数据,O(n4)傻逼dp. 40%数据,O(n3)傻逼dp. 100%数据,令f(x1,y1,x2,y2)表示从(x1,y1)走到(x2,y2)的路径条数.于是所有路径就是f(1,2,n?1,m)×f(2,1,n,m?1).然而两条路径可能在中间的某个点相交,我们找出最早的交点,并在这个交点互换两条路径的后