5.20 省选模拟赛 T1 图 启发式合并 线段树合并 染色计数问题
LINK:图
在说这道题之前吐槽一下今天的日子 520 = 1+1+4+514. /cy


这道题今天做的非常失败 一点分都没拿到手 关键是今天的T3 把我整个人给搞崩了。
先考虑 如果得到了这么一张图 怎么得到染色的方案数。
发现很难计算 容斥?总方案-2个相同的+3个相同的 我都觉得不太靠谱且复杂度过高。
考虑直接用乘法原理计数 随便从一个点dfs 然后把相邻的点能选择的方案-1.
这样也是错误的 如一个四个点的环(可能不满足题目中的条件类似的 不过也是可以构造出来的。
第一个点贡献为n 第二个点贡献为n-1 第三个点贡献为n-1 第四个点贡献为n-2
容易发现端倪 如果第三个点和第一个点颜色一样 那么第四个点方案就可以为 n-1了。
所以这是不正确的。
进一步的我们发现 关键是和某个点相邻的多个点可能没有边相连 所以 他们颜色可能相同。
结合题目条件 一个编号比周围都小的点一定没有这个限制。
那么从大到小考虑 染色就可以使相邻的点 颜色不同了。
考虑50分的做法 把图建出来 暴力染色 即可。
这里建图 我直接枚举了两个点b c 寻找是否存在一个a 满足 a<b,a<c 且a和b 以及a和c有边相连。
bitset优化一下即可。
const int MAXN=1002;
int n,m;
bitset<MAXN>b[MAXN],s,cc;
int a[MAXN],vis[MAXN],ans=1;
inline int mul(int a,int b){return (ll)a*b%mod;}
inline int add(int a,int b){return a+b>=mod?a+b-mod:a+b;}
inline int mus(int a,int b){return a-b<0?a-b+mod:a-b;}
inline int ksm(int b,int p)
{
	int cnt=1;
	while(p)
	{
		if(p&1)cnt=mul(cnt,b);
		b=mul(b,b);p=p>>1;
	}
	return cnt;
}
inline void dfs(int x)
{
	vis[x]=1;ans=mul(ans,a[x]);
	for(int i=1;i<=n;++i)
	{
		if(!b[x][i])continue;
		if(!vis[i])--a[i];
	}
	for(int i=1;i<=n;++i)
	{
		if(!b[x][i])continue;
		if(!vis[i])dfs(i);
	}
}
int main()
{
	freopen("a.in","r",stdin);
    freopen("a.out","w",stdout);
	get(n);get(m);
	rep(1,m,i)
	{
		int get(x),get(y);
		b[x][y]=b[y][x]=1;
	}
	s.reset();
	rep(2,n,i)
	{
		s[i-1]=1;
		rep(i+1,n,j)
		{
			if(b[i][j])continue;
			cc=b[i]&b[j]&s;
			if(cc.count())b[i][j]=b[j][i]=1;
		}
	}
	rep(1,n,i)a[i]=n;
	//rep(1,n,i){rep(1,n,j)cout<<b[i][j]<<' ';cout<<endl;}
	fep(n,1,i)
	{
		int cnt=0;
		rep(i+1,n,j)if(b[i][j])++cnt;
		ans=mul(ans,n-cnt);
	}
	put(ans);return 0;
}
考虑正解:
此时问题变成了 求每个点和其相连的 编号比其大的个数.
还是考虑题目中的a,b,c 和刚才的暴力一样需要从小到大做。
考虑最小的a 显然可以直接得到答案。
此时容易想到 和a相连的那些点必然也相连。
在从小到大考虑到某个和a相连的点w的时候 w和a能连得集合是相同的。
把a的集合拿过来用即可得到和自己相连的点集。
可以发现这个关系具有传递性 且每次只会传递到一个点上。
所以 由于要支持查询最小值和数量 所以可以利用set 进行启发式合并。nlog^2
当然也可以使用线段树合并来优化复杂度.nlog
值得注意的是 需要去重 所以无法使用可并堆/堆 来做。这里给出线段树合并的code。
const int MAXN=1000010;
int n,m,id;
int a[MAXN],root[MAXN],ans=1;
inline int mul(int a,int b){return (ll)a*b%mod;}
inline int add(int a,int b){return a+b>=mod?a+b-mod:a+b;}
inline int mus(int a,int b){return a-b<0?a-b+mod:a-b;}
inline int ksm(int b,int p)
{
	int cnt=1;
	while(p)
	{
		if(p&1)cnt=mul(cnt,b);
		b=mul(b,b);p=p>>1;
	}
	return cnt;
}
struct wy{int l,r;int sum;}t[MAXN*30];
inline void insert(int &p,int l,int r,int x)
{
	if(!p)p=++id;++sum(p);
	if(l==r)return;
	int mid=(l+r)>>1;
	if(x<=mid)insert(l(p),l,mid,x);
	else insert(r(p),mid+1,r,x);
}
inline int ask(int p,int l,int r)
{
	if(!sum(p))return 0;
	--sum(p);
	if(l==r)return l;
	int mid=(l+r)>>1;
	if(sum(l(p)))return ask(l(p),l,mid);
	return ask(r(p),mid+1,r);
}
inline int merge(int x,int y,int l,int r)
{
	if(!x||!y)return x|y;
	if(l==r)return x;
	int mid=(l+r)>>1;
	l(x)=merge(l(x),l(y),l,mid);
	r(x)=merge(r(x),r(y),mid+1,r);
	sum(x)=sum(l(x))+sum(r(x));
	return x;
}
int main()
{
	freopen("1.in","r",stdin);
    //freopen("a.out","w",stdout);
	get(n);get(m);
	rep(1,m,i)
	{
		int x,y;
		get(x);get(y);
		if(x>y)swap(x,y);
		insert(root[x],1,n,y);
	}
	rep(1,n,i)
	{
		ans=mul(ans,n-sum(root[i]));
		//cout<<sum(root[i])<<endl;
		int ww=ask(root[i],1,n);
		//cout<<sum(root[i])<<endl;
		if(ww)root[ww]=merge(root[ww],root[i],1,n);
	}
	put(ans);
	return 0;
}
5.20 省选模拟赛 T1 图 启发式合并 线段树合并 染色计数问题的更多相关文章
- 2016湖南省赛 I Tree Intersection(线段树合并,树链剖分)
		2016湖南省赛 I Tree Intersection(线段树合并,树链剖分) 传送门:https://ac.nowcoder.com/acm/contest/1112/I 题意: 给你一个n个结点 ... 
- 【洛谷比赛】[LnOI2019]长脖子鹿省选模拟赛 T1 题解
		今天是[LnOI2019]长脖子鹿省选模拟赛的时间,小编表示考的不怎么样,改了半天也只会改第一题,那也先呈上题解吧. T1:P5248 [LnOI2019SP]快速多项式变换(FPT) 一看这题就很手 ... 
- 【2018.06.26NOIP模拟】T1纪念碑square 【线段树】*
		[2018.06.26NOIP模拟]T1纪念碑square 题目描述 2034年,纪念中学决定修建校庆100周年纪念碑,作为杰出校友的你被找了过来,帮校方确定纪念碑的选址. 纪念中学的土地可以看作是一 ... 
- 7.9模拟赛T1图的遍历(dfs)
		图的遍历(dfs) [题目描述] 对于一个有向图G来说,我们存在一个经典的遍历算法,就是DFS (深度优先搜索遍历).将G以1号点为起点进行DFS后,我们可以 得到G的一棵DFS遍历树T.就此,我们可 ... 
- 5.15 省选模拟赛 T1 点分治 FFT
		LINK:5.15 T1 对于60分的暴力 都很水 就不一一赘述了. 由于是询问所有点的这种信息 确实不太会. 想了一下 如果只是询问子树内的话 dsu on tree还是可以做的. 可以自己思考一下 ... 
- NOI 2019 省选模拟赛 T1【JZOJ6082】 染色问题(color) (多项式,数论优化)
		题面 一根长为 n 的无色纸条,每个位置依次编号为 1,2,3,-,n ,m 次操作,第 i 次操作把纸条的一段区间 [l,r] (l <= r , l,r ∈ {1,2,3,-,n})涂成颜色 ... 
- 洛谷[LnOI2019]长脖子鹿省选模拟赛t1 -> 快速多项式变换
		快速多项式 做法:刚拿到此题有点蒙,一开始真没想出来怎么做,于是试着去自己写几个例子. 自己枚举几种情况之后就基本看出来了,其实本题中 n 就是f(m)在m进制下的位数,每项的系数就是f(m)在m进制 ... 
- 5.20 省选模拟赛 求和 组合数的性质 EGF CRT
		LINK:求和 绝妙的一道题目.没做绝对亏了. 对于第一个subtask 考虑直接递推出组合数. 对于第二个subtask 考虑EGF 设两个EGF 都只含偶数项指标且系数为1的那种 一个到n一个到m ... 
- 5.19 省选模拟赛 T1 小B的棋盘 双指针 性质
		LINK:小B的棋盘 考试的时候没有认真的思考 导致没做出来. 容易发现 当k>=n的时候存在无限解 其余都存在有限解 对于30分 容易想到暴力枚举 对称中心 然后 n^2判断. 对于前者 容易 ... 
随机推荐
- 记一次线上服务CPU 100%的处理过程
			告警 正在开会,突然钉钉告警声响个不停,同时市场人员反馈客户在投诉系统登不进了,报504错误.查看钉钉上的告警信息,几台业务服务器节点全部报CPU超过告警阈值,达100%. 赶紧从会上下来,SSH登录 ... 
- 实现new关键字
			一.new做了什么 1.创建了一个全新的对象. 2.这个对象会被执行[[Prototype]](也就是__proto__)链接. 3.生成的新对象会绑定到函数调用的this. 4.通过new创建的每个 ... 
- MySQL入门(alter语法 与 外键)
			MySQL入门(三) 字段的修改.添加.与删除 修改表字段使用alter table语句,谨记! create table tf1( id int primary key auto_increment ... 
- Django适当进阶篇
			本节内容 学员管理系统练习 Django ORM操作进阶 用户认证 Django练习小项目:学员管理系统设计开发 带着项目需求学习是最有趣和效率最高的,今天就来基于下面的需求来继续学习Django 项 ... 
- 深度剖析分布式单点登录框架XXL-SSO
			于2018年初,在github上创建XXL-SSO项目仓库并提交第一个commit,随之进行系统结构设计,UI选型,交互设计-- 于2018年初,在github上创建XXL-SSO项目仓库并提交第一个 ... 
- Android 对接硬件串口篇
			简介:硬件设备有IC卡片,指压测试仪(测试脉搏信号.心率.血压),经过串口获取硬件设备发送的数据. 正文:第一步:获得硬件设备,应用市场下载串口调适软件,测试一下在Android环境下数据是否能正常获 ... 
- 想用@Autowired注入static静态成员?官方不推荐你却还偏要这么做
			生命太短暂,不要去做一些根本没有人想要的东西.本文已被 https://www.yourbatman.cn 收录,里面一并有Spring技术栈.MyBatis.JVM.中间件等小而美的专栏供以免费学习 ... 
- 占个坑   未来学qt的时候专用
			今天看了一个大佬发了一个上位机图片便向大佬问道 ”上位机是用什么软件做的“大佬抛下一句qt ,在业界内很通用,windows和linux通吃,便让我萌生了一个想法,去学qt.虽说上位机时常听到,但是自 ... 
- CCNA - Part11 - 隔离广播域的 VLAN 来了
			之前在对交换机的介绍中,我们知道交换机的作用就是隔离广播域,在不需要跨网段传输时,在同一子网中转发数据包从而进行通信.实现的核心原理就是在交换机中拥有一张 MAC 表,记录了对应终端设备和接口之间的关 ... 
- 水题----根据O出现次数判断分数
			There is an objective test result such as \OOXXOXXOOO". An `O' means a correct answer of a prob ... 
