python 小游戏《2048》字符版非图形界面

参考链接: 闲谈2048小游戏和数组的旋转及翻转和转置

目录

2048 

一、方阵类

二、随机插入1或2

三、 合并和递增

四、 判断和移动

五、 键盘控制

完整源代码

玩法过程


2048 

上回说到2048小游戏中数组的各种旋转、翻转的方法,就是为代码编程作准备的;有了这些再就加上二维数组各行列上元素的合并、能否被合并的判断、成功失败的判断等等;以及再加上键盘按键的控制,小游戏就基本完成了。

一、方阵类

方阵就是高宽相同的矩阵,2048用方阵就行了,写代码也省事一点,方阵的类如下:

>>> from random import sample
>>> class Matrix:
...     def __init__(self, order=4):
...         self.order = order
...         self.matrix = self.new()
...     def __repr__(self):
...         m, n = [], len(str(2**max(sum(self.matrix,[]))))
...         for mat in self.matrix:
...             m.append(', '.join(f'{2**x if x else 0:>{n}}' for x in mat))
...         return '],\n ['.join(m).join(['[[',']]'])
...     def new(self):
...         n = self.order
...         m = sample([0]*(n*n-2)+sample([0,1,1],2),n*n)
...         return  [m[i*n:i*n+n] for i in range(n)]
... 
...     
>>> Matrix()
[[0, 0, 2, 0],
 [0, 0, 0, 0],
 [0, 0, 2, 0],
 [0, 0, 0, 0]]
>>> Matrix()
[[0, 0, 0, 0],
 [0, 0, 0, 0],
 [0, 0, 2, 0],
 [0, 0, 0, 0]]

在方阵中随机产生1~2个1,sample([0,1,1],2) 生成的1个还是2个,比例为2:1;

在__repr__方法中显示时,这些1作为2的指数,所以显示为2^1=2。

二、随机插入1或2

    def insert(self):
        n = self.order
        m = [i for i,n in enumerate(sum(self.matrix,[])) if not n]
        if m:
            i = sample(m, 1)[0]
            self.matrix[i//n][i%n] = sum(sample([1,1,1,2],1))

或者:

    def insert(self):
        n = self.order
        m = [(i,j) for j in range(n) for i in range(n) if not self.matrix[i][j]]
        if m:
            i, j = sample(m, 1)[0]
            self.matrix[i][j] = sum(sample([1,1,1,2],1))

三、 合并和递增

   ......     
   for i,array in enumerate(self.matrix):
            self.matrix[i] = Matrix.update(array)
    ......
    def update(array):
        split = lambda a: [_ for _ in a if _]+[_ for _ in a if not _]
        array = split(array)
        for i,a in enumerate(array):
            if i and a and a==array[i-1]:
                array[i-1] += 1
                array[i] = 0
        return split(array)

四、 判断和移动

略……写得有点复杂,可以到完整代码中阅读。

五、 键盘控制

引入keyboard库控制键盘,示例如下:

import keyboard  
  
def keys0():  
    print("Left key pressed")  
  
def keys1():  
    print("Right key pressed")  
  
def keys2():  
    print("Up key pressed")  
  
def keys3():  
    print("Down key pressed")  
  
def restart():  
    print("Enter key pressed")  
  
# 添加热键  
keyboard.add_hotkey('left', keys0)  
keyboard.add_hotkey('right', keys1)  
keyboard.add_hotkey('up', keys2)  
keyboard.add_hotkey('down', keys3)  
keyboard.add_hotkey('enter', restart)  
  
# 等待用户按下esc键  
print("Waiting for ESC to exit...")  
keyboard.wait('esc')  
  
# 在这里移除所有热键  
print("Removing all hotkeys...")  
keyboard.unhook_all_hotkeys()

Waiting for ESC to exit...
Left key pressed
Right key pressed
Up key pressed
Down key pressed
Enter key pressed
Removing all hotkeys...
>>> 

注:最后一行代码keyboard.unhook_all_hotkeys()很关键,一定都有否则即使按了ESC键退出程序,热键还是驻留在内存里,键盘还会响应keyboard.add_hotkey()添加的热键。


完整源代码

import keyboard
from random import sample

class Matrix:
    def __init__(self, order=4):
        self.over = False
        self.order = order
        self.matrix = self.new()
        self.victory = False
    def __repr__(self):
        m, n = [], len(str(2**max(sum(self.matrix,[]))))
        for mat in self.matrix:
            m.append(', '.join(f'{2**x if x else 0:>{n}}' for x in mat))
        return '],\n ['.join(m).join(['[[',']]'])
    def new(self):
        n = self.order
        m = sample([0]*(n*n-2)+sample([0,1,1],2),n*n)
        return  [m[i*n:i*n+n] for i in range(n)]
    def show(self):
        if self.over or self.victory:
            print('Enter to restart...')
        else:
            print(self)
            print()
    def insert(self):
        n = self.order
        m = [(i,j) for j in range(n) for i in range(n) if not self.matrix[i][j]]
        if m:
            i, j = sample(m, 1)[0]
            self.matrix[i][j] = sum(sample([1,1,1,2],1))
    def full(self):
        return all(sum(self.matrix,[]))
    def move(self, direction=0):
        if self.over or self.victory: return
        direction %= 4
        if direction == 0:  #left
            if self.cannotmove(0): return
        elif direction == 1:  #right
            if self.cannotmove(1): return
            self.matrix = self.flipH()
        elif direction == 2:  #up
            if self.cannotmove(2): return
            self.matrix = self.rotL()
        elif direction == 3:  #down
            if self.cannotmove(3): return
            self.matrix = self.rotR()
        for i,array in enumerate(self.matrix):
            self.matrix[i] = Matrix.update(array)
        if direction == 1:
            self.matrix = self.flipH()
        elif direction == 2:
            self.matrix = self.rotR()
        elif direction == 3:
            self.matrix = self.rotL()
        indexmax = max(sum(self.matrix,[]))
        if self.order==2 and indexmax==4 or self.order==3 and indexmax==7 or indexmax==self.order+7:
            self.victory = True
            print(self)
            print('Win! Enter to restart...')
            return
        self.insert()
        self.over = self.cannotmove()
        if self.over:
            print(self)
            print('Gave over!')
    def cannotmove(self, direction = 4):
        m, n = self.matrix, self.rotR()
        p, q = self.rotL(), self.flipH()
        if direction==0:
            return all(n[0]) and Matrix.cannotupdate(m)
        elif direction==1:
            return all(n[-1]) and Matrix.cannotupdate(q)
        elif direction==2:
            return all(m[0]) and Matrix.cannotupdate(n)
        elif direction==3:
            return all(m[-1]) and Matrix.cannotupdate(p)
        else:
            return (self.full() and self.cannotmove(0) and self.cannotmove(1)
                and self.cannotmove(2) and self.cannotmove(3))
    def cannotupdate(matrix):
        return all([m==Matrix.update(m) for m in matrix])
    def update(array):
        split = lambda a: [_ for _ in a if _]+[_ for _ in a if not _]
        array = split(array)
        for i,a in enumerate(array):
            if i and a and a==array[i-1]:
                array[i-1] += 1
                array[i] = 0
        return split(array)
    def flipH(self):
        m, n = self.matrix, self.order
        return [[m[i][n-j-1] for j in range(n)] for i in range(n)]
    def flipV(self):
        m, n = self.matrix, self.order
        return [[m[n-j-1][i] for i in range(n)] for j in range(n)]
    def rotL(self):
        m, n = self.matrix, self.order
        return [[m[j][n-i-1] for j in range(n)] for i in range(n)]
    def rotR(self):
        m, n = self.matrix, self.order
        return [[m[n-j-1][i] for j in range(n)] for i in range(n)]

def move(i):
    mat.move(i)
    mat.show()
def keys0():  
    move(0)
def keys1():  
    move(1)
def keys2():  
    move(2)
def keys3():  
    move(3)
def restart():
    global mat
    if mat.victory:
        mat.order += 1
        mat = Matrix(mat.order)
        mat.show()
    if mat.over:
        if mat.order>3:
            mat.order -= 1
        mat = Matrix(mat.order)
        mat.show()

if __name__ == '__main__':
    print("《2048小游戏》")
    print("上下左右前头控制方向,按ESC退出...") 
    mat = Matrix(2)
    mat.show()
    keyboard.add_hotkey('left', keys0)
    keyboard.add_hotkey('right', keys1) 
    keyboard.add_hotkey('up', keys2)
    keyboard.add_hotkey('down', keys3)
    keyboard.add_hotkey('enter', restart)
    keyboard.wait('esc')  # 等待用户按下esc键退出
    print('bye!')
    keyboard.unhook_all_hotkeys()  # 退出后移除所有热键

玩法过程

《2048小游戏》
上下左右前头控制方向,按ESC退出...
[[0, 0],
 [0, 2]]

[[0, 0],
 [2, 2]]

[[0, 4],
 [0, 4]]

[[0, 2],
 [0, 8]]

[[2, 2],
 [8, 0]]

[[4, 2],
 [8, 0]]

[[4, 4],
 [8, 2]]

[[8, 2],
 [8, 2]]

[[ 0,  0],
 [16,  4]]
Win! Enter to restart...
Enter to restart...
[[0, 2, 0],
 [0, 0, 2],
 [0, 0, 0]]

[[0, 0, 0],
 [2, 0, 0],
 [0, 2, 2]]

[[4, 0, 0],
 [0, 0, 2],
 [0, 0, 4]]

[[0, 2, 0],
 [0, 0, 2],
 [4, 0, 4]]

[[0, 0, 2],
 [0, 0, 2],
 [4, 0, 8]]

[[0, 0, 2],
 [0, 2, 2],
 [0, 4, 8]]

[[0, 0, 0],
 [0, 2, 4],
 [2, 4, 8]]

[[2, 0, 0],
 [2, 4, 0],
 [2, 4, 8]]

[[2, 0, 0],
 [2, 0, 0],
 [4, 8, 8]]

[[ 0,  2,  2],
 [ 0,  0,  2],
 [ 0,  4, 16]]

[[ 0,  0,  0],
 [ 2,  2,  4],
 [ 0,  4, 16]]

[[ 0,  0,  0],
 [ 0,  4,  4],
 [ 4,  4, 16]]

[[ 2,  0,  0],
 [ 0,  0,  4],
 [ 4,  8, 16]]

[[ 2,  0,  0],
 [ 2,  0,  4],
 [ 4,  8, 16]]

[[ 2,  0,  0],
 [ 4,  0,  4],
 [ 4,  8, 16]]

[[ 0,  4,  0],
 [ 2,  0,  4],
 [ 8,  8, 16]]

[[ 0,  2,  4],
 [ 0,  2,  4],
 [ 0, 16, 16]]

[[ 2,  2,  4],
 [ 0,  2,  4],
 [ 0,  0, 32]]

[[ 0,  0,  0],
 [ 2,  0,  8],
 [ 2,  4, 32]]

[[ 0,  0,  0],
 [ 2,  0,  8],
 [ 4,  4, 32]]

[[ 0,  0,  2],
 [ 0,  2,  8],
 [ 0,  8, 32]]

[[ 0,  0,  2],
 [ 0,  2,  8],
 [ 2,  8, 32]]

[[ 2,  0,  4],
 [ 2,  8,  0],
 [ 2,  8, 32]]

[[ 0,  0,  2],
 [ 2,  0,  4],
 [ 4, 16, 32]]

[[ 2,  0,  2],
 [ 2,  4,  0],
 [ 4, 16, 32]]

[[ 0,  0,  4],
 [ 4,  4,  2],
 [ 4, 16, 32]]

[[ 2,  0,  4],
 [ 0,  4,  2],
 [ 8, 16, 32]]

[[ 2,  2,  4],
 [ 0,  4,  2],
 [ 8, 16, 32]]

[[ 0,  4,  4],
 [ 2,  4,  2],
 [ 8, 16, 32]]

[[ 2,  0,  4],
 [ 2,  8,  2],
 [ 8, 16, 32]]

[[ 0,  2,  4],
 [ 4,  8,  2],
 [ 8, 16, 32]]

[[ 2,  4,  2],
 [ 4,  8,  2],
 [ 8, 16, 32]]

[[ 2,  4,  2],
 [ 4,  8,  4],
 [ 8, 16, 32]]
Gave over!
Enter to restart...

注:二阶方阵玩出16就过关,三阶方阵玩出128就过关,四阶就要到2048才能过关,五阶则要到4096才过关,六阶以上类推。演示时玩到三阶方阵就over了,如果三阶方阵过关会自动升级到四阶方阵,以此类推。


本文完

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/418746.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

单词规律00

题目链接 单词规律 题目描述 注意点 pattern只包含小写英文字母s只包含小写英文字母和 ’ ’s不包含任何前导或尾随对空格s中每个单词都被 单个空格 分隔 解答思路 本题与上一次同构字符串类似,思路可以参照同构字符串 代码 class Solution {public boolean …

C++——内存管理(new和delete)详解

目录 C/C内存管理 案例:变量在内存中到底会在哪? New和delete Operator new和operator delete函数 New和delete的原理 对内置类型 对自定义类型 定位new New/delete和malloc/free的区别 C/C内存管理 C/C内存管理分布图:(从…

C语言while 语句的基本格式是什么?

一、问题 C语⾔中有三种循环语句,while 语句是其中的⼀个,它的基本格式是怎样的呢? 二、解答 while 语句的⼀般形式为: while(表达式) 语句; 其中,表达式是循环条件,语句为循环体。 注意: …

uni-app 微信小程序:启用组件按需注入

原文地址:uni-app 微信小程序:启用组件按需注入 - 掘金 只需添加以下代码"lazyCodeLoading": "requiredComponents"

养娃到大学毕业要68万?不算不知道一算要起跳!

最近看到的新话题,第一反映就是养一个娃要这么多费用吗? 以我马上就要上大学的娃家长身份,索性粗针大线的做了一个预估,让我们一起来探讨一下吧: 现正在考虑娃是可以读国内正常的公办大学还是能读国内的合作办学大学…

【机器学习实战1】泰坦尼克号:灾难中的机器学习(一)数据预处理

🌸博主主页:釉色清风🌸文章专栏:机器学习实战🌸今日语录:不要一直责怪过去的自己,她曾经站在雾里也很迷茫。 🌼实战项目简介 本次项目是kaggle上的一个入门比赛 :Titani…

护眼灯到底有用吗知乎?5大超强实力护眼台灯测评推荐!

护眼台灯无疑是每个学子必备的学习用具,那到底有没有用呢?其实真正好的护眼灯在一定程度上对保护学生的眼睛来说肯定是有用的。当然前提是要选购到优质的护眼台灯,保障各项性能都是合格的,使用起来才安全放心。 一、护眼台灯是如何…

Python web框架fastapi数据库操作ORM(一)

文章目录 Fastapi ORM操作1、创建模型2、创建数据库连接配置文件3、启动项目4、根据模型类创建数据库表1. 初始化配置,只需要使用一次2. 初始化数据库,一般情况下只用一次3. 更新模型并进行迁移4. 重新执行迁移,写入数据库5. 回到上一个版本6…

uniapp+vue基于Android的图书馆借阅系统qb4y3-nodejs-php-pyton

uni-app框架:使用Vue.js开发跨平台应用的前端框架,编写一套代码,可编译到Android、小程序等平台。 框架支持:springboot/django/php/Ssm/flask/express均支持 前端开发:vue 语言:pythonjavanode.jsphp均支持 运行软件:idea/eclip…

【Javascript编程实操01】判断最大数、奇偶数、是否成年

目录 前言 1、求两个数的最大数 代码: 实现效果: 2、判断一个整数是偶数还是奇数 代码: 实现效果: 3、判断一个人的年龄是否满18岁 代码: 实现效果: 总结 前言 从今天开始正式进入了Web前端第二…

GitLab--Merge Request 权限管理

场景 团队在日常开发工作中需要进行分支管理,通常使用feature分支进行开发,然后依次合并到dev分支、release分支,整个代码合并过程不仅仅是代码合并还需要对代码进行审核,如果在线下进行审核合并,这样操作无法保留痕迹…

反射详解-获取构造方法-动态代理

反射是Java语言中的一个关键特性,它允许程序在运行时访问类、接口、字段和方法的信息,并且可以动态地调用方法或改变字段值。 以下是关于Java反射的一些主要方面: 类加载器(Class Loader): Java反射开始于…

内网穿透 nas/树莓派+ipv4服务器 (ipv6)

nas 1.有个服务器 2.有个nas https://github.com/snail007/goproxy/blob/master/README_ZH.md https://github.com/snail007/proxy_admin_free/blob/master/README_ZH.md 2个官网一个是程序,一个是网站 手册 https://snail007.host900.com/goproxy/manual/zh/#/?i…

Tiktok矩阵系统搭建的逻辑和源代码!

很多和我一样从事外贸工具开发的朋友都清楚,TikTok矩阵系统不仅确保了平台的高效运行,还为用户提供了个性化的内容推荐,从而大大提升了用户黏性,因此很多人都乐意去开发类似的工具,下面我们就来说说Tiktok矩阵系统搭建…

Vue3 在SCSS中使用v-bind

template 先创建一个通用的页面结构 <template><div class"v-bubble-bg"></div> </template>js 在JS中先对需要用的数据进行定义&#xff1a; 可以是参数&#xff0c;也可以是data <script setup>const props defineProps({bgCol…

[万字长文] 从 Vue 3 的项目模板学习 tsconfig 配置

文章目录 一、tsconfig.json 的作用二、基本介绍三、Vue 3 的 tsconfig.json 的结构分析1. 总配置 tsconfig.json2. Web 侧 tsconfig.app.jsona. 继承基础配置b. 包含和排除的文件c. 编译器选项 3. 测试 tsconfig.vitest.jsona. 继承的基础配置b. 包含和排除的文件c. 编译器选项…

Swiper实现轮播效果

swiper官网&#xff1a;https://3.swiper.com.cn/ <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title&…

苹果的“汽车梦”宣告失败,转战AI?能hold住吗?

文 | BFT机器人 一觉睡醒&#xff0c;苹果就宣布了一件爆炸性的新闻&#xff0c;那就是坚持多年和受到很多行业大佬支持和期待的“泰坦”宣布结束了&#xff01;将原有负责汽车项目的大部分主力军转战到AI核心上&#xff0c;这消息一出&#xff0c;业内外纷纷对苹果的这个决定…

仓储自动化新解:托盘四向穿梭车驶入智能工厂 智能仓储与产线紧密结合

目前&#xff0c;由于对仓库存储量的要求越来越高&#xff0c;拣选、输送以及出入库频率等要求也越来越高&#xff0c;对此&#xff0c;在物流仓储领域&#xff0c;自动化与智能化控制技术得以快速发展&#xff0c;货架穿梭车在自动库领域的应用越来越广泛。现阶段&#xff0c;…

从 iOS 设备恢复数据的 20 个iOS 数据恢复工具

作为 iPhone、iPad 或 iPod 用户&#xff0c;您可能普遍担心自己可能会丢失存储在珍贵 iOS 设备中的所有宝贵数据。数据丢失的原因多种多样&#xff0c;这里列出了一些常见原因&#xff1a; 1. iOS 软件更新 2. 恢复出厂设置 3. 越狱 4. 误操作删除数据 5. iOS 设备崩溃 …