题意:如果摸到的14张麻将,可以组成4副三张麻将连续或者相同的,以及两个一样的就能获胜。

思路:直接暴力枚举每种可以摸到的牌型,用dfs判断当前拿到的14张牌型能否获胜。

如果搜索时不优化会超时,如果你选择了第cur张牌,那么你下次不必再选择cur以前的牌,因为会造成重复搜索。加上这个剪枝就能过掉。我还自己加了一个剪枝:先排序,然后如果组合到第cur张牌,前面还有牌没有组合,说明不可能组合完,因为每张牌只会和它自身以及比它更大的牌组合。

字符的处理:s、b、c看做0,1,2,那么1s就表示10,5b表示51,就能愉快地转换成数字了。

AC代码:78ms

#include <cstdio>
#include <cmath>
#include <algorithm>
#include <cstring>
#include <utility>
#include <string>
#include <iostream>
#include <map>
#include <set>
#include <vector>
#include <queue>
#include <stack>
using namespace std;
#define eps 1e-10
#define inf 0x3f3f3f3f
#define PI pair<int, int>
typedef long long LL;
const int maxn = 100+5;
char p[] = {'s', 'b', 'c'};
int vis[maxn], cnt[maxn], a[20];
bool dfs(int k, int cur) {
	for(int i = 0; i < cur; ++i) { //剪枝
		if(cnt[a[i]]) return false;
	}
	if(k == 5) return true;
	if(k == 0) {
		for(int i = 0; i < 14; ++i) {
			int x = a[i];
			if(cnt[x] >= 2 && !vis[x]) {
				vis[x] = 1;
				cnt[x] -= 2;
				if(dfs(k+1, cur)) return true;
				cnt[x] += 2;
			}
		}
	}
	else {
		for(int i = cur; i < 14; ++i) {
			int x = a[i];
			//取三个一样的
			if(cnt[x] >= 3) {
				cnt[x] -= 3;
				if(dfs(k+1, i)) return true;
				cnt[x] += 3;
			}
			//取三个连续的
			int y = x / 10, z = x % 10;
			if(y + 2 <= 9) {
				int flag = 1;
				for(int j = 0; j < 3; ++j) {
					if(!cnt[(y+j)*10+z]) {
						flag = 0;
						break;
					}
				}
				if(flag) {
					for(int j = 0; j < 3; ++j) cnt[(y+j)*10+z]--;
					if(dfs(k+1, i)) return true;
					for(int j = 0; j < 3; ++j) cnt[(y+j)*10+z]++;
				}
			}
		}
	}
	return false;
}

int main() {
	char ma[5];
	int T, kase = 1;
	scanf("%d", &T);
	while(T--) {
		memset(cnt, 0, sizeof(cnt));
		for(int i = 0; i < 13; ++i) {
			scanf("%s", ma);
			for(int j = 0; j < 3; ++j) {
				if(ma[1] == p[j]) {
					a[i] = (ma[0]-'0')*10+j;
					cnt[a[i]]++;
				}
			}
		}
		int old[maxn] , b[maxn];
		memcpy(old, cnt, sizeof(cnt));
		memcpy(b, a, sizeof(a));
		printf("Case %d:", kase++);
		int ans = 0;
		for(int i = 0; i <= 2; ++i)
			for(int j = 1; j <= 9; ++j) {
				int x = j * 10 + i;
				if(cnt[x] == 4) continue;
				memset(vis, 0, sizeof(vis));
				memcpy(cnt, old, sizeof(old));
				memcpy(a, b, sizeof(a));
				a[13] = x;
				sort(a, a+14);
				cnt[x]++;
				if(dfs(0, 0)) {
					++ans;
					printf(" %d%c", j, p[i]);
				}
			}
		if(!ans) printf(" None");
		printf("\n");
	}
	return 0;
}

如有不当之处欢迎指出!

HDU - 3391 C - Mahjong的更多相关文章

  1. HDU 5379 Mahjong tree(dfs)

    题目链接:pid=5379">http://acm.hdu.edu.cn/showproblem.php? pid=5379 Problem Description Little su ...

  2. HDU 5379 Mahjong tree(树的遍历&amp;组合数学)

    本文纯属原创,转载请注明出处.谢谢. http://blog.csdn.net/zip_fan 题目传送门:http://acm.hdu.edu.cn/showproblem.php? pid=537 ...

  3. Hdu 5379 Mahjong tree (dfs + 组合数)

    题目链接: Hdu 5379 Mahjong tree 题目描述: 给出一个有n个节点的树,以节点1为根节点.问在满足兄弟节点连续 以及 子树包含节点连续 的条件下,有多少种编号方案给树上的n个点编号 ...

  4. HDU 4431 Mahjong (DFS,暴力枚举,剪枝)

    题意:给定 13 张麻将牌,问你是不是“听”牌,如果是输出“听”哪张. 析:这个题,很明显的暴力,就是在原来的基础上再放上一张牌,看看是不是能胡,想法很简单,也比较好实现,结果就是TLE,一直TLE, ...

  5. 2015多校第7场 HDU 5379 Mahjong tree 构造,DFS

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5379 题意:一颗n个节点n-1条边的树,现在要给每个节点标号(1~n),要求:(1)每一层的兄弟节点的 ...

  6. HDU 4431 Mahjong(枚举+模拟)(2012 Asia Tianjin Regional Contest)

    Problem Description Japanese Mahjong is a four-player game. The game needs four people to sit around ...

  7. HDU 5379——Mahjong tree——————【搜索】

    Mahjong tree Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Tota ...

  8. 2015 Multi-University Training Contest 7 hdu 5379 Mahjong tree

    Mahjong tree Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Tota ...

  9. HDU 4431 Mahjong 模拟

    http://acm.hdu.edu.cn/showproblem.php?pid=4431 不能说是水题了,具体实现还是很恶心的...几乎优化到哭但是DFS(还加了几个剪枝)还是不行...搜索一直T ...

随机推荐

  1. python_缩进_格式化代码

    pycharm如何格式化代码? ctrl + alt + l pycharm如何缩进代码? tab  向右缩进4格 shift + tab 向左缩进4格

  2. python_语法糖_装饰器

    什么是高阶函数? -- 把函数名当做参数传给另外一个函数,在另外一个函数中通过参数调用执行 #!/usr/bin/python3 __author__ = 'beimenchuixue' __blog ...

  3. Python Django连接(听明白了是连接不是创建!)Mysql已存在的数据库

    再声明一次!是连接不是创建!网上的一些人难道连接和创建这俩词都弄不懂就在那里瞎写一些文章! (Python Django连接存在的数据库) Python连接存在的数据库-------MySql 1.首 ...

  4. Navicat查询结果不能修改的原因

    问题: 开发中常使用Navicat查询数据库,并修改数据库中的值.今天发现查询结果为只读,不能修改.一般连表查不能修改我是知道的,但是单表查居然不能修改. 解决方法: 查了下,有说表是只读,也有说是权 ...

  5. 基于 xorm 的服务端框架 XGoServer

    作者:林冠宏 / 指尖下的幽灵 掘金:https://juejin.im/user/587f0dfe128fe100570ce2d8 博客:http://www.cnblogs.com/linguan ...

  6. 【视频编解码·学习笔记】5. NAL Unit 结构分析

    在上篇笔记中通过一个小程序,可以提取NAL Unit所包含的的字节数据.H.264码流中的每一个NAL Unit的作用并不是相同的,而是根据不同的类型起不同的作用.下面将对NAL Unit中的数据进行 ...

  7. SpringBoot整合Redis、ApachSolr和SpringSession

    SpringBoot整合Redis.ApachSolr和SpringSession 一.简介 SpringBoot自从问世以来,以其方便的配置受到了广大开发者的青睐.它提供了各种starter简化很多 ...

  8. oracle学习(一)

    作为一个入门选手,怕忘记,所以所有东西都尽量写下来.(省略oracle11g的安装过程) 一.sqlpuls用sys账户登录 (sqlplus是客户端连上服务器的一个工具) 1.使用cmd控制台登录 ...

  9. 备忘:java在cmd中编译运行

    防止每次用的时候都想不起来 1.进入.java文件所在的路径 (cls清除控制台的命令) 2.javac编译 javac -d . demo1.java javac demo1.java 如所编译的文 ...

  10. spring之p命名空间注入

    <bean id="personId" class="com.itheima.f_xml.c_p.Person" p:pname="禹太璞&qu ...