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< ...
随机推荐
- ASP.NET MVC铵钮Click后下载文件
本篇Insus.NET练习的是FilePathResult和FileStreamResult操作.本篇也算是<如何把Json格式字符写进text文件中>http://www.cnblogs ...
- scala 学习笔记(05) OOP(中)灵活的trait
trait -- 不仅仅只是接口! 接上回继续,scala是一个非常有想法的语言,从接口的设计上就可以发现它的与众不同.scala中与java的接口最接近的概念是trait,见下面的代码: packa ...
- oracle:db-link使用
二个oracle instance,如果需要在一个instance上,直接查询另一个instance上的数据,就要用到db-link 创建: create public database link 链 ...
- TinyFrame再续篇:整合Spring AOP实现日志拦截
上一篇中主要讲解了如何使用Spring IOC实现依赖注入的.但是操作的时候,有个很明显的问题没有解决,就是日志记录问题.如果手动添加,上百个上千个操作,每个操作都要写一遍WriteLog方法,工作量 ...
- QTableView 一列添加两个按钮
在QTableView的一列里添加两个按钮,之前添加一个按钮的思路是一样的,只是计算了一下按钮的宽,放两个按钮而已. 添加一个按钮的例子:QTableView 添加按钮 本例源代码:QtTowButt ...
- 让时间处理简单化 【第三方扩展类库org.apache.commons.lang.time】
JAVA的时间日期处理一直是一个比较复杂的问题,大多数程序员都不能很轻松的来处理这些问题.首先Java中关于时间的类,从 JDK 1.1 开始,Date的作用很有限,相应的功能已由Calendar与D ...
- win7下IIS配置以及域名映射方法
win7下IIS配置以及域名映射方法 第一步:打开控制面板,选择程序与功能,如下图: 第二步:双击打开程序与功能面板,如下图: 第三步:打开”打开或关闭windows功能”(红线圈起来的地方),如下图 ...
- web安全——数据库(mysql)
简介 数据安全是现在互联网安全非常重要一个环节.而且一旦数据出现问题是不可逆的,甚至是灾难性的. 有一些防护措施应该在前面几个博文说过了,就不再赘述.比如通过防火墙控制,通过系统的用户控制,通过web ...
- Scala入门之函数
/** * 函数可以被简单的被认为是包裹了一条或者几条语句的代码体,该代码体接收若干参数,经过代码体处理后返回结果,形如数学中的f(x) = x + 1 * 在Scala中函数式一等公民,可以向变量一 ...
- DbEntry在Vs2012里的配置
dbentry官方的版本还不支持vs2012,要再vs2012中使用,必须做下调整 1:新建类库项目,然后添加dbentry 的dll引用. 2:在建好的类库项目中.csproj 新添加了类库项目后, ...