python模拟春晚刘谦魔术

在今年的春晚上,刘谦表演了一个与"约瑟夫问题"相关的卡牌魔术。这两天忙来忙去,终于有空用代码来复现一下。

这是魔术的过程:

  • 步骤 1:将准备好的 4 张扑克牌平均撕成两份,并叠在一起。
  • 步骤 2:将牌堆顶数量为【名字字数】的牌移至牌堆底。
  • 步骤 3:将前三张牌放在牌堆中间并取出牌堆顶的牌,放置在一旁。
  • 步骤 4:取出牌堆顶的若干张牌插入牌堆中间,此处选择的牌数为南方人取 1 张,北方人取 2 张,若不确定是南方人还是北方人取 3 张。
  • 步骤 5:男生扔掉牌堆顶 1 张,女生扔掉牌堆顶 2 张。
  • 步骤 6:执行"见证奇迹的时刻"循环,每说一个字,就取出牌堆顶一张牌放置在牌堆底。
  • 步骤 7:从牌堆顶开始,每次先将牌堆顶的一张牌放在牌堆底,再扔掉牌堆顶的一张牌,重复以上操作直到只剩一张牌,检查此牌和放置在一旁的牌是否吻合。若吻合,则魔术成功。

抽卡

我们先模拟抽卡的过程,这里需要先从排队中抽取4张牌,这里我使用了python的randint模块,他可以生成随机整数,这里需要注意的是randint的取值是前闭后闭。这里要分别取得牌的数字和花色,需要注意的是,为了能区分结果,在抽牌时我们用 if pork not in porks 判断已抽取的牌堆中没有重复的牌:

python 复制代码
from random import randint

porks = []
while len(porks) < 4:
    num = randint(0, 12)
    ranks = [
        "Ace",
        "2",
        "3",
        "4",
        "5",
        "6",
        "7",
        "8",
        "9",
        "10",
        "Jack",
        "Queen",
        "King",
    ]
    colornnum = randint(0,4)
    color = ["♥", "♠", "♦", "♣"]

    pork = ranks[num] + color[colornnum]
    if pork not in porks:
        porks.append(pork)

要模拟撕卡的过程很简单,在代码中重复抽取的卡片即可。

python 复制代码
porks = porks + porks

根据名字移动卡片

第二步,我们要将牌堆顶数量为【名字字数】的牌移至牌堆底。这里我对名字的长度进行模拟,randint(2, 8) 随机生成2-8个字长度的名字。如果需要的话,也可以改成认为输入名字并判断长度。nameporks表示抽出的卡片,porks[num:] + nameporks把抽出点卡片放到牌堆底部。

python 复制代码
num = randint(2, 8)
print(f"名字的字数是{num}")

nameporks = porks[:num]
print(f"名字抽出的卡片是{nameporks}")
porks = porks[num:] + nameporks
print(f"此时的卡片是{porks}")

移动三张卡片和抽取卡片

这一步需要我们抽出牌堆顶部的三张牌,牌堆顶部的三张卡片并插入到剩下的牌堆中,其中我用headporks 代表抽出点头三张牌,bottomporks是剩下的牌。

num = randint(1, 4) 是要插入纸牌的位置,需要注意点是,这里我插入纸牌的位置要排除掉首位和末尾。在完成卡片的插入后,我们取出牌堆顶的第一张卡片。

python 复制代码
headporks = porks[:3]
bottomporks = porks[3:]

num = randint(1, 4)
print(f"插入扑克的位置是{num}")

porks = bottomporks[:num] + headporks + bottomporks[num:]
print(f"插入扑克卡片顺序是{porks}")

# 藏起卡片
print(f"删除的卡片是{porks[0]}")
porks = porks[1:]

抽出若干牌插入剩下的的牌堆

这里我通过生成随机数来判断是北方人、南方人还是其他,还是像之前做过的一样,通过生成随机数来判断是什么人num = randint(0, 2)

python 复制代码
num = randint(0, 2)

if num == 0:
    selectcards = porks[:1]
    porks = porks[1:]
    numpork = randint(1, 5)
    porks = porks[:numpork] + selectcards + porks[numpork:]
    print(num)
    print("南方人")
elif num == 1:
    selectcards = porks[:2]
    porks = porks[2:]
    numpork = randint(1, 4)
    porks = porks[:numpork] + selectcards + porks[numpork:]
    print(num)
    print("北方人")
elif num == 2:
    selectcards = porks[:3]
    porks = porks[3:]
    numpork = randint(1, 3)
    porks = porks[:numpork] + selectcards + porks[numpork:]
    print(numpork)
    print("外国人")
print(porks)

男生扔掉牌堆顶 1 张,女生扔掉牌堆顶 2 张。

这里根据男女,去掉顶部的牌,男生去掉一张,女生去掉三张。

python 复制代码
num = randint(0, 1)
if num == int(0):
    porks = porks[1:]
    print("性别为男")
elif num == int(1):
    porks = porks[2:]
    print("性别为女")
print(f"卡片上{porks}")

执行"见证奇迹的时刻"循环

这里循环七次即可,每次把顶部的第一张牌移动到底部。需要注意的是,在抽取牌的过程中,如果只抽出第一张牌,要人为增加一个列表[porks[0]]。

python 复制代码
for i in range(0, 7):

    headporks = [porks[0]]
    bottomporks = porks[1:]
    porks = bottomporks + headporks
    print(porks)

print(f"交换后的卡片是{porks}")

幸福留下来和烦恼丢出去

在这一步的过程中,如果牌堆中的牌数大于1,就进行循环,首先把顶部的牌放到底部,然后扔掉顶部的牌。

python 复制代码
while len(porks) > 1:
    headporks = [porks[0]]
    bottomporks = porks[1:]
    porks = bottomporks + headporks
    print(f"欢乐{porks}")

    # 烦恼
    porks = porks[1:]
    print(f"烦恼{porks}")

完整代码:

python 复制代码
from random import randint

# 步骤 1:将准备好的 4 张扑克牌平均撕成两份,并叠在一起。
porks = []
while len(porks) < 4:
    num = randint(0, 12)
    ranks = [
        "Ace",
        "2",
        "3",
        "4",
        "5",
        "6",
        "7",
        "8",
        "9",
        "10",
        "Jack",
        "Queen",
        "King",
    ]
    colornnum = randint(0, 3)
    color = ["♥", "♠", "♦", "♣"]

    pork = ranks[num] + color[colornnum]
    if pork not in porks:
        porks.append(pork)

print(porks)
porks = porks + porks


print(f"卡片顺序是{porks}")
# 步骤 2:将牌堆顶数量为【名字字数】的牌移至牌堆底。
num = randint(2, 8)
print(f"名字的字数是{num}")

nameporks = porks[:num]
print(f"名字抽出的卡片是{nameporks}")
porks = porks[num:] + nameporks
print(f"此时的卡片是{porks}")

# 步骤 3:将前三张牌放在牌堆中间并取出牌堆顶的牌,放置在一旁。
headporks = porks[:3]
bottomporks = porks[3:]

num = randint(1, 4)
print(f"插入扑克的位置是{num}")

porks = bottomporks[:num] + headporks + bottomporks[num:]
print(f"插入扑克卡片顺序是{porks}")

# 藏起卡片
print(f"删除的卡片是{porks[0]}")
porks = porks[1:]


print(porks)
# 南方人1,北方人2,其他3
num = randint(0, 2)

if num == 0:
    selectcards = porks[:1]
    porks = porks[1:]
    numpork = randint(1, 5)
    porks = porks[:numpork] + selectcards + porks[numpork:]
    print(num)
    print("南方人")
elif num == 1:
    selectcards = porks[:2]
    porks = porks[2:]

    numpork = randint(1, 4)

    porks = porks[:numpork] + selectcards + porks[numpork:]
    print(num)
    print("北方人")
elif num == 2:
    selectcards = porks[:3]
    porks = porks[3:]
    numpork = randint(1, 3)
    porks = porks[:numpork] + selectcards + porks[numpork:]
    print(numpork)
    print("外国人")
print(porks)
# 男生1,女生2
# 0 男生 1 女生

num = randint(0, 1)
if num == int(0):
    porks = porks[1:]
    print("性别为男")
elif num == int(1):
    porks = porks[2:]
    print("性别为女")
print(f"卡片上{porks}")

# 见证奇迹的时刻
for i in range(0, 7):

    headporks = [porks[0]]
    bottomporks = porks[1:]
    porks = bottomporks + headporks
    print(porks)

print(f"交换后的卡片是{porks}")


while len(porks) > 1:
    headporks = [porks[0]]
    bottomporks = porks[1:]
    porks = bottomporks + headporks
    print(f"欢乐{porks}")

    # 烦恼
    porks = porks[1:]
    print(f"烦恼{porks}")
print(porks)
相关推荐
西柚小萌新4 分钟前
【深入浅出PyTorch】--4.PyTorch基础实战
人工智能·pytorch·python
用户8356290780517 分钟前
掌控PDF页面:使用Python轻松实现添加与删除
后端·python
用户37215742613542 分钟前
Python 实现 Excel 文件加密与保护
python
Derrick__11 小时前
Python访问数据库——使用SQLite
数据库·python·sqlite
总有刁民想爱朕ha1 小时前
AI大模型学习(17)python-flask AI大模型和图片处理工具的从一张图到多平台适配的简单方法
人工智能·python·学习·电商图片处理
小虎鲸001 小时前
PyTorch的安装与使用
人工智能·pytorch·python·深度学习
加油吧zkf2 小时前
Python入门:从零开始的完整学习指南
开发语言·前端·python
杰瑞学AI2 小时前
我的全栈学习之旅:FastAPI (持续更新!!!)
后端·python·websocket·学习·http·restful·fastapi
用户3721574261352 小时前
Python 高效实现 Excel 与 CSV 互转:用自动化提升效率
python
CodeCraft Studio3 小时前
CAD文件处理控件Aspose.CAD教程:在 Python 中将 SVG 转换为 PDF
开发语言·python·pdf·svg·cad·aspose·aspose.cad