UOJ147 斗地主
题目描述
牛牛最近迷上了一种叫斗地主的扑克游戏。斗地主是一种使用黑桃、红心、梅花、方片的A到K加上大小王的共54张牌来进行的扑克牌游戏。在斗地主中,牌的大小关 系根据牌的数码表示如下:3<4<5<6<7<8<9<10<J<Q<K<A<2<小王<大王3<4<5<6<7<8<9<10<J<Q<K<A<2<小王<大王,而花色并不对牌的大小产生影响。每一局游戏中,一副手牌由 nn 张牌组成。游戏者每次可以根据规定的牌型进行出牌,首先打光自己的手牌一方取得游戏的胜利。
现在,牛牛只想知道,对于自己的若干组手牌,分别最少需要多少次出牌可以将它们打光。请你帮他解决这个问题。
需要注意的是,本题中游戏者每次可以出手的牌型与一般的斗地主相似而略有不同。具体规则如下:
牌型 | 牌型说明 | 牌型举例 |
---|---|---|
火箭 | 即双王(双鬼牌) | ♂ ♀ |
炸弹 | 四张同点牌。 | ♠A ♥A ♣A ♦A |
单张牌 | 单张牌 | ♠3 |
对子牌 | 两张码数相同的牌 | ♠2 ♥2 |
三张牌 | 三张码数相同的牌 | ♠3 ♥3 ♣3 |
三带一 | 三张码数相同的牌 + 一张单牌 | ♠3 ♥3 ♣3 ♠4 |
三带二 | 三张码数相同的牌 + 一对牌 | ♠3 ♥3 ♣3 ♠4 ♥4 |
单顺子 | 五张或更多码数连续的单牌(不包括 2 点和双王) | ♠7 ♣8 ♠9 ♣10 ♣J |
双顺子 | 三对或更多码数连续的对牌(不包括 2 点和双王) | ♣3 ♥3 ♠4 ♥4 ♠5 ♥5 |
三顺子 | 二个或更多码数连续的三张牌(不能包括 2 点和双王) | ♠3 ♥3 ♣3 ♠4 ♥4 ♣4 ♠5 ♦5 ♥5 |
四带二 | 四张码数相同的牌+任意两张单牌(或任意两对牌) | ♠5 ♥5 ♣5 ♦5 ♣3 ♣8 |
输入格式
第一行包含用空格隔开的2个正整数 T,nT,n ,表示手牌的组数以及每组手牌的张数。
接下来 TT 组数据,每组数据 nn 行,每行一个非负整数对 ai,biai,bi ,表示一张牌,其中 aiai 表示牌的数码, bibi 表示牌的花色,中间用空格隔开。特别的,我们用 11 来表示数码 A, 1111表示数码 J, 1212 表示数码 Q, 1313 表示数码 K;黑桃、红心、梅花、方片分别用 1-4 来表示;小王的表示方法为 0 1 ,大王的表示方法为 0 2 。
输出格式
共 TT 行,每行一个整数,表示打光第 ii 组手牌的最少次数。
样例一
input
1 8
7 4
8 4
9 1
10 4
11 1
5 1
1 4
1 1
output
3
explanation
共有 11 组手牌,包含 88 张牌:方片 7,方片 8,黑桃 9,方片 10,黑桃 J,黑桃 5,方片 A以及黑桃 A。可以通过打单顺子(方片 7,方片 8,黑桃 9,方片 10,黑桃 J),单张牌(黑桃 5)以及对子牌(黑桃 A以及方片 A)在 33 次内打光。
样例二
input
1 17
12 3
4 3
2 3
5 4
10 2
3 3
12 2
0 1
1 3
10 1
6 2
12 1
11 3
5 2
12 4
2 2
7 2
output
6
数据规模与约定
对于不同的测试点,我们约定手牌组数 TT ,与张数 nn 的规模如下:
测试点编号 | TT 的规模 | nn 的规模 | 测试点编号 | TT 的规模 | nn 的规模 |
---|---|---|---|---|---|
1 | 100100 | 22 | 11 | 100100 | 1414 |
2 | 100100 | 22 | 12 | 100100 | 1515 |
3 | 100100 | 33 | 13 | 1010 | 1616 |
4 | 100100 | 33 | 14 | 1010 | 1717 |
5 | 100100 | 44 | 15 | 1010 | 1818 |
6 | 100100 | 44 | 16 | 1010 | 1919 |
7 | 100100 | 1010 | 17 | 1010 | 2020 |
8 | 100100 | 1111 | 18 | 1010 | 2121 |
9 | 100100 | 1212 | 19 | 1010 | 2222 |
10 | 100100 | 1313 | 20 | 1010 | 2323 |
数据保证:所有的手牌都是随机生成的。
正解:搜索
解题报告:
这道题做法很多,可以状压,也可以搜索+剪枝。
我的做法的话就是搜索+最优性剪枝,每次对于当前局面得到一个答案上界,就是能带就带,然后我只需要枚举每次打了什么顺子(三顺子、双顺子、单顺子)就可以了。
//It is made by jump~
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <ctime>
#include <vector>
#include <queue>
#include <map>
#include <set>
using namespace std;
typedef long long LL;
#define RG register
const int inf = (<<);
int n,ans;
int a[],cnt[]; inline int getint()
{
int w=,q=; char c=getchar();
while((c<'' || c>'') && c!='-') c=getchar(); if(c=='-') q=,c=getchar();
while (c>='' && c<='') w=w*+c-'', c=getchar(); return q ? -w : w;
} inline int suan(){
for(int i=;i<=;i++) cnt[i]=; for(int i=;i<=;i++) cnt[a[i]]++;
int tot=;
while(cnt[]> && cnt[]>=) tot++,cnt[]--,cnt[]-=;//计算四带二
while(cnt[]> && cnt[]>=) tot++,cnt[]--,cnt[]-=;//计算四带一
while(cnt[]> && cnt[]>) tot++,cnt[]--,cnt[]--;//计算三带二
while(cnt[]> && cnt[]>) tot++,cnt[]--,cnt[]--;//计算三带一
return tot+cnt[]+cnt[]+cnt[]+cnt[];
} inline void dfs(int step){//每次尽可能消耗的牌多
if(step>=ans) return ;
ans=min(ans,step+suan());
int now;
for(int i=;i<=;i++) {//三顺子
now=; for(int j=i;j<=;j++) if(a[j]<) { now=j; break; }
if(now-i>=) {
for(int k=now-i;k>=;k--) {
for(int l=i;l<i+k;l++) a[l]-=;
dfs(step+);
for(int l=i;l<i+k;l++) a[l]+=;
}
}
}
for(int i=;i<=;i++) {//双顺子
now=; for(int j=i;j<=;j++) if(a[j]<) { now=j; break; }
if(now-i>=) {
for(int k=now-i;k>=;k--) {
for(int l=i;l<i+k;l++) a[l]-=;
dfs(step+);
for(int l=i;l<i+k;l++) a[l]+=;
}
}
}
for(int i=;i<=;i++) {//单顺子
now=; for(int j=i;j<=;j++) if(a[j]<) { now=j; break; }
if(now-i>=) {
for(int k=now-i;k>=;k--) {
for(int l=i;l<i+k;l++) a[l]--;
dfs(step+);
for(int l=i;l<i+k;l++) a[l]++;
}
}
}
} inline void work(){
int T=getint(); n=getint(); int x;
while(T--) {
memset(a,,sizeof(a));
for(int i=;i<=n;i++) {
x=getint();
if(x==) a[]++;
else if(x==) a[]++;
else a[x-]++;
x=getint();
}
ans=suan();dfs();
printf("%d\n",ans);
}
} int main()
{
work();
return ;
}
UOJ147 斗地主的更多相关文章
- 【uoj147】NOIP2015—斗地主
http://uoj.ac/problem/147 (题目链接) 题意 打牌... Solution 其实很简单的搜索,当年还是太年轻了.稍微想一想,顺子肯定是要先打掉的,因为顺子所包含的牌最多,所以 ...
- UOJ 151 斗地主“加强”版
#151. [NOIP2015]斗地主“加强”版 统计 描述 提交 自定义测试 本题开放Hack 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54 ...
- NOIP2015斗地主[DFS 贪心]
题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3<4< ...
- Android斗地主棋牌游戏牌桌实现源码下载
本次给大家分享下Android斗地主棋牌游戏牌桌实现源码下载如下: 为了节约内存资源,每张扑克牌都是剪切形成的,当然这也是当前编程的主流方法. 1.主Activity package com.biso ...
- Android开源益智游戏“斗地主”单机版源代码
Android开源益智游戏"斗地主"单机版源代码 这是一个网上流传的Android开源斗地主单机版项目,运行结果如图: 项目源代码导入到Eclipse后可直接运行,我把ecl ...
- [NOIP2015] 斗地主(搜索)
题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3<4< ...
- Java基础之如何解决斗地主问题
难的是逻辑的分析,把逻辑转化成代码是一种能力,这种能力需要多练习总结. 多多指教,共同进步. 问题: 要求实现斗地主游戏发牌过程,打印三个玩家的牌和底牌.在不看底牌的情况下,统计出三个 ...
- Java写的斗地主游戏源码
源码下载在最后 我们的前年的课设要求做一个斗地主程序,当时正在愁如何做界面,当时刚好在学习C#,于是就用C#完成了这个程序.一方面,当时我C#功底还很差(其实现在也不怎么样),很多地方用了“笨办法”, ...
- 斗地主(Noip2015Day1T3)
题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3<4< ...
随机推荐
- zabbix常用术语
zabbix常用术语
- 柯里化/偏函数/Curring用法
把接受多个参数的函数变成一个单一参数的函数,并且返回接受余下的参数而有返回结果的新函数的技术 下面我们以实例说明: var toString = {}.toString; var isString = ...
- Map集合 总结
(本人第一次写博客,部分内容有参照李刚老师的疯狂java系列图书,如有遗漏错误,请多指教,谢谢.) Java的集合类可分为Set.List.Map.Queue,其中Set.List.Queue都有共同 ...
- express:webpack dev-server中如何将对后端的http请求转到https的后端服务器中?
在上一篇文章(Webpack系列:在Webpack+Vue开发中如何调用tomcat的后端服务器的接口?)我们介绍了如何将对于webpack-dev-server的数据请求转发到后端服务器上,这在大部 ...
- [MetaHook] Find a function signature
Find a non-public function signature, we need a tool "IDA Pro" ( You can open picture in a ...
- [BZOJ2730][HNOI2012]矿场搭建(求割点)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2730 分析: 如果坍塌的点不是割点,那没什么影响,主要考虑坍塌的点是割点的情况. 显然 ...
- 【AHOI2014复仇】
RT,NOIP全挂,屌丝要逆袭……原本准备在QQ空间写,结果发现打不开,然后发现了这个……
- JavaScript变量的作用域和函数的作用域的区别
变量作用域和函数作用域都涉及到变量值的变化,本文旨在让大家明白他们之间的区别 变量的作用域: 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部可以直接 ...
- poj 1698 Alice‘s Chance
poj 1698 Alice's Chance 题目地址: http://poj.org/problem?id=1698 题意: 演员Alice ,面对n场电影,每场电影拍摄持续w周,每周特定几天拍 ...
- Service之来电监听(失败的案例)
Service:服务,可以理解成一个运行再后台没有界面的Activity,集成于Seriver,是四大组件之一 Service的继承关系:Service-->ContextWrapper--&g ...