番外篇:分享一道用Python基础+蒙特卡洛算法实现排列组合的题目(附源码)

点击上方“Python爬虫与数据挖掘”,进行关注

回复“书籍”即可获赠Python从入门到进阶共10本电子书

夕阳无限好,只是近黄昏。

    大家好,我是Python进阶者。

    是不是觉得很诧异?明明上周刚发布了这篇:分享一道用Python基础+蒙特卡洛算法实现排列组合的题目(附源码),今天又来一篇,名曰番外篇!其实今天是想给大家分享【🌑(这是月亮的背面)】大佬的解法,拍案叫绝!

bfb052084956127984b2df1869bdd1eb.png

前情回顾

    前几天在才哥交流群里,有个叫【Rick Xiang】的粉丝在Python交流群里问了一道关于排列组合的问题,初步一看觉得很简单,实际上确实是有难度的。

    题目是:一个列表中有随机15个数,没有重复值。从列表里面任意选5个数,如何选出来包含a, a+1的所有组合。a可以是15个数中的任意一个。

b23fbe89c463a87d3a0cba8a61508879.png

    关于思路和解决方法,这篇文章分享一道用Python基础+蒙特卡洛算法实现排列组合的题目(附源码)中提供了【张老师】和【有点意思】大佬的想法和解决方案,一共有5份代码,足够大家学习了,感兴趣的小伙伴快去学习吧,干货满满。

二、新代码

    上周五的时候,发布了这篇分享一道用Python基础+蒙特卡洛算法实现排列组合的题目(附源码)原创文章,很庆幸还有粉丝亲自实践,并给出了建设性的方案,如下图所示。

b486eb394d08f4434e9473db69e659ae.png

    这里先给出【🌑(这是月亮的背面)】大佬的伪代码,这样看上去大家也更加好理解一些。

# -*- coding: utf-8 -*-
# 模块化
import random
import numpy as np
import time




# 取出随机的15个数值
def get_random15():
    random_array = [np.array(random.sample(range(2000), 15)) for i in range(100000)]
    random5 = {get_random5(random15) for random15 in random_array}
    return [i for i in random5 if i]




# 遍历随机的15个数值,取相邻的两个随机数,判断后返回满足条件的值
def get_random5(random_15):
    random_5 = set(random_15[random.sample(range(15), 5)])  # np.array的索引替换choice取值
    # 利用set特性判断元素是否含有给定的元素
    random_5_resp = {True if len(random_5.intersection({num, num + 1})) == 2 else False for num in random_5}
    return tuple(random_5) if True in random_5_resp else ()




if __name__ == '__main__':
    start_time = time.time()
    final_result = get_random15()
    print("共%d个符合题意的列表" % len(final_result))
    print("分别是:%s" % final_result)
    end_time = time.time()
    used_time = end_time - start_time
    print()
    print("本次程序用时:{}".format(time.strftime('%H(小时):%M(分钟):%S(秒)', time.gmtime(used_time))))

    这个代码写的真的很好,没有Python基础的小伙伴看上去肯定有些吃力的,小编自己初看的时候,也觉得有点难以吸收,需要多看几遍,领悟。

    这个代码亲测有效,用之前的代码大概需要12秒,改用这个只需要1.5秒。

552f017bcf746e1e969d946f4b7c04b4.png

    他这里做了三个优化,其一是之前从15个数中随机取5个值耗时较长,这里用使用了numpy.array的特性来优化代码,在科学计算中,可以省掉很多循环语句,代码使用方面比Python列表简单,Python list 无法直接运算, Numpy Array 可直接运算;其二是删除了之前的去重函数,这里他也用set去优化,所以在这块也节约了时间;其三是使用了集合的交集运算(Intersection),较之前的if判断来说,节约了时间。

    想到这里,不得不感叹一下,【人生苦短,我用python】!

三、总结

    我是Python进阶者。本文基于粉丝针对排列组合问题的提问,给出了一个利用Python基础+蒙特卡洛算法的解决方案,基本上可以达到了粉丝的要求。

    不过话说回来,这个方案虽是当下最优,但不是永远最优。

994dba96e571607f1d4dbb7fa6507708.png

    最后感谢【🌑(这是月亮的背面】大佬提出的改进算法建议和代码,感谢【Rick Xiang】提问,感谢【张老师】提供的思路,感谢【有点意思】大佬一直和我探讨交流学习,这一过程,虽然耗时,但是也是学到了很多知识。

    如果你也有其他的方法,也可以随时分享给我噢!人生苦短,我用python!

446e2383a4e1aa51bf85233dba209d03.png

    小伙伴们,快快用实践一下吧!如果在学习过程中,有遇到任何问题,欢迎加我好友,我拉你进Python学习交流群共同探讨学习。

2927789350ee77caa3fb89b72c97714d.png

------------------- End -------------------

往期精彩文章推荐:

8d0807f1101eda679d2b517dd5ad2c84.png

欢迎大家点赞,留言,转发,转载,感谢大家的相伴与支持

想加入Python学习群请在后台回复【入群

万水千山总是情,点个【在看】行不行

/今日留言主题/

随便说一两句吧~~

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值