【原题】

1860: [Zjoi2006]Mahjong麻将

Time Limit: 1 Sec  Memory Limit: 64 MB

Submit: 211  Solved: 122

[Submit][Status]

Description

Input

第一行一个整数N(N<=100),表示玩了N次超级麻将。 接下来N行。每行100个数a1..a100,描写叙述每次玩牌手中各种牌的数量。ai表示数字为i的牌有ai张。(0<=ai<=100)

Output

输出N行,若胡了则输出Yes,否则输出No,注意区分Yes。No的大写和小写!

Sample Input

3

2 4 0 0 0 0 0 …… 0(一共98个0)

2 4 2 0 0 0 0 …… 0(一共97个0)

2 3 2 0 0 0 0 …… 0(一共97个0)

Sample Output

Yes

Yes

No

HINT

Source

【分析1】原来记得也有一道麻将的判和问题(当然不是国家集训队的那道)。

当时随便用贪心使过去了。可是这里3张牌和4张牌结合在了一起,于是贪心肯定挂了。然后就百思不得其解。

瞄了一眼题解:原来是爆搜。然后開始自己YY。由于最多是三张连续的牌,所以假设1--K-1张牌都没了,第K张牌一定仅仅会影响到K+1和K+2。

然后我就把K去爆搜。

显然光是这样肯定T了。

假设我把1--K-1以多种方法所有消掉了,第K种的状态就仅仅需搜一遍就可以。因此我依据K,K+1,K+2眼下的数量以及K的位置HASH一下。

PS:用了凌神的64位自然溢出。

【代码1】

#include<cstdio>
#include<set>
using namespace std;
typedef unsigned long long ULL;
set<ULL>S;ULL base[101],a[101],mul=97;
inline ULL calc(int k) {return base[k]*a[k]+base[k+1]*a[k+1]+base[k+2]*a[k+2];}
int Test,i;
inline bool dfs(int k,bool two)
{
ULL t=calc(k);
if (S.find(t)!=S.end())
return 0;
S.insert(t);
while (!a[k]) k++;
if (k==101) return two;
if (a[k]&&a[k+1]&&a[k+2])
{
a[k]--;a[k+1]--;a[k+2]--;
if (dfs(k,two)) return 1;
a[k]++;a[k+1]++;a[k+2]++;
}
if (a[k]>=4)
{
a[k]-=4;if (dfs(k,two)) return 1;a[k]+=4;
}
if (a[k]>=3)
{
a[k]-=3;if (dfs(k,two)) return 1;a[k]+=3;
}
if (a[k]>=2&&!two)
{
a[k]-=2;if (dfs(k,1)) return 1;a[k]+=2;
}
return 0;
}
int main()
{
base[1]=1;for (i=2;i<=100;i++) base[i]=base[i-1]*mul;
scanf("%d",&Test);
while (Test--)
{
for (i=1;i<=100;i++) scanf("%lld",&a[i]);
S.clear();
if (dfs(1,0)) puts("Yes");else puts("No");
}
return 0;
}

【分析2】上述做法尽管例子过了。可是狂WA不止。后来发现这种状态数太多了,HASH肯定会被卡掉。在WCY大神的指导下。我发现仅仅需把K+1~N的情况哈希一下,这样子状态数显然变小了。

【代码】

#include<cstdio>
#include<set>
using namespace std;
typedef unsigned long long ULL;
set<ULL>S;ULL base[105],a[105],sum,mul=131ll;
int Test,i;
inline bool dfs(int k,bool two,ULL status)
{
if (S.find(status)!=S.end()) return 0;
S.insert(status);
while (!a[k]&&k<=100) k++;
if (k==101) return two;
if (a[k]&&a[k+1]&&a[k+2]&&k<=98)
{
a[k]--;a[k+1]--;a[k+2]--;
if (dfs(k,two,status-base[k]-base[k+1]-base[k+2])) return 1;
a[k]++;a[k+1]++;a[k+2]++;
}
if (a[k]>=4)
{
a[k]-=4;if (dfs(k,two,status-base[k]*4)) return 1;a[k]+=4;
}
if (a[k]>=3)
{
a[k]-=3;if (dfs(k,two,status-base[k]*3)) return 1;a[k]+=3;
}
if (a[k]>=2&&!two)
{
a[k]-=2;if (dfs(k,1,status-base[k]*2-base[100])) return 1;a[k]+=2;
}
return 0;
}
int main()
{
base[1]=1ll;for (i=2;i<=100;i++) base[i]=base[i-1]*mul;
scanf("%d",&Test);
while (Test--)
{
sum=0;for (i=1;i<=100;i++) scanf("%lld",&a[i]),sum+=base[i]*a[i];
S.clear();
if (dfs(1,0,sum)) puts("Yes");else puts("No");
}
return 0;
}

bzoj 1860: [Zjoi2006]Mahjong麻将 题解的更多相关文章

  1. [BZOJ1860][ZJOI2006]Mahjong(DP)

    1860: [Zjoi2006]Mahjong麻将 Time Limit: 1 Sec  Memory Limit: 64 MBSubmit: 412  Solved: 248[Submit][Sta ...

  2. BZOJ 4033: [HAOI2015]树上染色题解

    BZOJ 4033: [HAOI2015]树上染色题解(树形dp) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1327400 原题地址: BZOJ 403 ...

  3. BZOJ 1861: [Zjoi2006]Book 书架 splay

    1861: [Zjoi2006]Book 书架 Description 小T有一个很大的书柜.这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列.她用1到n的正整数给每本书都编了号. 小T在看书 ...

  4. 【DP/二分】BZOJ 1863:[Zjoi2006]trouble 皇帝的烦恼

    863: [Zjoi2006]trouble 皇帝的烦恼 Time Limit: 1 Sec  Memory Limit: 64 MBSubmit: 465  Solved: 240[Submit][ ...

  5. BZOJ 1003 [ZJOI2006]物流运输trans

    1003: [ZJOI2006]物流运输trans Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4242  Solved: 1765[Submit] ...

  6. [ZJOI2006]超级麻将

    题目描述 很多人都知道玩麻将,当然也有人不知道,呵呵,不要紧,我在这里简要地介绍一下麻将规则: 普通麻将有砣.索.万三种类型的牌,每种牌有1~9个数字,其中相同的牌每个有四张,例如1砣~9砣,1索~9 ...

  7. bzoj 1003 [ZJOI2006]物流运输(最短路+dp)

    [ZJOI2006]物流运输 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 8973  Solved: 3839[Submit][Status][Di ...

  8. BZOJ 1003[ZJOI2006]物流运输(SPFA+DP)

    Problem 1003. -- [ZJOI2006]物流运输 1003: [ZJOI2006]物流运输 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: ...

  9. BZOJ 1862: [Zjoi2006]GameZ游戏排名系统 [treap hash]

    1862: [Zjoi2006]GameZ游戏排名系统 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1318  Solved: 498[Submit][ ...

随机推荐

  1. [Javascript] Prototype 2 Object.create()

    function Fencepost (x, y, postNum){ this.x = x; this.y = y; this.postNum = postNum; this.connections ...

  2. rapidxml 序列化

    void TestRapidXml() { ]; sprintf(xmlContent,"<root><head>aaa</head><body&g ...

  3. ArcGIS中的批量处理

    在实际生产过程中,经常遇到批量处理数据的情况.在ArcGIS中,除自己写代码来处理这类问题外,它提供了一个批量处理的工具,在ToolBox对应的工具上右键即可选择批处理工具. 和单个处理方式一样,输入 ...

  4. listView下拉刷新加载数据

    这个下拉效果在网上最早的例子恐怕就是Johan Nilsson的实现,http://johannilsson.com/2011/03/13/android-pull-to-refresh-update ...

  5. android:View的setTag和getTag使用

    1.用于区分非常多类似的View 比如: button1.setOnClickListener(new OnClickListener ... ); button2.setOnClickListene ...

  6. jsp中的js嵌入Extjs与后台action交互

    近期做前台须要和后台交互数据,直接使用js一直没实现.最后使用extjs实现了,extjs代码直接嵌入到jsp的js代码中就可以(0跟jsp里使用extjs一样,须要载入extjs的几个文件) < ...

  7. linux的fork()函数-进程控制

    进程作为构成系统的基本细胞,不仅是系统中独立活动的实体,而且是独立竞争资源的基本实体.它要经历创建.执行.等待.终止等一系列过程. 一.fork入门知识(转载) 一个进程,包括代码.数据和分配给进程的 ...

  8. MySQL优化小案例:key_buffer_size

    key_buffer_size是对MyISAM表性能影响最大的一个参数,下面一台以MyISAM为主要存储引擎服务器的配置: mysql> SHOW VARIABLES LIKE '%key_bu ...

  9. 通过jdbc使用PreparedStatement,提升性能,防止sql注入

    为什么要使用PreparedStatement? 一.通过PreparedStatement提升性能 Statement主要用于执行静态SQL语句,即内容固定不变的SQL语句.Statement每执行 ...

  10. LR函数基础(二)

    (一)用到的函数: (1) web_set_option()   //重定向设置 (2)web_reg_save_param和custom_request都常于处理参数的动态生成. web_reg_s ...