题目链接

Problem Description

速算24点相信绝大多数人都玩过。就是随机给你四张牌,包括A(1),2,3,4,5,6,7,8,9,10,J(11),Q(12),K(13)。要求只用'+','-','*','/'运算符以及括号改变运算顺序,使得最终运算结果为24(每个数必须且仅能用一次)。游戏很简单,但遇到无解的情况往往让人很郁闷。你的任务就是针对每一组随机产生的四张牌,判断是否有解。我们另外规定,整个计算过程中都不能出现小数。

Input

每组输入数据占一行,给定四张牌。

Output

每一组输入数据对应一行输出。如果有解则输出"Yes",无解则输出"No"。

Sample Input

A 2 3 6

3 3 8 8

Sample Output

Yes

No

题目描述:

给定4个数字,判定这4个数字是否可以通过运算得到结果24。运算符包括:加、减、乘、除,允许变换数字的顺序,当然也可以使用括号改变运算顺序。

即:判定是否存在一种通过在下面的圆圈中添加运算符以及添加括号的方式使得等式成立:

a ○ b ○ c ○ d = 24

例如:数字 5, 5, 5, 1 可以通过运算得到结果24:

5 × (5 - 1 ÷ 5) = 24

数字 9, 9, 9, 9 则无论通过怎样的运算都无法得到24的结果。

分析:

最开始的时候想的情况比较复杂,后来发现其实对于使用小括号来改变运算顺序的话,完全可以通过改变数字的运算顺序来达到相同的目的,考虑的情况就会大大减少,因此可以采用暴力搜索的方式进行解答,枚举所有的可能的运算方式,如果有1种方案可以达到24,则表明给定的数据有解,否则无解。

枚举的方式如下:

由于加法和乘法满足交换律,而减法和除法则不满足,这里我们添加两种运算:“反减”和“反除”,分别记作 ~- 和 ~÷,此时:

a ~- b = b - a

a ~÷ b = b ÷ a

添加这两种运算操作后,就使得减法和除法也同样符合交换律。此时,所有情况的运算顺序就只有两种情况,用※表示任意的运算符,则两种运算顺序为:

((a ※ b) ※ c) ※ d)) 从左到右依次计算

((a ※ b) ※ (c ※ d)) 分别计算前两个和后两个数字的结果后,在将得到的两个结果进行运算,得到最终的结果

而其他的运算顺序均可以通过调整数字的排列顺序得到用以上两种情况表示的等价情况,例如:

(a ※ ((b ※ c) ※ d)) =====> (((b ※ c) ※ d) ※ a)

((a ※ (b ※ c)) ※ d) =====> (((b ※ c) ※ a) ※ d)

这样,通过枚举4个数字排列情况和三个位置的运算符的不同情况,就可以枚举到所有的运算情况。其中:4个数字排列,共有4!= 24种结果,而3个运算符中每个运算符都有6种不同的情况,因此共有6^3 = 216中情况,而运算顺序又有2中情况,因此,总的情况数为:

24 × 216 × 2 = 10368

由于加法和乘法本身就具有交换律,并且给定的4个数字也有某些数字相同的情况,因此,在实际枚举过程中,10368种情况会有某些是重复的。本题数据量比较小,因此重复计算的问题可以忽略。

代码:

  1. #include<stdio.h>
  2. #include<iostream>
  3. #include<algorithm>
  4. #include<string>
  5. #include<string.h>
  6. using namespace std;
  7. int num[4];
  8. bool flag;
  9. void dfs(int sum,int now,int times)
  10. {
  11. if(!flag)
  12. return ;
  13. if(times==3)
  14. {
  15. if(sum+now==24||sum-now==24||sum*now==24)
  16. flag=false;
  17. if(now!=0&&sum%now==0&&sum/now==24)
  18. flag=false;
  19. return;
  20. }
  21. //先计算前面的一部分
  22. dfs(sum+now,num[times+1],times+1);
  23. dfs(sum-now,num[times+1],times+1);
  24. dfs(sum*now,num[times+1],times+1);
  25. if(now!=0&&sum%now==0)
  26. dfs(sum/now,num[times+1],times+1);
  27. //再计算后面一部分
  28. dfs(sum,now+num[times+1],times+1);
  29. dfs(sum,now-num[times+1],times+1);
  30. dfs(sum,now*num[times+1],times+1);
  31. if(num[times+1]!=0&&now%num[times+1]==0)
  32. dfs(sum,now/num[times+1],times+1);
  33. }
  34. int main()
  35. {
  36. char ch[6];
  37. while(~scanf("%s",ch))
  38. {
  39. flag=true;
  40. if(strlen(ch)==2)//除了数字10之外,所有的数字都是一位字符来表示的e
  41. num[0]=10;
  42. else
  43. {
  44. if(ch[0]=='A')
  45. num[0]=1;
  46. else if(ch[0]=='J')
  47. num[0]=11;
  48. else if(ch[0]=='Q')
  49. num[0]=12;
  50. else if(ch[0]=='K')
  51. num[0]=13;
  52. else num[0]=ch[0]-'0';
  53. }
  54. for(int i=1; i<=3; i++)
  55. {
  56. scanf("%s",ch);
  57. if(strlen(ch)==2)//除了数字10之外,所有的数字都是一位字符来表示的e
  58. num[i]=10;
  59. else
  60. {
  61. if(ch[0]=='A')
  62. num[i]=1;
  63. else if(ch[0]=='J')
  64. num[i]=11;
  65. else if(ch[0]=='Q')
  66. num[i]=12;
  67. else if(ch[0]=='K')
  68. num[i]=13;
  69. else num[i]=ch[0]-'0';
  70. }
  71. }
  72. sort(num,num+4);
  73. do
  74. {
  75. dfs(num[0],num[1],1);
  76. }
  77. while(next_permutation(num,num+4)&&flag);//全排列函数来枚举四个数字的任意的顺序
  78. if(!flag)//只要有一种情况计算出来结果
  79. printf("Yes\n");
  80. else
  81. printf("No\n");
  82. }
  83. return 0;
  84. }

HDU 1427 速算24点 (深搜)的更多相关文章

  1. hdu 1427 速算24点

    题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=1427 速算24点 Description 速算24点相信绝大多数人都玩过.就是随机给你四张牌,包括A( ...

  2. hdu 1427 速算24点【暴力枚举】

    速算24点 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submi ...

  3. hdu 1427 速算24点 dfs暴力搜索

    速算24点 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Problem De ...

  4. HDU 1427 速算24点【数值型DFS】

    速算24点 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submi ...

  5. 24点游戏&&速算24点(dfs)

    24点游戏 Time Limit: 3000/1000MS (Java/Others)     Memory Limit: 65535/65535KB (Java/Others) Submit Sta ...

  6. hdu1427之速算24点

    速算24点 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Subm ...

  7. Hdu1427 速算24点 2017-01-18 17:26 46人阅读 评论(0) 收藏

    速算24点 Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 65536/32768K (Java/Other) Total Submiss ...

  8. hdu 5648 DZY Loves Math 组合数+深搜(子集法)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5648 题意:给定n,m(1<= n,m <= 15,000),求Σgcd(i|j,i&am ...

  9. HDU 2553 N皇后问题 (深搜)

    题目链接 Problem Description 在N*N的方格棋盘放置了N个皇后,使得它们不相互攻击(即任意2个皇后不允许处在同一排,同一列,也不允许处在与棋盘边框成45角的斜线上. 你的任务是,对 ...

随机推荐

  1. 性能分析_linux服务器CPU_中断

    中断 1.  指标范围 1.1  Interrupt rate 应该与cpu利用率结合分析,如果cpu利用率在合理范围内,大量的中断也是可以接受的.一个巨大的中断值,同时伴随着缓慢的系统性能表现,指示 ...

  2. js dom学习

    创建dom元素 var oLi = document.creteElement('li'); //创建livar aLi = oUl.getElementsByTagName('li');oLi.in ...

  3. 通过Get-Group导出组的成员

    导出组邮箱的前十个成员,需要注意的是: Get-Group没有Get-GroupMember命令,但是在结果中有一个Members的属性,这个属性包含了所有子成员的对象,用循环将它们列出来即可.有点对 ...

  4. [转帖]UsingDiskspdforSQLServer --- 改天做下练习 官方文档.

    Using DiskSpd in SQL Server environments Writer: Robert Beene Contributors: Jose Barreto, Ramu Konid ...

  5. vue 组件 组件2

    <!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title>T ...

  6. mysql索引的优化

    MySQL索引的优化 上面都在说使用索引的好处,但过多的使用索引将会造成滥用.因此索引也会有它的缺点:虽然索引大大提高了查询速度,同时却会降低更新表的速度,如对表进行INSERT.UPDATE和DEL ...

  7. 第212天:15种CSS居中的方式,最全了

    CSS居中是前端工程师经常要面对的问题,也是基本技能之一.今天有时间把CSS居中的方案整理了一下,目前包括水平居中,垂直居中及水平垂直居中方案共15种.如有漏掉的,还会陆续的补充进来,算做是一个备忘录 ...

  8. TeX-换行换页与段落命令

    换行换页与段落命令1 UTF8nsung Abstract 文档在排版时往往要求每一行具有相同的长度, LATEX 为了对整段的文挡进行优化,将插入必要的换行和空恪.如果必要的话对于一行中不好放的单词 ...

  9. 浅谈Tarjan算法及思想

    在有向图G中,如果两个顶点间至少存在一条路径,称两个顶点强连通(strongly connected).如果有向图G的每两个顶点都强连通,称G是一个强连通图.非强连通图有向图的极大强连通子图,称为强连 ...

  10. 【Cf #290 B】Fox And Jumping(dp,扩展gcd)

    根据裴蜀定理,当且仅当选出来的集合的L[i]的gcd等于1时,才能表示任何数. 考虑普通的dp,dp[i][j]表示前i个数gcd为j的最少花费,j比较大,但状态数不多,拿个map转移就好了. $ \ ...