[Educational Codeforces Round 6][Codeforces 620C]Pearls in a Row
[Codeforces Round #248][Codeforces 434D] Nanami's Power Plant

[Testing Round #13][Codeforces 753C] Interactive Bulls and Cows (Hard)

Zarxdy34 posted @ 2017年12月07日 15:45 in Codeforces with tags 搜索 , 418 阅读

    题意:Bull and Cows游戏。猜一个四位数码(保证数码各位互不相同),每次交互时选手给电脑一个四位数码,电脑会返回两个值a和b,a表示同一位置上数码相同的位置个数,b表示电脑与选手的四位数码共同包含但是位置不同的数码个数。要求7次内猜出数字。

    题解:这题要求出完美解状态数量过多,不过贪心解也能在7次之内完成。设当前可选数的集合为S,现在要找到一个数字t,使得通过该返回值排除掉的可选数在最坏情况下最多。易见第一次无论选什么都是等价的,所以可以考虑第一次选0123;第二次选取时,对每个a,b都事先考虑好选什么数去猜(否则容易超时),这个可以预先打表实现。然后就决策树继续做就好了。

 

#include <bits/stdc++.h>
using namespace std;

void Get_Num(int x,int &x0,int &x1,int &x2,int &x3)
{
	x0=x%10;
	x1=x%100/10;
	x2=x%1000/100;
	x3=x/1000;
}

void Compare(int x,int y,int &b,int &c)
{
	b=0,c=0;
	int num[2][4];
	Get_Num(x,num[0][0],num[0][1],num[0][2],num[0][3]);
	Get_Num(y,num[1][0],num[1][1],num[1][2],num[1][3]);
	for (int i=0;i<4;i++) if (num[0][i]==num[1][i]) b++;
	sort(num[0],num[0]+4);
	sort(num[1],num[1]+4);
	for (int i=0,j=0;i<4;i++)
	{
		while (j<4 && num[0][i]>num[1][j]) j++;
		if (j<4 && num[0][i]==num[1][j]) j++,c++;
	}
	c-=b;
}

int a[2][10000],num[2];

void Fliter(int o,int x,int b,int c)
{
	num[o^1]=0;
	for (int i=0;i<num[o];i++)
	{
		int b0,c0;
		Compare(x,a[o][i],b0,c0);
		if (b0==b && c0==c) a[o^1][num[o^1]++]=a[o][i];
	}
}

int Val(int o,int x)
{
	int cnt[25];
	memset(cnt,0,sizeof(cnt));
	for (int i=0;i<num[o];i++)
	{
		int b,c;
		Compare(x,a[o][i],b,c);
		cnt[b*4+c]++;
	}
	return *max_element(cnt,cnt+25);
}

int main()
{
	for (int i=123;i<10000;i++)
	{
		int x0,x1,x2,x3;
		Get_Num(i,x0,x1,x2,x3);
		if (x0==x1 || x0==x2 || x0==x3 || x1==x2 || x1==x3 || x2==x3) continue;
		a[0][num[0]++]=i;
	}
	int last=123;
	printf("0123\n");
	fflush(stdout);
	for (int t=0;t<7;t++)
	{
		int b,c;
		scanf("%d%d",&b,&c);
		if (b==4) return 0;
		Fliter(t&1,last,b,c);
		if (t==0)
		{
			if (b==0 && c==0) last=4567;
			if (b==1 && c==0) last=456;
			if (b==0 && c==1) last=456;
			if (b==2 && c==0) last=145;
			if (b==1 && c==1) last=145;
			if (b==0 && c==2) last=1045;
			if (b+c==3) last=124;
			if (b+c==4) last=1230;
		}
		else
		{
			int minv=10000,o=(t&1)^1,tmp;
			for (int i=0;i<num[o];i++)
				if ((tmp=Val(o,a[o][i]))<minv) minv=tmp,last=a[o][i];
		}
		printf("%04d\n",last);
		fflush(stdout);
	}
	return 0;
}

登录 *


loading captcha image...
(输入验证码)
or Ctrl+Enter