2021 Shandong Provincial Collegiate Programming Contest C - Cat Virus

题意

构造一个树使该树的情况数为k种,树上的节点可以是黑色或是白色;当一个点为黑色时,它的所有子节点全为黑色;如果为白色,子节点可为黑色和白色。输出构造出来的树的顶点数以及树的边。(树的顶点数不能超过1e5) (2<=k<=2e18)

1为点是黑色,0为点是白色. 

 

 思路:

一开始,通过画图发现就只有一条链的时候,每次新加一个节点他的情况数就可以增加1(图1),然后就想着单独的一个链,但是很显然会超过1e5.这个思路就被否定了。
于是就开始看题解——

(1)通过画图分析可以发现当一个根节点再连接一个点的时候,会发现他的方案数是上面所有的方案数+1;(图1) 

(2)当他连接两个节点时,新的两个各有两种情况,于是处理的时候就让一边先固定另一边继续向下处理(图2)两个节点互不干扰,情况数就是2*2+1;
以此类推会发现在中间的点也是是一样的 
因此父节点的情况就是它的子节点的所有情况相乘+1;//+1是父节点为黑的时候只有一种情况; 

正确思路:
      1.如果当前剩下的种类数为奇数的时候,1是为黑的啥时候的特殊情况;当前要连接点的父节点应该连接两个点,分成两个部分此时左节点有2种情况,此时剩下的就相当于原本剩下的数除以2;
      2.如果为偶数的时候,通过图一分析,单独加一个点的时候,会使得剩下的情况数减去1,使其剩下的数变成奇数;
      3.如果剩下的数为2的时候,就不再进行,因为他是最后的子节点,本身一定会有2种情况;
      4.如果剩下的数为3的时候,就在连接一个点使剩下的为3-1,为2.d

代码实现:

#include<iostream>
#include<stdio.h>
#include<algorithm>
using namespace std;
#define int long long 
int k,n;
pair<int,int>v[100005];
signed main()
{
	scanf("%lld",&k);
	int cnt=0;//边数;
	n=1;//顶点数;
	while(k)
	{
		if(k==2)
		break;
		if(k%2==0||k==3)
		{
			v[++cnt]={n,n+1};
			n+=1;//作为新的根节点;
			k-=1;//情况数减少1;
			continue; 
		}
		if(k&1)
		{
			v[++cnt]={n,n+1};
			v[++cnt]={n,n+2};//相当于n连接两个点
			n+=2;//以右边的为新的根节点
			k/=2;//相当于情况数减少一半 
		}
	}
	printf("%lld\n",n);
	for(int i=1;i<=cnt;i++)
	printf("%lld %lld\n",v[i].first,v[i].second);
	return 0; 
}

  • 4
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值