模拟、枚举与贪心3

链接:登录—专业IT笔试面试备考平台_牛客网
来源:牛客网
 

Flip Game

有一4*4的灯泡,每次按某一个可以使得其本身以及其上下左右共五个灯的开关反向,给定初始状态,问能否把灯都灭掉。

推论:

1、当按2遍同一灯泡时,相当于还原初始状态,所以只需考虑按一次的情况。

2、易得,当操作第三行灯泡时,要想达到要求必须保证第一行所有灯泡都灭了,因为第三行灯泡不能对第一行产生影响,后方的行以此类推。

综上,只要知道第一行的按法,后面的可以推出。而第一行不妨使用暴力枚举,依次尝试。

此处引入位运算异或(^),由1^1=0,1^0=1,0^0=0,0^1=1可知:对于1或0来说,异或1结果为原数取反,异或0为原数。例:当用1表示灯亮或按下,0表示灯灭或不按,第一行灯的情况为1100、按键情况为1010时,灯亮灭的结果为1100^1010^0101(1010右移,因为会影响右边的灯)^0100(1010左移,会影响左边的灯)。另外,位运算时一定注意优先级!这里直接劝加括号。

写代码的细节:scanf(" %c",&c); %前的空格是为了防止读入换行(忽略掉空格、回车和tab键)。

#include<iostream>
#include<cstdio> 
#include<queue>
#include<cstring>
#include<algorithm>
using namespace std;
const int m=4, n=4;
int a[m+10], b[m+10]; //a:黑的1 b:白的1 有可能翻成全1或0 
int change[n+10]; 
int cal(int x){
	int cnt = 0;
	while(x){
//		cout<<0;
		if(x & 1) cnt++;
		x >>= 1;
	}
	return cnt;
}
int deal(int a[]){
	int cur[n + 10]={0}, cnt, ans = 0x7f7f7f7f; //存操作后的a 
	for(change[0] = 0; change[0] <= (1 << m)-1; ++ change[0]){
//		cout<<1;
		cnt = cal(change[0]); // 
		cur[0] = a[0] ^ change[0] ^ (change[0] >> 1) ^ ((change[0] << 1) & ((1 << m) - 1)); //当前第0行/// 
		cur[1] = a[1] ^ change[0]; //下面一行灯的状态 
		for(int i = 1; i < n; ++ i){
//			cout<<2;
			change[i] = cur[i-1]; //怎样按取决于上一行的灯怎么亮的;
			cnt += cal(change[i]); 
			cur[i] = cur[i] ^ change[i] ^ (change[i] >> 1) ^ ((change[i] << 1) & ((1 << m) - 1)); // 所以由上述,此行状态也确定了 
			cur[i+1] = a[i+1] ^ change[i]; //下面一行灯的状态 
		} 
		if(cur[n-1] == 0) { //若为0,说明全按灭了,计算 按了多少下 
//			cout<<cnt<<endl;
			ans = min(ans, cnt); 
		}
	}
	return ans;
}
int main(){
	for(int i = 0; i < 4; ++ i)
		for(int j = 0; j < 4; ++ j){
			char c;
			scanf(" %c", &c);//读入字符
			if(c == 'b') a[i] |= (1 << j); //a[i]的第j位改为1 
			else  b[i] |= (1 << j);  
		}
	int ans = min(deal(a), deal(b));
	if(ans > n*m) cout<<"Impossible" << endl;
	else cout << ans; 
} 

 拓展:

1、若为N*M的灯泡(N<=10 M<=100),则由于需要枚举可能超时,故不可用行枚举(2^100*10),但可以枚举列(2^10*100)。

2、若有a、b两数,a异或两次b结果仍为a。

点名

已知n个同学的学号,现在有一场活动,来了n-1个同学,每个同学都把自己的学号写下来了,告诉n-1个同学的学号,问哪个同学没来。

当N<=10^5,学号为5位数时,可直接标记。

当N<=10^7,学号为9位数时,可利用减法,减去已知同学的学号数即可得到,需开long long。

当N<=10^7,学号为17位数时,可通过异或法,由上题拓展2中所述性质,可以不超二进制范围地得到没来的同学的学号。

若在上一个条件的基础上,有两个同学没来呢?由于两个同学学号至少有一位不相同,故可以通过在得到所有来的同学的异或结果后,通过查找二进制为1的第一个位置,将已到的同学分为2批,这样又回到上一问的问题。

Summer Earings

N个点,任选其中3个为圆心画三个半径相同的圆,圆不能相交,求能画出的最大的半径为多少。N<=3000。

根据题意,由于三个圆心中最大半径只能是三角形最短边的一半,所以转化为了求三个点构成的三角形的最小边长。若暴力循环去枚举三个点并寻找最短边,时间复杂度过高。可以将所有边算出来存储排序,从大到小枚举边,每个顶点利用位串来记录与哪个点可以构成三角形,第一个构成了三角形的即满足条件。这个超长的位串可以用bitset类型进行存储

#include<iostream>
#include<cstdio> 
#include<queue>
#include<cstring>
#include<algorithm>
#include<bitset>
#include<cmath>
using namespace std;
typedef long long ll;
const ll mod  = 1000000007;
const ll maxx = 1005;
const ll inf  = 20000000005;
int m,n;
int x[4000], y[4000];
struct ty{
	int len;//为了防止误差的产生,储存距离的平方。 
	int x,y;//不要总用longlong,有时会超时,应该尽量算出来 
}a[3000*3000];
bitset<4000>b[4000];
int dis(int i,int j){
	return (x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j]);
} 
bool cmp(ty a, ty b){
	return a.len>b.len;
}
int main(){
	scanf("%d",&n);
	for(int i=1;i<=n;++i)
		scanf("%d%d",&x[i],&y[i]);
	for(int i=1;i<=n;++i){
		for(int j=1;j<=n;++j){
			a[++m].len=dis(i,j);
			a[m].x=i; a[m].y=j;
		}
	}
	sort(a+1,a+1+n,cmp); 
	for(int i=1;i<=m;++i){
		if((b[a[i].x]&b[a[i].y])!=0){
			printf("%.8f",sqrt(a[i].len)/2.0);
			break;
		}else{
			b[a[i].x][a[i].y]=1;
			b[a[i].y][a[i].x]=1;
		}
	}
	return 0;
}

链接:登录—专业IT笔试面试备考平台_牛客网
来源:牛客网
 

月月查华华的手机

华华想与某个小姐姐搭讪,当且仅当小姐姐的昵称是他的昵称的子序列。为了方便,华华和小姐姐的昵称只由小写字母构成。为了更加方便,保证小姐姐的昵称长度不会比华华的长。
现在月月要快速的判断出哪些推荐好友要删掉,因为华华快回来了,时间紧迫,月月有点手忙脚乱,所以你赶紧写个程序帮帮她吧!数据范围如下:

A为华华昵称,N为小姐姐昵称个数,Bi为第i位小姐姐昵称长度。

如果暴力挨个搜索,时间复杂度过高,如果能确认当前字符在华华昵称中的后续位置不存在就好办了。由此可以设置一个nxt[i][j]数组存取i字符后的最近的j字符的位置,last[i]数组存字符i最近的位置,问题即可迎刃而解。

#include<iostream>
#include<cstdio> 
#include<queue>
#include<cstring>
#include<algorithm>
#include<bitset>
#include<cmath>
using namespace std;
typedef long long ll;
const ll mod  = 1000000007;
const ll maxx = 1005;
const ll inf  = 20000000005;
string s;
int n,nxt[1001000][40],last[40];
//nxt:在华华串的第i个元素下一个同样字符在哪里 
//last:最近的字符的位置 
int main(){
	cin>>s;
	int len=s.length();
	memset(last,-1,sizeof(last));
	for(int i=len-1;i>=0;--i){
		for(int j=0;j<26;++j){
			nxt[i][j]=last[j];
		}
		last[s[i]-'a']=i;
	}
	cin>>n;
	for(int i=1;i<=n;++i){
		int flag=1;
		string s1;
		cin>>s1;
		int pos=last[s1[0]-'a'];
		for(int j=1;j<s1.length();++j){
			pos=nxt[pos][s1[j]-'a'];
			if(pos==-1) {
				cout<<"No"<<endl;
				flag=0; break;
			}
		}
		if(flag) cout<<"Yes\n";
	} 
	return 0;
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值