Python秒算24点,行仍是不行?

在这里插入图片描述

周末闲来无事,看到隔壁家的老王在和隔壁家的媳妇玩24点,就进屋看了看。发现老王是真不行啊,那不行,这也不行。python

就连个24点都玩不过他媳妇,给他媳妇气的,啥都不能知足,这不能,那也不能。git

在这里插入图片描述

我坐下来和他媳妇玩了两把,那都是无出其右,把把赢!github

我要走的时候,他媳妇还挽留我多玩几把,有意思。算法

为了能让老王在他媳妇面前抬起头来,我决定帮他一把……就用python写了个算24点的玩意,老王对我感激不尽。windows

在这里插入图片描述

什么是24点

咱们先来约定下老王和他媳妇玩的24点规则:给定4个任意数字(0-9),而后经过+,-,*,/,将这4个数字计算出24。编辑器

小时候玩的都是这个规则,长大了才有根号,才有各类莫名其妙的高级算法,很差玩了,由于我不会。函数

可能有人会以为很简单,可是真的简单吗?测试

好比:spa

  • 8,3,3,3
  • 7,3,3,3

你能一眼看出来答案吗?好像真的能够……操作系统

大体思路

这样想,将四个数字进行全排列,在他们之间添加运算符号。

运算符咱们须要进行排列组合,由于只有四个数字,因此只须要三个运算符,并且算法符可能会重复,好比三个都是+

再遍历四个数字的全排列,对每一组数字而言,遍历全部组合的操做符。最后将数字和操做符进行拼接运算,就能够获得最终结果了。

演示环境

操做系统:windows10

python版本:python 3.7

代码编辑器:pycharm 2018.2

使用模块:math,itertools, collections.abc

具体代码

一、首先咱们对全部数字进行去全排列,这里咱们使用 itertools.permutations 来帮助咱们完成。

iertools.permutations 用法演示

from itertools import permutations

data_list = permutations([1,2,3,4],2)
for data in data_list:
print(data)
复制代码

结果显示

(1, 2)
(1, 3)
(1, 4)
(2, 1)
(2, 3)
(2, 4)
(3, 1)
(3, 2)
(3, 4)
(4, 1)
(4, 2)
(4, 3)
复制代码

permutations 第一个参数是接收一个课迭代的对象,第二个参数指定每次排列时从课迭代对象中选着几个字符进行排列。也能够不传入第二个参数,那么默认就是可迭代对象的长度。而且返回一个生成器。

因此咱们须要对全部数字进行全排列,就能够像下面这样写:

def get_all_data_sequence(data_iter):
    return permutations(data_iter)
复制代码

二、而后咱们须要拿到全部的操做运算符的全部组合方式。这里咱们就会使用 itertools.product 函数了。

itertools.product 用法演示

from itertools import product

sequence1 = product('ABCD','xy')
sequence2 = product([0,1],repeat=3)

for sequence in sequence1:
    print(sequence)

print('-'*30)

for sequence in sequence2:
    print(sequence)
复制代码

结果显示

('A','x')
('A','y')
('B','x')
('B','y')
('C','x')
('C','y')
('D','x')
('D','y')
------------------------------
(0, 0, 0)
(0, 0, 1)
(0, 1, 0)
(0, 1, 1)
(1, 0, 0)
(1, 0, 1)
(1, 1, 0)
(1, 1, 1)
复制代码

itertools.product,返回传入全部序列中笛卡尔积的元祖,repeat参数表示传入序列的重复次数。返回的是一个生成器。

那么获取全部的操做运算符就能够经过这个函数来获取了

def get_all_operations_sequence():
    operations = ['+','-','*','/']
    return product(operations,repeat=3)
复制代码

三、如今咱们已经拿到了全部可能组合的操做符和数字了,接下来就须要对他们进行拼接了。而后执行运算。

这一步操做咱们会用到 itertools.zip_longest()itertools.chain.form_iterable() 函数。

itertools.zip_longest() 用法演示

data = zip_longest([1,2,3,4],['*','-','+'],fillvalue='')
for value in data:
    print(value)
复制代码

结果显示

(1, '*')
(2, '-')
(3, '+')
(4, '')
复制代码

zip_longest() 其实和 python 内置的 zip() 函数用法差很少,只是 zip_longest 是以最长的一个序列为基准,缺失值就使用 fillvalue 参数的值进行填充

itertools.chain.form_iterable() 用法演示

data = zip_longest([1,2,3,4],['*','-','+'],fillvalue='')
data_chain = chain.from_iterable(data)
for value in data_chain:    
    print(value)
复制代码

结果显示

1
*
2
-
3
+
4
复制代码

这里的data是什么样的你们知道了吧,而后咱们将data传入 chain.form_iterable() 中,它就能将里面的值依次拿出来。

了解了这两个函数以后,那么咱们就能够开始拼接数字和操做运算符了。

def calculate(self):
    ''' 计算值,返回对应的表达式和值 :return: '''    
    for data_sequence in get_all_data_sequence():       
        operation_sequences = get_all_operation_sequence()       
        for operation_sequence in operation_sequences:            
            value = zip_longest(data_sequence, operation_sequence, 
        fillvalue='')            
            value_chain = chain.from_iterable(value)           
            calculate_str = ''           
            # 对获得的字符进行拼接成为表达式 calculate_str
            for _ in value_chain:                
                calculate_str += _          
            try:
                result = eval(calculate_str
            # 处理被除数可能为零的状况,而后就直接跳过此次循环
            except ZeroDivisionError:
                continue
            if math.isclose(result, 24):                    
               return calculate_str,result
    return None,None
复制代码

代码分析

一、eval() 函数,接受一个字符串,能让这个字符串当成 python 代码运行,返回运行的结果。

二、math.isclose():为何这里须要使用 math.isclose() ,而不是直接使用==运算符呢?这是由于最后算出来的表达式可能有精度问题,例如23.9...或者24.0...等数字,因此咱们就须要使用math.isclose()函数来帮助咱们判断两个数字是否相等了,这个函数就有一个精度范围。这样出现上面状况的时候,咱们也能匹配获得条件了。

咱们运行代码,而后测试代码是否能达到咱们的需求。

首先咱们测试1,2,3,4四个数字,

程序出来告终果 1*2*3*4 24

看来好像咱们写的代码是正确的


咱们再来测试一组数据8,8,3,3.

嗯?咱们并无获得结果?这四个数字不能运算出24吗?

8 / ( 3 - 8 / 3 ) 这样组合能够吧,为何没有算出来这种结果呢?


这是由于咱们没有考虑括号的缘由。括号是能够改变运算优先级的。因此咱们得把括号考虑进去。

那么想一下括号最多能够有几个呢?怎样给咱们的表达式添加括号呢?


在4个数字的运算中,括号最多只能有三个。

而且,在这里,咱们使用一种简单的方法添加括号,咱们把全部可能出现括号的状况所有罗列出来,而后在将获得的运算表达式拼接进去。

可能你们会以为罗列出全部括号出现的状况不现实,由于有不少状况

其实否则,当咱们去罗列的时候,你就会发现,只有11种状况。

FORM_STRS = [
    # 数字 运算符 数字 运算符 数字 运算符 数字
    # 一个括号 的状况
    '(%s %s %s) %s %s %s %s',
    '(%s %s %s %s %s) %s %s',
    '(%s %s %s %s %s %s %s)',
    '%s %s (%s %s %s) %s %s',
    '%s %s (%s %s %s %s %s)',
    '%s %s %s %s (%s %s %s)',
    # 两个括号 的状况
    '(%s %s %s) %s (%s %s %s)',
    '( (%s %s %s) %s %s) %s %s',
    '( %s %s (%s %s %s)) %s %s',
    '%s %s ((%s %s %s) %s %s)',
    '%s %s (%s %s (%s %s %s))',
    # 三个括号是重复的,就不用罗列出来了
]
复制代码

而后咱们对获得的表达式在进行遍历拼接,而后咱们再运算表达式。

这样咱们就能得出正确的结果了

代码写完了,终于能够开始和媳妇,哦不,老王家的媳妇玩起来了

代码已所有上传至Github:github.com/MiracleYoun…

关注公众号「Python专栏」,更多好玩有趣的Python等着你

相关文章
相关标签/搜索