飞行员兄弟(两种方法)

“飞行员兄弟”这个游戏,需要玩家顺利的打开一个拥有16个把手的冰箱。

已知每个把手可以处于以下两种状态之一:打开或关闭。

只有当所有把手都打开时,冰箱才会打开。

把手可以表示为一个4х4的矩阵,您可以改变任何一个位置[i,j]上把手的状态。

但是,这也会使得第i行和第j列上的所有把手的状态也随着改变。

请你求出打开冰箱所需的切换把手的次数最小值是多少。

输入格式
输入一共包含四行,每行包含四个把手的初始状态。

符号“+”表示把手处于闭合状态,而符号“-”表示把手处于打开状态。

至少一个手柄的初始状态是关闭的。

输出格式
第一行输出一个整数N,表示所需的最小切换把手次数。

接下来N行描述切换顺序,每行输入两个整数,代表被切换状态的把手的行号和列号,数字之间用空格隔开。

注意:如果存在多种打开冰箱的方式,则按照优先级整体从上到下,同行从左到右打开。

数据范围
1≤i,j≤4

输入样例:
-+--
----
----
-+--
输出样例:
6
1 1
1 3
1 4
4 1
4 3
4 4
//数组写法
#include<cstring>
#include<iostream>
#include<cstdio> 
#include<vector> 
using namespace std;
typedef pair<int, int> PII;
const int N=10;
char light[N][N];
int get(int x, int y)
{
    return x * 4 + y;
}
void turn(int x,int y){
	for(int i=0;i<4;i++){
			light[x][i]^=1;
			light[i][y]^=1;
		}
		light[x][y]^=1;
	}
vector<PII>res;
void work(){
	for(int k=0;k< 1<<16;k++){ 
		vector<PII>temp;
		char backup[10][10];
		memcpy(backup,light,sizeof light);
		  for(int i=0;i<4;i++){
		    for(int j=0;j<4;j++){
			if(k>>get(i,j)&1){ //i,j对应的位置在k的位置
				turn(i,j);
				temp.push_back({i,j});
			}
		   } 
		  } 
		  bool s=true;
		  for(int i=0;i<4;i++)
		for(int i=0;i<4;i++){
			for(int j=0;j<4;j++){
				if(light[i][j]=='1'){
				 s=false;
			}
		}
	}
	   if(s==true){
	       if(res.empty()||temp.size()<res.size()){
	           res=temp;
	       }
	   }
	   memcpy(light,backup,sizeof backup);
  }
}
int main(){
	int n;
	 for(int i=0;i<4;i++)scanf("%s",light[i]);	
	 for(int i=0;i<4;i++){
	     for(int j=0;j<4;j++){
	         if(light[i][j]=='+')light[i][j]='1';
	         else light[i][j]='0';
	     }
	 }
	 work();
	 cout<<res.size()<<endl;
	 for(int i=0;i<res.size();i++){
	     cout<<res[i].first+1<<' '<<res[i].second+1<<endl;
	 }
	return 0;
}
//位运算写法
#include<cstring>
#include<iostream>
#include<cstdio> 
#include<vector> 
using namespace std;
typedef pair<int, int> PII;
const int N=10;
char light[N][N];
int get(int x, int y)
{
    return x * 4 + y;
}
vector<PII>res;

void work(){
    int state=0;
     for(int i=0;i<4;i++){
        for(int j=0;j<4;j++){
            if(light[i][j]=='1')
              state+=1<<get(i,j);
        }
    }
    int change[4][4]={0};//注意赋初值或者设置成全局变量
    for(int i=0;i<4;i++){
        for(int j=0;j<4;j++){
            for(int k=0;k<4;k++){
                change[i][j]+=1<<get(i,k);
                change[i][j]+=1<<get(k,j);
            }
            change[i][j]-=1<<get(i,j);
        }
    }
	for(int k=0;k< 1<<16;k++){ 
	    int now=state;
		vector<PII>temp;
		  for(int i=0;i<16;i++){
		      int x=i/4,y=i%4;
			if(k>>i&1){ 
		     	int x=i/4,y=i%4;
				now^=change[x][y];
				temp.push_back({x,y});
			}
		   } 
	   if(!now){
	       if(res.empty()||temp.size()<res.size()){
	           res=temp;
	       }
	   }
	}
  }
int main(){
	 for(int i=0;i<4;i++)scanf("%s",light[i]);	
	 for(int i=0;i<4;i++){
	     for(int j=0;j<4;j++){
	         if(light[i][j]=='+')light[i][j]='1';
	         else light[i][j]='0';
	     }
	 }
	 work();
	 cout<<res.size()<<endl;
	 for(int i=0;i<res.size();i++){
	     cout<<res[i].first+1<<' '<<res[i].second+1<<endl;
	 }
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小王子y

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值