技术路线

GUI的实现

  1. 使用PyQt技术作为基础。PyQt是一个支持多平台的客户端开发SDK,使用它实现的客户端可以运行在目前几乎所有主流平台之上。
  2. 使用PyQt,Qt设计器实现UI,通过pyuic4 -x -o main_page.py untitled.ui命令将设计好的xml文件转换为python程序。
  3. 继承纯UI的mainWindow类,重写setupUI方法,在UI建立完成之后进行控件的信号-槽的绑定。
  4. 在子类中完成各类事件的响应。

棋盘的绘制

  1. PyQt提供了丰富的控件,如TableView,虽然和棋盘有相似之处,但提供的API过于死板。所以我决定用Widget绘制一个棋盘。
  2. 建立一个QiPan类,继承于QtGui.QWidget 实现它的paintEvent接口,在此接口中可进行Widget的绘制。
  3. 将QiPan的长宽和棋盘格子数进行计算,绘制横纵的直线,完成棋盘的绘制。
  4. 将QiPan类实例化,得到的对象在主窗口setupUI的时候添加到上一层的容器内。

落子位置的判定

  1. 因为用户不可能每次恰好点中横纵线的交点处,所以要对用户点击的位置的坐标进行判定,判断用户想要落子到哪个点
  2. 如图,我将每个点的有效范围扩充至它到邻点的1/2组成的矩形。如图阴影区域: 
  3. 如此,棋盘中每个点都能映射到对应的交点上。

胜利条件判定算法

  1. 因为每次落棋都需要判定当前状态是否有一方胜利,所以胜利条件的判定应当在每次落棋之后进行。
  2. 又因为每次落棋时,落子的一方只有(赢/还没赢)两种状态,所以只需判断落子的一方是否赢得比赛,另一方的棋子无需判断。所以只需从当前落子的棋的4条线上进行判定是否连珠即可。
  3. 4条线分为8个方向,两两对称。因为情况很多,所以我构造了两个数组,dx和dy,如图:

  4. 将数组从[0,3][4,7]分为两部分,分别对应四条线每条线的两个方向。

  5. 如此只需循环遍历8个方向,每个方向出发的4个棋子子,统计这些棋子有多少和中心棋子颜色相同,每个方向的除以4取余数的存储计算结果。

  6. 如果某个方向上第x个棋子已经与中心棋子不同,立即跳出循环,不再继续遍历该方向。

  7. 核心判断代码:

def check_line(self, x, y, map, width, height):
tag = map[x][y]
dx = [+1, -1, 0, -1, -1, + 1, 0, +1]
dy = [-1, 0, -1, -1, +1, 0, +1, +1]
count = [0] * 4
for k in xrange(0, 8):
for i in xrange(1, 5):
if 0 < x + dx[k] * i < width and 0 < y + dy[k] * i < height and map[x + dx[k] * i][y + dy[k] * i] == tag:
count[k % 4] += 1
else:
break
for k in count:
if k == 4:
return True
return False

悔棋功能的实现

  1. 建立两个栈,分别用作黑棋和白棋的棋子状态,在每次落子时push进该棋子的状态。
  2. 当用户点击悔棋按钮时,从相应的栈中pop出来,恢复上一状态。

工程结构

采用MVC的模式

  • build: 生成可执行文件目录
  • controller:UI的相关控制,相当于C的作用

    1. MyMainWindow.py:继承Ui_MainWindow类,主要控制主窗口
    2. QiPan.py:棋盘类,继承于QWidget,用于棋盘的相关控制
  • libs: 各项判定条件的算法,充当了一部分Model的作用

    1. AI.py:AI类,用来判定输赢条件
    2. BitmapTools.py:封装了对落子在棋盘map中的位置的判定方法
  • resource:图片资源文件夹

  • ui:UI相关代码,View的作用

    1. main_page.py:UI界面的python代码
    2. untitled.ui:UI界面的xml代码

阿萨德 - app.py:主函数入口,用于启动MainWindow

设计模式

单例模式

AI类和BitMapTools都采用单例模式: 好处在于可以只用实例化一次,占用资源少。在棋盘初始化的时候就加载这两个工具类,将它们的实例保存为成员,以后可以频繁地使用这个实例。

class BitMapTools:
tools = None
@staticmethod
def getinstance():
if BitMapTools.tools is None:
BitMapTools.tools = BitMapTools()
return BitMapTools.tools

观察者模式

继承UI相关的类,并重写setupUI,在其中订阅各类点击事件,在相关的函数中处理这些事件发生之后的工作。

self.father.restart_btn.clicked.connect(self.restart)

这样每一个事件对应一个函数,使得结构更加清晰。

from: PyQt写的五子棋

blog: http://cyhhao.zhusun.in/

PyQt写的五子棋的更多相关文章

  1. PyQt写的浏览单web页面的browser - 开源中国社区

    PyQt写的浏览单web页面的browser - 开源中国社区 PyQt写的浏览单web页面的browser

  2. python使用pyqt写带界面工具

    上篇介绍的使用python自带tkinter包,来写带界面的工具. 此篇介绍使用pyqt来开发测试工具. tkinter的好处是python官方自带,上手容易(但手写控件复杂),布局和摆放都不直观和容 ...

  3. 使用pyqt写了一个检查大数据环境的gui

    通过pyqt做了一个大数据最佳实践检查的gui界面 1.首先是需要用到的模块 from PyQt5.QtWidgets import QApplication from PyQt5.QtWidgets ...

  4. 用C++基础语句写一个五子棋游戏

    (这是一个颜色会变化的呦) #include <iostream> using namespace std; int b[][]; int n; int m; void qipan() { ...

  5. 基于Blazor写一个简单的五子棋游戏

    写这个五子棋游戏,其实主要目的是想尝试一下微软新作Blazor.Blazor对于那些搞.NET的程序员,又想做一些前端工作,真的挺友好,不用一句JS就可搞定前端交互,美哉.现在已经有很流行的前端框架, ...

  6. 【源码分享】iOS-OC版五子棋

    五子棋是程序猿比较熟悉的一款小游戏,相信很多人大学时期就用多种语言写过五子棋小游戏,笔者工作闲暇之余,试着用OC实现了一下,在这里给大家分享一下.有不足之处,欢迎大家提供建议和指点! GitHub源码 ...

  7. 五子棋-b

    五子棋是程序猿比较熟悉的一款小游戏,相信很多人大学时期就用多种语言写过五子棋小游戏.笔者工作闲暇之余,试着用OC实现了一下,在这里给大家分享一下.有不足之处,欢迎大家提供建议和指点!!!GitHub源 ...

  8. 纯CSS3之五子棋(黑白棋)画法

    无聊想用JS写个五子棋玩玩,这边先用CSS3画出了五子棋,感觉挺惊艳的.发上来看看 PS:第一次发博文T-T  此为个人原创. 不多说上图: 代码如下: <!DOCTYPE html> & ...

  9. C++ 之 简单的五子棋AI程序

    本人是大一新生,寒假无聊,抱着试试看的心态(没有想到可以完成),写了C++的简单五子棋程序,开心.     下面是效果图:     一.首先讲讲大致思路.            五子棋实现的基础:  ...

随机推荐

  1. Amazon AWS EC2开启Web服务器配置

    在Amazon AWS EC2申请了一年的免费使用权,安装了CentOS + Mono + Jexus环境做一个Web Server使用. 在上述系统安装好之后,把TCP 80端口开启(iptable ...

  2. 我的ORM之六-- 批量

    我的ORM索引 很多时候,批量方案是解决性能的关键 批量插入 SqlBulkCopy 利用 SqlBulkCopy 是性能最高的方式 实现: var s = dbr.Menu.NewMyOqlSet( ...

  3. Linux 网络编程(IO模型)

    针对linux 操作系统的5类IO模型,阻塞式.非阻塞式.多路复用.信号驱动和异步IO进行整理,参考<linux网络编程>及相关网络资料. 阻塞模式 在socket编程(如下图)中调用如下 ...

  4. Java设计模式3:工厂方法模式

    工厂方法模式 工厂方法模式是类的创建模式.工厂方法模式的用意是定义一个创建产品对象的工厂接口,将实际创建工厂推迟到子类中. 工厂方法模式 工厂方法模式是对简单工厂模式进一步抽象的结果. 假如是不使用反 ...

  5. Wix 安装部署教程(七) 获取管理员权限

    应用程序运行的时候,难免会读写文件,产生新的数据.但Program Files下的文件是不能随便更改,Win7下如果没有权限,将会被拒绝.我现在有两种方式,一种是将数据路径移到Program Data ...

  6. Android怎么找到最优适配资源

    当我们将一些提供了不同的资源文件可供Android系统选择的时候,Android会在运行时会根据一套适配的规则选择最符合当前配置的资源.为了说明Android怎么选择资源,假设我们有以下可选的资源文件 ...

  7. Ehcache BigMemory: 摆脱GC困扰

    问题 使用java开源项目经常需要调优jvm,以优化gc.对于gc,如果对象都是短时对象,那么jvm相对容易优化,假如碰上像solr使用自带java cache的项目,那么gc严重受限于cache,因 ...

  8. [译]C++, Java和C#的编译过程解析

    1.1.1 摘要 我们知道计算机不能直接理解高级语言,它只能理解机器语言,所以我们必须要把高级语言翻译成机器语言,这样计算机才能执行高级语言编写的程序,在接下来的博文中,我们将介绍非托管和托管语音的编 ...

  9. .NET学习笔记 -- 那堆名词到底是啥(CLR、CLI、CTS、CLS、IL、JIT)

    什么是CLR? CLR,公共语言运行时(Common Language Runtime)是一个由多种语言使用的“运行时”.他的核心功能包括(内存管理.程序集加载.安全性.异常处理和线程同步),可以被面 ...

  10. MVVM架构~knockoutjs实现简单的购物车

    返回目录 概念相关 购物车相信大家都用过,很方便,可以将多个商品添加到购物车,并且可以修改购买商品的数据,当然为了用户体验好,在修改数据时,你的价格也会出现变化的,这使用JS可以实现,但我认为,代码量 ...