jzoj 100047. 【NOIP2017提高A组模拟7.14】基因变异

题目

Description

21 世纪是生物学的世纪,以遗传与进化为代表的现代生物理论越来越多的 进入了我们的视野。 如同大家所熟知的,基因是遗传因子,它记录了生命的基本构造和性能。 因此生物进化与基因的变异息息相关,考察基因变异的途径对研究生物学有着 至关重要的作用。现在,让我们来看这样一个模型:
1、所有的基因都可以看作一个整数或该整数对应的二进制码;
2、在 1 单位时间内,基因 x 可能会在其某一个二进制位上发生反转;
3、在 1 单位时间内,基因 x 可能会遭到可感染基因库内任一基因 y 的影响 而突变为 x XOR y。
现在给出可感染基因库,Q 组询问,每组给出初始基因与终止基因,请你 分别计算出每种变异最少要花费多少个单位时间。

Input

第 1 行两个整数 N, Q; 第 2 行 N 个用空格隔开的整数分别表示可感染基因库内的基因; 接下来 Q 行每行两个整数 S、T,分别表示初始基因与终止基因。

Output

输出 Q 行,依次表示每组初始基因到终止基因间最少所花时间。


解题思路

  • 对于翻转每一位,既是异或2的次幂。
  • 我们会发现: 假设 z = x   x o r   y z=x\ xor\ y z=x xor y,那么 z z z在二进制下的 1 1 1的个数既是答案。
  • 所以直接枚举所有情况,然后逐个输出。

代码

#include<cstdio>
#include<string>
#define rr register 
using namespace std; 
const int LW=2e6+1;
int n,m,u,v,dis[LW],que[LW],f[41]; bool bz[LW]; 
inline int read()
{
	int p=0; char c=getchar();
	while (!isdigit(c)) c=getchar(); 
	while (isdigit(c)) p=(p<<3)+(p<<1)+c-48,c=getchar(); 
	return p; 
}
int main()
{
	n=read(),m=read(); 
	for (rr int i=1;i<=n;i++) f[i]=read(); 
	for (rr int i=0;i<20;i++) f[++n]=1<<i; 
	int l=que[1]=dis[0]=0,r=bz[0]=1; 
	while (l<r){
		int x=que[l++],y; 
		for (rr int i=1;i<=n;i++)
		 if (!bz[(y=x^f[i])]) dis[y]=dis[x]+1,bz[(que[++r]=y)]=true; 
	}
	for (rr int i=1;i<=m;i++)
	 printf("%d\n",dis[(u=read())^(v=read())]); 
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值