斗地主:一步一步推性质就能做出来的剪枝题。

这题思路和小木棒的剪枝思路极其相似,剪枝的角度都差不多。

其实大部分搜索剪枝题都是先观察性质,列出性质后选择几个比较关键且代码好写的性质进行剪枝,特别要注意避免重复搜索相同状态的剪枝。同时注意想好了之后再写代码。

并且大部分搜索题会把正向的搜索树 hack 得很大,当被卡的时候不妨试试倒着搜索,可能有奇效。

思路

观察题目,我们可以发现如下几点性质:

  • 出牌的顺序不影响最终的结果,先出某一组牌和后出某一组牌本质是一样的。
  • 最后只剩下单牌、对子、三张牌、炸弹和王炸的时候,我们可以贪心地扫一遍统计至少存在一张牌的码数的个数,特判掉大小王得出答案。
  • 在顺子和带牌中,顺子打出的方案数一般比带牌的方案数更少。

因此,我们可以设计出如下剪枝方案:

  • 将搜索阶段拆解为 \(6\) 步,依次完成。分别是三顺子、双顺子、单顺子、四带二、三带一和二、最后剩下的统一出掉,大王小王绑一起出。
  • 记录下当前的搜索阶段 \(p\),以及上一次在同阶段最后搜索到的牌 \(lst\)。每次接着 \(lst\) 后面来搜索,避免搜到重复状态。
  • 把三顺子、双顺子、单顺子、四带二、三带一和二全部出完之后,我们可以线性扫一遍算出最终的答案,避免了搜索多余状态。

实现的时候对每个搜索阶段分步实现,代码在写顺子和带牌内部时可以稍作修改之后重复利用。

时间复杂度玄学,但是能过大部分数据。

代码

#include <bits/stdc++.h>
#define fi first
#define se second
using namespace std;
using pi=pair<int,int>;
/*
三顺子 1
双顺子 2
单顺子 3
四带二 4
三带一、二 5
最后剩下的统一出掉,大王小王绑一起出。
1~11 3~K
12 A
13 2
14 little
15 big
*/
int tot[20],ans=0x3f3f3f3f,now=0,n;
void dfs(int p,int lst)
{
if(p>=6)
{
int cur=0;
for(int i=1;i<=15;i++)if(tot[i])cur++;
if(tot[14]&&tot[15])cur--;
ans=min(ans,now+cur);
return;
}
if(p==1)
{
int pre=0;
for(int i=1;i<=12;i++)
{
if(tot[i]>=3)pre++;
else pre=0;
if(pre>=2&&i>=lst)
{
int tmp[20];
memcpy(tmp,tot,sizeof(tmp));
tot[i]-=3;
for(int j=2;j<=pre;j++)
{
tot[i-j+1]-=3;
now++;
dfs(p,i);
now--;
}
memcpy(tot,tmp,sizeof(tmp));
}
}
dfs(p+1,1);
return;
}
if(p==2)
{
int pre=0;
for(int i=1;i<=12;i++)
{
if(tot[i]>=2)pre++;
else pre=0;
if(pre>=3&&i>=lst)
{
int tmp[20];
memcpy(tmp,tot,sizeof(tmp));
tot[i]-=2;
tot[i-1]-=2;
for(int j=3;j<=pre;j++)
{
tot[i-j+1]-=2;
now++;
dfs(p,i);
now--;
}
memcpy(tot,tmp,sizeof(tmp));
}
}
dfs(p+1,1);
return;
}
if(p==3)
{
int pre=0;
for(int i=1;i<=12;i++)
{
if(tot[i]>=1)pre++;
else pre=0;
if(pre>=5&&i>=lst)
{
int tmp[20];
memcpy(tmp,tot,sizeof(tmp));
tot[i]-=1;
tot[i-1]-=1;
tot[i-2]-=1;
tot[i-3]-=1;
for(int j=5;j<=pre&&i-j+1>=1;j++)
{
tot[i-j+1]-=1;
now++;
dfs(p,i);
now--;
}
memcpy(tot,tmp,sizeof(tmp));
}
}
dfs(p+1,1);
return;
}
if(p==4)
{
for(int i=1;i<=13;i++)
{
if(tot[i]>=4)
{
tot[i]-=4;
for(int j=1;j<=15;j++)
{
if(tot[j]<=0)continue;
for(int k=1;k<=15;k++)
{
if(tot[k]<=0)continue;
if(tot[j]&&tot[k]&&(j!=k||tot[j]>=2))
{
tot[j]--;
tot[k]--;
now++;
dfs(p,i);
tot[j]++;
tot[k]++;
now--;
}
if(tot[j]>=2&&tot[k]>=2&&(j!=k||tot[j]>=4))
{
tot[j]-=2;
tot[k]-=2;
now++;
dfs(p,i);
tot[j]+=2;
tot[k]+=2;
now--;
}
}
}
tot[i]+=4;
}
}
dfs(p+1,1);
return;
}
if(p==5)
{
for(int i=1;i<=13;i++)
{
if(tot[i]>=3)
{
tot[i]-=3;
for(int j=1;j<=15;j++)
{
if(tot[j]>=1)
{
tot[j]--;
now++;
dfs(p,i);
tot[j]++;
now--;
}
if(tot[j]>=2)
{
tot[j]-=2;
now++;
dfs(p,i);
tot[j]+=2;
now--;
}
}
tot[i]+=3;
}
}
dfs(p+1,1);
return;
}
}
void solve()
{
memset(tot,0,sizeof(tot));
ans=0x3f3f3f3f;
now=0;
for(int i=1;i<=n;i++)
{
int a,b;
cin>>a>>b;
if(3<=a&&a<=13)tot[a-2]++;
else if(a<3&&a>0)tot[a+11]++;
else tot[b+13]++;
}
dfs(1,1);
cout<<ans<<'\n';
}
int main()
{
ios::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
int t;
cin>>t>>n;
while(t--)solve();
return 0;
}

Luogu P2540 NOIP2015提高组 斗地主 加强版 题解 [ 紫 ] [ 深搜 ] [ 剪枝 ]的更多相关文章

  1. [NOIP2015 提高组] 运输计划题解

    题目链接:P2680 [NOIP2015 提高组] 运输计划 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 看了好长时间题解才终于懂的,有关lca和二分答案的题解解释的不详细,一时 ...

  2. 一本通例题-生日蛋糕——题解<超强深搜剪枝,从无限到有限>

    题目传送 显然是道深搜题.由于蛋糕上表面在最底层的半径确认后就确认了,所以搜索时的面积着重看侧面积. 找维度/搜索面临状态/对象:当前体积v,当前外表面面积s,各层的半径r[],各层的高度h[]. 可 ...

  3. 【题解】NOIP2015提高组 复赛

    [题解]NOIP2015提高组 复赛 传送门: 神奇的幻方 \([P2615]\) 信息传递 \([P2661]\) 斗地主 \([P2668]\) 跳石头 \([P2678]\) 子串 \([P26 ...

  4. 洛谷 P2678 & [NOIP2015提高组] 跳石头

    题目链接 https://www.luogu.org/problemnew/show/P2678 题目背景 一年一度的“跳石头”比赛又要开始了! 题目描述 这项比赛将在一条笔直的河道中进行,河道中分布 ...

  5. 刷题总结——子串(NOIP2015提高组)

    题目: 题目背景 NOIP2015 提高组 Day2 T2 题目描述 有两个仅包含小写英文字母的字符串 A 和 B .现在要从字符串 A 中取出 k 个互不重叠的非空子串,然后把这 k 个子串按照其在 ...

  6. NOIP2015 提高组] 运输计划

    码农题啊兄弟们. 随便考虑二分一下,然后发现要取一条满足性质的边. 被所有大于\(mid\)的路径都覆盖,取了之后能把他们都弄到小于\(mid\) 那就树上差分再处理一下. 写了\(180h\),老年 ...

  7. [NOIP2015] 提高组 洛谷P2615 神奇的幻方

    题目描述 幻方是一种很神奇的N*N矩阵:它由数字1,2,3,……,N*N构成,且每行.每列及两条对角线上的数字之和都相同. 当N为奇数时,我们可以通过以下方法构建一个幻方: 首先将1写在第一行的中间. ...

  8. 洛谷-神奇的幻方-NOIP2015提高组复赛

    题目描述 幻方是一种很神奇的N*N矩阵:它由数字1,2,3,--,N*N构成,且每行.每列及两条对角线上的数字之和都相同. 当N为奇数时,我们可以通过以下方法构建一个幻方: 首先将1写在第一行的中间. ...

  9. 【数据结构】运输计划 NOIP2015提高组D2T3

    [数据结构]运输计划 NOIP2015提高组D2T3 >>>>题目 [题目描述] 公元 2044 年,人类进入了宇宙纪元.L 国有 n 个星球,还有 n−1 条双向航道,每条航 ...

  10. 【二分查找】 跳石头NOIP2015提高组 D2T1

    [二分查找]跳石头NOIP2015提高组 D2T1 >>>>题目 [题目描述] 一年一度的“跳石头”比赛又要开始了! 这项比赛将在一条笔直的河道中进行,河道中分布着一些巨大岩石 ...

随机推荐

  1. MySQL同步ES方案

    1. 前言 在实际项目开发中,我们经常将 MySQL 作为业务数据库,ES 作为查询数据库,用来实现读写分离,缓解 MySQL 数据库的查询压力,应对海量数据的复杂查询. 这其中有一个很重要的问题,就 ...

  2. Mybatis【9】-- Mybatis占位符#{}和拼接符${}有什么区别?

    代码直接放在Github仓库[https://github.com/Damaer/Mybatis-Learning ],可直接运行,就不占篇幅了. 目录 1.#{}占位符 2.${}拼接符 3.#{} ...

  3. 3-XSS渗透与防御

    1.HTTP协议回顾 XSS又名跨站脚本攻击 web页面登陆页面,往往有一个"记住密码"功能 ---> Cookie 1.1 HTTP流程 1.2 HTTP特点: 请求应答模 ...

  4. MySQL底层概述—6.索引原理

    大纲 1.索引原理 2.二叉查找树 3.平衡二叉树(AVL树) 4.红黑树 5.B-Tree 6.B+Tree 7.Hash索引 8.聚簇索引与非聚簇索引 1.索引原理 索引会在数据文件中(ibd文件 ...

  5. kube-apiserver 高可用,keepalived + haproxy

    为什么要做高可用 在生产环境中,kubernetes 集群中会多多个 master 节点,每个 master 节点上都会部署 kube-apiserver 服务,实现高可用.但是 client 访问 ...

  6. ChatGPT在功能测试用例生成方面的优势

    功能测试是软件测试的非常重要的分类,所有软件系统都要保证功能的正确性,而测试用例则是功能测试的重中之重.测试用例的编写是测试人员必须认真面对的一件耗时费力.枯燥乏味的工作.如何才能快速高效地编写测试用 ...

  7. 零基础学习人工智能—Python—Pytorch学习(十三)

    前言 最近学习了一新概念,叫科学发现和科技发明,科学发现是高于科技发明的,而这个说法我觉得还是挺有道理的,我们总说中国的科技不如欧美,但我们实际感觉上,不论建筑,硬件还是软件,理论,我们都已经高于欧美 ...

  8. WinDbg: Failed to find runtime module (coreclr.dll or clr.dll or libcoreclr.so)

    当我们通过 WinDbg 启动一个 .NET 的程序时,WinDbg 将会在运行可执行之前执行一个中断,此时还没有加载 .NET 的运行时. 但是,SOS 扩展需要 clr.dll 或者 corecl ...

  9. Qt音视频开发28-ffmpeg解码本地摄像头(yuv422转yuv420)

    一.前言 一开始用ffmpeg做的是视频流的解析,后面增加了本地视频文件的支持,到后面发现ffmpeg也是支持本地摄像头设备的,只要是原则上打通的比如win系统上相机程序.linux上茄子程序可以正常 ...

  10. Qt开发经验小技巧196-200

    关于Qt延时的几种方法. void QUIHelperCore::sleep(int msec) { if (msec <= 0) { return; } #if 1 //非阻塞方式延时,现在很 ...