计算24点的程序及分析过程

最近正在学习数据结构。和同事闲聊时,聊到了24点的问题(就是给出4个数,然后用加减剩除及括号算出24来),记得前一阵还在网上看了一道24点的题: 5 5 5 1 ,如何算出24来?当时想了好长时间没想出来。最近工作不是很忙就自己写一下程序练习一下吧。
以下讲述的是我思考该问题的思路及方法,如有不当之处请赐教:)
我首先想到的是用穷举法,因为就4 个数和4 个运算符再加上括号,表达式的数量不会很多,我算了一下,运算数:4×3×2×1=24,运算符:4×4×4×4=256,再括号应该是10种(没有优化的情况),总共是:24×256×10=61440,用穷举的话也很快(其实是想不到时其它办法:)
要穷举的话,可以大体分为三部分:
1.生成表达式
2.计算表达式的值
3.输出正确的结果
其中以上三步最复杂的是第一步(其实要是写程序的话,第2步应该是最麻烦的,但是php已经为我们提供的简单的方法,稍后讨论)
生成表达式时,我决定采用递归来写,原因是写起来简单些,但效率会稍低一些。
生成表达式也就是一个组合的问题,大体算法就像这样:

$str = array(1,2,3,4);
    $r   = array();

function c($k)
        {
                global $str;
                global $r;
                if ( $k == 4 ) {
                        display();
                        return;
                }
                                                                                                               
                for ($i = 0; $i < 4; $i++) {
                        $r[$k] = $str[$i];
                        c($k+1);
                }
        }

以上程序稍加改动就成了N皇后的解法。
还是感觉用递归写比较简洁一些。
有了以上的程序,再做一睦简单的改动便可以生成我们所需要的表达式(详见后面的程序)。
另外还有一点就是括号的问题,为了减少复杂度,我是直接在生成的表达式里添加上括号的,把所有的括号形式都写上(最好优化一下,要不然会产生很多有多余括号的表达式)。

写好了生成表达式的函数后,这个程序的工作就算做完一多半了。第二步计算表达式的值我用到了php的一个函数:eval(),有了这个函数可真是省了不少力,只需一行便能把字符串的表达式算出结果(具体用法可以查一下手册),妙!

剩下的第三点就简单了,真是输出一下结果而以。
以下是程序的完整代码,如有好的建议请多多指教:

<?php
                                                                                                               
        /***************************************************************************************
         * 功能:本程序的功能是给出4个数,求出通过 '+'、'-'、'*'、'/'及括号运算,结果为24的所有解
         * 参数说明:接收4个数字,用空格分开
         * es: 24point 1 2 3 4
         * 输出:输出所有正确的解
         *
         * Author : xks
         * Email  : xks@anymacro.com
         *
         * 一  8月 22 16:45:48 CST 2005
         ***************************************************************************************/
                                                                                                               
        define('OPERANDNUM', 4); //支持的运算数的个数,本程序只支持4个
        define('RESULT', 24);   //所要求的结果是24, 也可改为其它值
        $operand = array(); //运算数
        $operator = array('+', '-', '*', '/'); //运算符(不包括括号,括号在本程序中直接加上的)
        $exp    = array(); //生成的表达式
        $rightexp = array(); //正确的表达式
        $rightnum = 0; //正确的表达式的数目
        $lenoperator = count($operator); //运算符的个数(不包括括号)
        $lenexp = $lenoperator + OPERANDNUM; //整个表达式的长度(不包括号)
                                                                                                               
        //从命令行接收运算数,本程序只支持4个
        if ($argc != OPERANDNUM + 1) {
                echo "Arguments error!/n";
                echo "es: 24point 1 2 3 4/n";
                return;
        }
        else {
                for ($i = 0; $i < OPERANDNUM; $i++)
                        $operand[$i] = $argv[$i+1];
        }

$opetimes = array_count_values(array_slice($operand, 0));//统计每个运算数出现的次数,
                                                                 //用来检测是否有重复表达式
        //开始计算
        Createexp(0);
                                                                                                               
        //递归函数,用来根据输入的运算数生成表达式,并且调用 calculate()函数计算表达式的结果
        function CreateExp($k)
        {
                global $operator;
                global $operand;
                global $lenoperator;
                global $exp;
                global $lenexp;
                global $opetimes;
                                                                                                               
                $i = 0;//运算数计数
                $j = 0;//运算符计数
                                                                                                               
                //生成表达式后传给 calculate()计算
                if ($k == $lenexp - 1) {
                        calculate($exp);
                        return;
                }
                while ($i < OPERANDNUM && $j < $lenoperator) {
                        //当k为奇数时,则$exp[$k]应该填入运算符,否则$exp[$k]应该填入运算数
                        //ps:我在写这条注释的时候,突然头脑中闪过一个问题:0是偶数吧?呵呵小学里的概念都记不清
                        //了,我问了两位同事竟然也有不同的答案,google了一下,竟然发现有好多人在讨论 0是奇数
                        //还是偶然的问题,真是有意思。所以这里先定为奇数和非奇数吧:)
                        if ($k % 2 == 0) {
                                /* 去除相同数字开头的表达式,避免产生重复的表达式
                                 * 例如:3 5 3 2 ,以第一个3开头的表达式和以第二个3开头的表达式是相同的,
                                 * 所以如果运算数中有相同的数字,则只处理第一个出现的数字
                                */
                                if ( in_array($operand[$i], array_slice($operand, 0, $i)) && $k == 0) {
                                        break;
                                }
                                                                                                               
                                //测试一个表达式中是否有超过运算数出现的次数的数,不太好理解,也就是说
                                //比如4个运算数为:3 5 3 2 ,下面的语句就是为了防止一个表达式中
                                //出现3个3的情况
                                $expcount = array_count_values(array_slice($exp, 0, $k));
                                if ($expcount[$operand[$i]]=='' ||
                                                $expcount[$operand[$i]] < $opetimes[$operand[$i]] ) {
                                        $exp[$k] = $operand[$i++];
                                }
                                else {
                                        $i++;
                                        continue;
                                }
                        }
                        else {  //奇数的情况
                                $exp[$k] = $operator[$j++];
                        }
                        //递归调用
                        CreateExp($k+1);
                }
        }
                                                                                                               
        function calculate($exp)
        {
                global $rightexp;
                global $rightnum;
                global $rightnum;
                                                                                                               
                //去除重复的表达式,我在本程序中的做法是将正确的表达式存到一个数组中,产生新的表达式时,
                //先在数组中查找一下是否已经出现过。Not a good idea! 但暂时想不到更好的办法。
                //虽然在前面已经做了部分过滤工作,但是当运算数有重复值的时候还是无法避免出现产生重复表达式
                //只好采用此方法
                $strexp = implode('', $exp);
                if ( array_search( $strexp, $rightexp ))
                        return;
                                                                                                               
                $rightexp[$rightnum++] = $strexp;
                $exp2 = array();
                $i = 0;
                                                                                                               
                /* 以下是对添加上括号的表达式处理方法,已经做过优化,不会产生多余括号的情况
                 * 具体方法可以自己分析一下,写的不是很优雅。
                 * 4个数可以产生的表达式变形如下(运算数之间的空格用运算符填充):
                 *               a   b    c   d
                 *               a   b   (c   d)
                 *              (a   b)  (c   d)
                 *              (a   b   c)   d
                 *               a  (b   c)   d
                 *               a  (b   c   d)
                 *              (a  b) c  d
                 * 另外还有4种情况,经过分析后都可以优化掉。
                */
                                                                                                               
                $exp2[$i++] = "$exp[0]$exp[1]$exp[2]$exp[3]$exp[4]$exp[5]$exp[6]";
                if ( mulordiv($exp[3]) && addorsub($exp[5]))
                        $exp2[$i++] = "$exp[0]$exp[1]$exp[2]$exp[3]($exp[4]$exp[5]$exp[6])";
                if ( mulordiv($exp[3]) && mulordiv($exp[5]) && addorsub($exp[3]))
                        $exp2[$i++] = "($exp[0]$exp[1]$exp[2])$exp[3]($exp[4]$exp[5]$exp[6])";
                if ( mulordiv($exp[5])  &&  (addorsub($exp[1]) || addorsub($exp[3])) )
                        $exp2[$i++] = "($exp[0]$exp[1]$exp[2]$exp[3]$exp[4])$exp[5]$exp[6]";
                if ( (mulordiv($exp[1])  || mulordiv($exp[5])) && addorsub($exp[3]) )
                        $exp2[$i++] = "$exp[0]$exp[1]($exp[2]$exp[3]$exp[4])$exp[5]$exp[6]";
                if ( (mulordiv($exp[1])  || mulordiv($exp[5])) && addorsub($exp[3]) )
                        $exp2[$i++] = "$exp[0]$exp[1]($exp[2]$exp[3]$exp[4])$exp[5]$exp[6]";
                if ( (addorsub($exp[1]) ||  addorsub($exp[3]))  && mulordiv($exp[1]) )
                        $exp2[$i++] = "$exp[0]$exp[1]($exp[2]$exp[3]$exp[4]$exp[5]$exp[6])";
                if ( addorsub($exp[1]) && mulordiv($exp[3]) )
                        $exp2[$i++] = "($exp[0]$exp[1]$exp[2])$exp[3]$exp[4]$exp[5]$exp[6]";
                                                                                                               
                // 计算表达式的值,其中有一个很重要的技巧:eval(),这个函数可真是好用,帮我省掉了1xx行代码
                // 利用这个函数可以直接求出表达式的值,而不用再去分析表达式以及运算符的优先级等等
                // 求表达式的值属于编译原理的知识,如果你正在学的话,那自己写个求表达式的程序练习一下
                //是个不错的建议 u
                for ($j = 0; $j < $i; $j++) {
                        $result = 0;
                        @eval("/$result=$exp2[$j];");
                        if ($result == RESULT) {
                                display($exp2[$j]);
                        }
                }
        }
                                                                                                               
        //显示表达式
        function display($exp)
        {
                echo $exp  . '=' . RESULT ."/n";
        }
                                                                                                               
        //测试$opt 是否为'+' or  '-'
        function addorsub($opt)
        {
                return ($opt == '+' || $opt == '-');
        }
                                                                                                               
        //测试$opt 是否为'*' or  '/'
        function mulordiv($opt)
        {
                return ($opt == '*' || $opt == '/');
        }
                                                                                                               
?>

计算24点的程序及分析过程相关推荐

  1. 24点游戏java_使用java编写计算24点游戏程序

    初学java,编写了一个计算24点的程序,时间有限,有些粗糙,不过可以使用. //-------------Cal24.java--------------- //计算24点程序 //作者:徒步天下( ...

  2. Android应用程序进程启动过程的源代码分析(1)

    Android应用程序框架层创建的应用程序进程具有两个特点,一是进程的入口函数是ActivityThread.main,二是进程天然支持Binder进程间通信机制:这两个特点都是在进程的初始化过程中实 ...

  3. java语言执行过程_Java程序的运行过程(执行流程)分析

    万事知其然,要知其所以然,所以本节带大家来详细了解一下 Java 程序的执行过程.从<使用记事本编写运行Java程序>一节的案例可以看出,Java 程序的运行必须经过编写.编译和运行 3 ...

  4. Java教程:Java程序的运行过程(执行流程)分析

    Java教程万事知其然,要知其所以然,所以本节带大家来详细了解一下 Java 程序的执行过程. 本篇文章由 泉州SEO www.234yp.com 整理发布,Java教程 www.234yp.com/ ...

  5. 习题3.6 阅读下面程序,分析其执行过程,写出输出结果

    谭浩强c++ 面向对象程序设计(第2版) 习题3.6 阅读下面程序,分析其执行过程,写出输出结果 #include <iostream> using namespace std; clas ...

  6. GitHub服务中断24小时11分钟事故分析报告\n

    上周,GitHub经历了一次事故,导致服务降级24小时11分钟.虽然平台的某些部分不受事故影响,但仍然有多个内部系统受到了影响,向用户显示了过时且不一致的内容.所幸没有用户数据丢失,但针对几秒钟数据库 ...

  7. 5.1 mxc_v4l2_capture.c应用程序追踪分析

    对于IPU在内核驱动中的执行过程,需要通过应用程序的函数调用来一步一步追踪,下面就根据mxc_v4l2_capture.c这个应用程序来分析.经过此轮分析,应该对IPU内部那些函数都有一个大致的认识. ...

  8. 开启 GODEBUG 更直观地理解 Go 程序的调度过程

    得益于 Go 语言优秀的运行时调度系统,即使开发人员没有多线程编程经验,也能很容易地开发并发程序. 调度系统,其中最核心的就是 GMP 的设计,欲深入理解 Go 语言设计的读者都应该看过这些知识.但是 ...

  9. c语言算24点答案,C语言-纸牌计算24点小游戏

    C语言实现纸牌计算24点小游戏 利用系统时间设定随机种子生成4个随机数,并对4个数字之间的运算次序以及运算符号进行枚举,从而计算判断是否能得出24,以达到程序目的.程序主要功能已完成,目前还有部分细节 ...

最新文章

  1. spring和hibernate的集成
  2. 图片浏览(CATransition)转场动画
  3. js中const,var,let的区别
  4. photo常用快捷键
  5. 王道操作系统考研笔记——2.3.4 信号量机制
  6. 怎样在数组末尾添加数据_如何利用C++实现可变长的数组?
  7. s5pv210——SDRAM的初始化
  8. makefile / CMake
  9. python测试框架untest怎么循环执行_unittest如何在循环遍历一条用例时生成多个测试结果...
  10. 欠20万信用卡卡奴自救方法
  11. 团购潮冲击佣金模式 携程拒三倍赔付藏消费陷阱
  12. 新物联网时代的整合战略
  13. 2019年2月CNNIC第43次《中国互联网络发展状况统计报告》
  14. Java语言基础Day07(API概述、Scanner、匿名对象、Random、对象数组、ArrayList)
  15. html5的calc,CSS 计算属性 calc()的完整指南(下)
  16. 机器学习7——线性回归
  17. C语言初阶(18) | 数组详解
  18. 网易云易盾朱星星:最容易被驳回的10大APP过检项
  19. 30.7.1 通过mysqladmin修改用户密码
  20. NFC通信基本原理 主动和被动通信

热门文章

  1. 富文本编辑器——百度UEditor插件Vue组件化
  2. RTT开发者大会学习
  3. JPA - Hibernate
  4. Hibernate的JPA
  5. 浙江义乌外商当志愿者助力春运 一口流利汉语获赞
  6. 最新《煤矿井下人员定位系统安全标志管理方案(试行)》等通知,助力煤矿新进程
  7. 新基建打开万亿ICT下沉市场,十四五科技大局激发新联想
  8. 服务器端验证表单_Web表单开发之实时格式化显示——Cleave.js
  9. 米哈游的春招实习面经,问的很基础
  10. 强大的CSS:图形绘制合集,方便你我!