GDOI2018 小学生图论题 [NTT]
并没有传送门qwq
思路
首先要知道一个结论(或者说是一个套路):一个竞赛图缩点之后必定是一条链。
那么强联通分量的个数,就是这条链的边数+1。
考虑一条边什么时候会出现:当且仅当点集可以被分成\(S,T\)两个集合且他们之间的边全都是\(S\rightarrow T\)的。
那么\(m=0\)时可以枚举点集大小,那么答案就是
\]
\(m>0\)时仍然考虑枚举点集大小,此时每条链上的点要么全都在一边,要么被分成了两半,相当于做了个背包,可以用NTT优化。
但分成两半时中间的那条边已经固定了下来,没有必要算上它的\(\frac 1 2\)。
考虑把这个\(\frac 1 2\)抵消掉,那么可以把它的系数置为2。
也就是说,对于一个\(k\)个点的链,它的多项式是\(1+2x+2x^2+\cdots+2x^{k-1}+x^k\)。
最后把所有链都乘起来变成\(A(x)\),那么答案就是
\]
代码
#include<bits/stdc++.h>
clock_t t=clock();
namespace my_std{
	using namespace std;
	#define pii pair<int,int>
	#define fir first
	#define sec second
	#define MP make_pair
	#define rep(i,x,y) for (int i=(x);i<=(y);i++)
	#define drep(i,x,y) for (int i=(x);i>=(y);i--)
	#define go(x) for (int i=head[x];i;i=edge[i].nxt)
	#define templ template<typename T>
	#define sz 401010
	#define mod 998244353ll
	typedef long long ll;
	typedef double db;
	mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
	templ inline T rnd(T l,T r) {return uniform_int_distribution<T>(l,r)(rng);}
	templ inline bool chkmax(T &x,T y){return x<y?x=y,1:0;}
	templ inline bool chkmin(T &x,T y){return x>y?x=y,1:0;}
	templ inline void read(T& t)
	{
		t=0;char f=0,ch=getchar();double d=0.1;
		while(ch>'9'||ch<'0') f|=(ch=='-'),ch=getchar();
		while(ch<='9'&&ch>='0') t=t*10+ch-48,ch=getchar();
		if(ch=='.'){ch=getchar();while(ch<='9'&&ch>='0') t+=d*(ch^48),d*=0.1,ch=getchar();}
		t=(f?-t:t);
	}
	template<typename T,typename... Args>inline void read(T& t,Args&... args){read(t); read(args...);}
	char __sr[1<<21],__z[20];int __C=-1,__zz=0;
	inline void Ot(){fwrite(__sr,1,__C+1,stdout),__C=-1;}
	inline void print(register int x)
	{
		if(__C>1<<20)Ot();if(x<0)__sr[++__C]='-',x=-x;
		while(__z[++__zz]=x%10+48,x/=10);
		while(__sr[++__C]=__z[__zz],--__zz);__sr[++__C]='\n';
	}
	void file()
	{
		#ifndef ONLINE_JUDGE
		freopen("a.in","r",stdin);
		#endif
	}
	inline void chktime()
	{
		#ifndef ONLINE_JUDGE
		cout<<(clock()-t)/1000.0<<'\n';
		#endif
	}
	#ifdef mod
	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;}
	ll inv(ll x){return ksm(x,mod-2);}
	#else
	ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x) if (y&1) ret=ret*x;return ret;}
	#endif
//	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;}
}
using namespace my_std;
int n,_m,m;
int cc[sz];
ll _tmp[sz];
ll *a[sz];
int r[sz],limit;
void NTT_init(int n)
{
	int l=-1;limit=1;
	for (;limit<=n;limit<<=1) ++l;
	rep(i,0,limit-1) r[i]=(r[i>>1]>>1)|((i&1)<<l);
}
void NTT(ll *a,int type)
{
	rep(i,0,limit-1) if (i<r[i]) swap(a[i],a[r[i]]);
	for (int mid=1;mid<limit;mid<<=1)
	{
		ll Wn=ksm(3,(mod-1)/mid>>1);if (type==-1) Wn=inv(Wn);
		for (int len=mid<<1,j=0;j<limit;j+=len)
		{
			ll w=1;
			for (int k=0;k<mid;k++,w=w*Wn%mod)
			{
				ll x=a[j+k],y=w*a[j+k+mid]%mod;
				a[j+k]=(x+y)%mod;a[j+k+mid]=(x-y+mod)%mod;
			}
		}
	}
	if (type==1) return;
	ll I=inv(limit);
	rep(i,0,limit-1) a[i]=a[i]*I%mod;
}
ll tmp[60][sz],len[60];
int st[60],top;
int NTT(int l,int r)
{
	if (l==r)
	{
		int ret=st[top--];
		rep(i,0,cc[l]) tmp[ret][i]=a[l][i];
		len[ret]=cc[l];
		return ret;
	}
	int mid=(l+r)>>1;
	int L=NTT(l,mid),R=NTT(mid+1,r);
	int cur=st[top--];
	NTT_init(len[L]+len[R]+2);
	NTT(tmp[L],1);NTT(tmp[R],1);
	rep(i,0,limit-1) tmp[cur][i]=tmp[L][i]*tmp[R][i]%mod;
	NTT(tmp[cur],-1);
	len[cur]=len[L]+len[R];
	rep(i,0,limit-1) tmp[L][i]=tmp[R][i]=0;
	st[++top]=L;st[++top]=R;
	return cur;
}
int main()
{
	file();
	read(n,_m);
	int m=n,cur=0;
	rep(i,1,_m) { read(cc[i]); m-=cc[i]-1; rep(j,1,cc[i]) read(cc[0]); }
	rep(i,_m+1,m) cc[i]=1;
	rep(i,1,m)
	{
		a[i]=_tmp+cur;
		a[i][0]=a[i][cc[i]]=1;
		rep(j,1,cc[i]-1) a[i][j]=2;
		cur+=cc[i]+1;
	}
	rep(i,1,50) st[i]=i;
	top=50;
	int p=NTT(1,m);
	ll ans=1;
	rep(i,1,n-1) (ans+=tmp[p][i]*inv(ksm(2,1ll*i*(n-i)%(mod-1)))%mod)%=mod;
	cout<<ans;
	return 0;
}
GDOI2018 小学生图论题 [NTT]的更多相关文章
- [gdoi2018 day1]小学生图论题【分治NTT】
		正题 题目大意 一张随机的\(n\)个点的竞赛图,给出它的\(m\)条相互无交简单路径,求这张竞赛图的期望强联通分量个数. \(1\leq n,m\leq 10^5\) 解题思路 先考虑\(m=0\) ... 
- 【XSY2887】【GDOI2018】小学生图论题 分治FFT 多项式exp
		题目描述 在一个 \(n\) 个点的有向图中,编号从 \(1\) 到 \(n\),任意两个点之间都有且仅有一条有向边.现在已知一些单向的简单路径(路径上任意两点各不相同),例如 \(2\to 4\to ... 
- csu  1978: LXX的图论题
		1978: LXX的图论题 Submit Page Summary Time Limit: 1 Sec Memory Limit: 128 Mb Submitted: 71 ... 
- cojs QAQ的图论题 题解报告
		话说这个题目应该叫做 斯特林数的逆袭 QAQ 先说一说部分分的算法 1.n<=5 直接暴力搜索就可以了 2.k=0的时候不难发现任意一张图的价值都是n,问题转化为计算有多少种图,显然是2^C(n ... 
- 【HDOJ图论题集】【转】
		=============================以下是最小生成树+并查集====================================== [HDU] How Many Table ... 
- HDU 6343 - Problem L. Graph Theory Homework - [(伪装成图论题的)简单数学题]
		题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6343 Time Limit: 2000/1000 MS (Java/Others) Memory Li ... 
- 图论常用算法之一  POJ图论题集【转载】
		POJ图论分类[转] 一个很不错的图论分类,非常感谢原版的作者!!!在这里分享给大家,爱好图论的ACMer不寂寞了... (很抱歉没有找到此题集整理的原创作者,感谢知情的朋友给个原创链接) POJ:h ... 
- HDU图论题单
		=============================以下是最小生成树+并查集====================================== [HDU] 1213 How Many ... 
- BZOJ2118: 墨墨的等式(同余类BFS)(数学转为图论题)
		2118: 墨墨的等式 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 2944 Solved: 1206[Submit][Status][Discu ... 
随机推荐
- Idea创建一个SpringBoot工程
			1.打开Idea,点击新建工程 File—New—Project 2.点击下一步后可能会很一直在请求,或者直接报如下错误, 解决办法:直接点OK后再点Previous返回上一步,继续重新Next 3. ... 
- Linux基础-7.Linux网络基础设置
			装好Linux,Linux一般会默认使用NetworkManager来辅助管理网络配置.对于配置Linux服务器来说,我们不需要NetworkManager来配置网络,所以要关闭它,不然它会影响手动配 ... 
- api封装
			const sql={ insert: function(collection,insertData){ return new Promise(function(resolve,reject){ co ... 
- JavaWeb 之 MVC 开发模式
			MVC 开发模式 一.JSP 演变历史 1. 早期只有servlet,只能使用response输出标签数据,非常麻烦 2. 后来又jsp,简化了Servlet的开发,如果过度使用jsp,在jsp中即写 ... 
- python之路第一天
			2018-07-11星期三 创建自己的博客(博客园): 登陆 我的博客 随笔:所有人在博客中都能看见的文章 文章:别人看不见,只能URL访问--我把网页地址发给你,你才能看到 日志:别人看不到,URL ... 
- Linux 里的 2>&1 究竟是什么
			原文 我们在Linux下经常会碰到nohup command>/dev/null 2>&1 &这样形式的命令.首先我们把这条命令大概分解下: 首先就是一个nohup:表示当 ... 
- ELK Stack部署
			部署ELK Stack 官网:https://www.elastic.co 环境准备: ip hostname 服务 用户.组 192.168.20.3 node2003 kibana6.5,file ... 
- Python 软件安装
			安装Python解释器 Python目前已支持所有主流操作系统,在Linux,Unix,Mac系统上自带Python环境,在Windows系统上需要安装一下,超简单 打开官网https://www.p ... 
- Python_类的继承与方法重写
			1.新建子类时,括号内要传入继承的父类名 2.super()方法:自动寻找当前类的父类,并调用父类的构造函数,初始化属性值 class Cup: #构造函数,初始化属性值 def __init__(s ... 
- MySQL/MariaDB数据库的触发器
			MySQL/MariaDB数据库的触发器 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.触发器概述 1>.什么是触发器 触发器的执行不是由程序调用,也不是由手工启动,而是 ... 
