百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

游戏百解——利用Python图像识别玩连连看,手把手教你成为大师!

off999 2025-05-30 16:54 28 浏览 0 评论

这是我自己用程序写的视频,利用Python图像识别算法玩转连连看。感兴趣可以自己看一下。

游戏百解——连连看(大神版)

前言:程序主要功能是先将练练看的整个大图切分成单个小图,然后进行循环遍历找出相似的图片,并在矩阵中进行记录,然后依据练练看两个图片连接的规则进行连接。直接放代码:

一、连连看连接逻辑程序:

import time
from pymouse import *
from LLKgame.baseFunctions import cutPicture, makeArray
class run_game():
	#初始截图左上右下坐标位置,初始截图总宽总高,初始单个图像宽高
	def __init__(self,left,top,right,botton,pWidth,pHeight,height,width):
		self.left = left
		self.top = top
		self.right = right
		self.botton = botton
		self.pWidth = pWidth
		self.pHeight = pHeight
		self.width = width
		self.height = height
		self.im2num_arr = []

	#点击事件
	def pClick(self, x1, y1, x2, y2):
		m = PyMouse()
		p1_x = int(self.left + (y1) * self.width - int((self.width / 2)))
		p1_y = int(self.top + (x1) * self.height - int((self.height / 2)))
		p2_x = int(self.left + (y2) * self.width - int((self.width / 2)))
		p2_y = int(self.top + (x2) * self.height - int((self.height / 2)))

		time.sleep(0.3)#沉睡时间
		m.click(int(p1_x / 1.25), int(p1_y / 1.25))
		time.sleep(0.3)
		m.click(int(p2_x / 1.25), int(p2_y / 1.25))
		# time.sleep(0.1)
		# 设置矩阵值为0
		self.im2num_arr[x1][y1] = 0
		self.im2num_arr[x2][y2] = 0
		print("消除:(%d, %d) (%d, %d)" % (x1, y1, x2, y2))

	# 是否为同行或同列且可连
	#X1 Y1 元素坐标x值以及y值 X2 Y2 元素坐标x值 y值
	def isReachable(self, x1, y1, x2, y2):
		# 1、先判断值是否相同
		if self.im2num_arr[x1][y1] != self.im2num_arr[x2][y2]:
			return False
		# 判断横向连通
		if self.isSameRow(x1, y1, x2, y2):
			return True
		# 判断纵向连通
		if self.isSameCol(x1, y1, x2, y2):
			return True
		# 判断一个拐点可连通
		if self.turnOnceCheck(x1, y1, x2, y2):
			return True
		# 判断两个拐点可连通
		if self.turnTwiceCheck(x1, y1, x2, y2):
			return True
		# 不可联通返回False
		return False

	#是否两个元素同行
	def isSameRow(self, x1, y1, x2, y2):
		if (abs(y1 - y2) == 1 and x1 == x2):  # 同行且相邻
			return True

		if (abs(y1 - y2) > 1 and x1 == x2):  # 同行不相邻
			# if (x1 == 0 or x1 == (self.pHeight - 1)):  # 最外行
			#     return True
			flag = 0
			for i in range(min(y1, y2) + 1, max(y1, y2)):
				if self.im2num_arr[x1][i] == 0:
					flag = flag + 0
				else:
					flag = flag + 1
			if (flag == 0):
				return True
		return False

	#是否同列
	def isSameCol(self, x1, y1, x2, y2):
		if (abs(x1 - x2) == 1 and y1 == y2):  # 同列且相邻
			return True

		if (abs(x1 - x2) > 1 and y1 == y2):  # 同列不相邻
			# if (y1 == 0 or y1 == (self.pWidth - 1)):  # 在最外列
			#     return True
			flag = 0
			for i in range(min(x1, x2) + 1, max(x1, x2)):
				if self.im2num_arr[i][y1] == 0:
					flag = flag + 0
				else:
					flag = flag + 1
			if (flag == 0):
				return True
		return False

	# 判断一个拐点可联通
	def turnOnceCheck(self, x1, y1, x2, y2):
		if x1 == x2 or y1 == y2:
			return False
		cx = x1
		cy = y2
		dx = x2
		dy = y1
		# 拐点为空,从第一个点到拐点并且从拐点到第二个点可通,则整条路可通。
		if self.im2num_arr[cx][cy] == 0:
			if self.isSameRow(x1, y1, cx, cy) and self.isSameCol(cx, cy, x2, y2):
				return True
		if self.im2num_arr[dx][dy] == 0:
			if self.isSameRow(x1, y1, dx, dy) and self.isSameCol(dx, dy, x2, y2):
				return True
		return False

	# 判断两个拐点可联通
	def turnTwiceCheck(self, x1, y1, x2, y2):
		if x1 == x2 and y1 == y2:
			return False
		# 遍历整个数组找合适的拐点
		for i in range(0, len(self.im2num_arr)):
			for j in range(0, len(self.im2num_arr[1])):
				# 不为空不能作为拐点
				if self.im2num_arr[i][j] != 0:
					continue
				# 不和被选方块在同一行列的不能作为拐点
				if i != x1 and i != x2 and j != y1 and j != y2:
					continue
				# 作为交点的方块不能作为拐点
				if (i == x1 and j == y2) or (i == x2 and j == y1):
					continue
				if self.turnOnceCheck(x1, y1, i, j) and (
						self.isSameRow(i, j, x2, y2) or self.isSameCol(i, j, x2, y2)):
					return True
				if self.turnOnceCheck(i, j, x2, y2) and (
						self.isSameRow(x1, y1, i, j) or self.isSameCol(x1, y1, i, j)):
					return True
		return False

	# 判断矩阵是否全为0
	def isAllZero(self, arr):
		for i in range(0, self.pWidth + 2):
			for j in range(0, self.pHeight + 2):
				if arr[i][j] != 0:
					return False
		return True

	def start(self):
		# 3、遍历查找可以相连的坐标
		global num1
		global num2
		num1 = 0#循环判断
		num2 = 0 #结束符
		print(self.im2num_arr)
		while not self.isAllZero(self.im2num_arr):
			if num1 == 10:
				for i in range(0, self.pWidth + 2):
					for j in range(0, self.pHeight + 2):
						self.im2num_arr[i][j] = 0
				num2 = 1
			for x1 in range(0, self.pWidth + 2):
				for y1 in range(0, self.pHeight + 2):
					if self.im2num_arr[x1][y1] == 0:
						continue
					for x2 in range(0, self.pWidth + 2):
						for y2 in range(0, self.pHeight + 2):
							if self.im2num_arr[x2][y2] == 0 or (x1 == x2 and y1 == y2):
								continue
							if self.im2num_arr[x1][y1] == self.im2num_arr[x2][y2]:
								if self.isReachable(x1, y1, x2, y2):
									self.pClick(x1, y1, x2, y2)
			num1 = num1 + 1
		return num2
if __name__ == '__main__':
	t = run_game(304, 271, 1226, 870,10,14,60.2,65.9)#设置元素坐标
	time.sleep(3)#等待3秒
	image = cutPicture(304, 271, 1226, 870,10,14,60.2,65.9)#截图
	arr = makeArray(10,14,image)#设置矩阵
	t.im2num_arr = arr
	num = t.start()
	print("游戏结束!")

二、选择一种图像识别算法,我这里写了四种,我是四种混着用,有两种算法通过则判断两个连连看图片一致,提高精准度。

import cv2
import numpy as np
import matplotlib
matplotlib.use('TkAgg')

# 均值哈希算法
def aHash(img):
    # 缩放为8*8
    # img = cv2.imread(img)
    img = np.asanyarray(img)
    img = cv2.resize(img, (8, 8))
    # 转换为灰度图
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
    # s为像素和初值为0,hash_str为hash值初值为''
    s = 0
    hash_str = ''
    # 遍历累加求像素和
    for i in range(8):
        for j in range(8):
            s = s+gray[i, j]
    # 求平均灰度
    avg = s/64
    # 灰度大于平均值为1相反为0生成图片的hash值
    for i in range(8):
        for j in range(8):
            if gray[i, j] > avg:
                hash_str = hash_str+'1'
            else:
                hash_str = hash_str+'0'
    return hash_str

# 差值哈希算法
def dHash(img):
    # 缩放8*8
    # img = cv2.imread(img) #此种方法只能用于本地图片读取
    img = np.asanyarray(img)
    img = cv2.resize(img, (9, 8))
    # 转换灰度图
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
    hash_str = ''
    # 每行前一个像素大于后一个像素为1,相反为0,生成哈希
    for i in range(8):
        for j in range(8):
            if gray[i, j] > gray[i, j+1]:
                hash_str = hash_str+'1'
            else:
                hash_str = hash_str+'0'
    return hash_str

# 感知哈希算法
def pHash(img):
    # 缩放32*32
    # img = cv2.imread(img)
    img = np.asanyarray(img)
    img = cv2.resize(img, (32, 32))  # , interpolation=cv2.INTER_CUBIC
    # 转换为灰度图
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
    # 将灰度图转为浮点型,再进行dct变换
    dct = cv2.dct(np.float32(gray))
    # opencv实现的掩码操作
    dct_roi = dct[0:8, 0:8]

    hash = []
    avreage = np.mean(dct_roi)
    for i in range(dct_roi.shape[0]):
        for j in range(dct_roi.shape[1]):
            if dct_roi[i, j] > avreage:
                hash.append(1)
            else:
                hash.append(0)
    return hash

# 灰度直方图算法
def calculate(image1, image2):
    # 计算单通道的直方图的相似值
    # 均值、差值、感知哈希算法三种算法值越小,则越相似,相同图片值为0
    # 三直方图算法和单通道的直方图 0-1之间,值越大,越相似。 相同图片为1
    # image1 = cv2.imread(image1)
    # image2 = cv2.imread(image2)
    image1 = np.asanyarray(image1)
    image2 = np.asanyarray(image2)
    hist1 = cv2.calcHist([image1], [0], None, [256], [0.0, 255.0])
    hist2 = cv2.calcHist([image2], [0], None, [256], [0.0, 255.0])
    # 计算直方图的重合度
    degree = 0
    for i in range(len(hist1)):
        if hist1[i] != hist2[i]:
            degree = degree + \
                (1 - abs(hist1[i] - hist2[i]) / max(hist1[i], hist2[i]))
        else:
            degree = degree + 1
    degree = degree / len(hist1)
    return degree

# 三通道直方图算法
def classify_calculate(image1, image2):
    # 计算单通道的直方图的相似值
    # 均值、差值、感知哈希算法三种算法值越小,则越相似,相同图片值为0
    # 三直方图算法和单通道的直方图 0-1之间,值越大,越相似。 相同图片为1
    hist1 = cv2.calcHist([image1], [0], None, [256], [0.0, 255.0])
    hist2 = cv2.calcHist([image2], [0], None, [256], [0.0, 255.0])
    # 计算直方图的重合度
    degree = 0
    for i in range(len(hist1)):
        if hist1[i] != hist2[i]:
            degree = degree + \
                (1 - abs(hist1[i] - hist2[i]) / max(hist1[i], hist2[i]))
        else:
            degree = degree + 1
    degree = degree / len(hist1)
    return degree

# RGB每个通道的直方图相似度
def classify_hist_with_split(image1, image2, size=(256, 256)):
    # 将图像resize后,分离为RGB三个通道,再计算每个通道的相似值
    image1 = np.asanyarray(image1)
    image2 = np.asanyarray(image2)
    image1 = cv2.resize(image1, size)
    image2 = cv2.resize(image2, size)
    sub_image1 = cv2.split(image1)
    sub_image2 = cv2.split(image2)
    sub_data = 0
    for im1, im2 in zip(sub_image1, sub_image2):
        sub_data += classify_calculate(im1, im2)
    sub_data = sub_data / 3
    return sub_data

# Hash值对比
def cmpHash(hash1, hash2):
    # 均值、差值、感知哈希算法三种算法值越小,则越相似,相同图片值为0
    # 三直方图算法和单通道的直方图 0-1之间,值越大,越相似。 相同图片为1
    # 算法中1和0顺序组合起来的即是图片的指纹hash。顺序不固定,但是比较的时候必须是相同的顺序。
    # 对比两幅图的指纹,计算汉明距离,即两个64位的hash值有多少是不一样的,不同的位数越小,图片越相似
    # 汉明距离:一组二进制数据变成另一组数据所需要的步骤,可以衡量两图的差异,汉明距离越小,则相似度越高。汉明距离为0,即两张图片完全一样
    n = 0
    # hash长度不同则返回-1代表传参出错
    if len(hash1) != len(hash2):
        return -1
    # 遍历判断
    for i in range(len(hash1)):
        # 不相等则n计数+1,n最终为相似度
        if hash1[i] != hash2[i]:
            n = n + 1
    return n


三、基本功能模块——截图、图片转换

from PIL import ImageGrab
import numpy as np

#截图方法
#传入图片像素位置以及所需比例宽度高度 #引入自己写的模块
from LLKgame.similarFunctinos import cmpHash, pHash, classify_hist_with_split, calculate

#传入截图的左上右下坐标,宽的小图数量,长的小图数量,小图的高度和宽度
#如cutPicture(304, 271, 1226, 870,10,14,60.2,65.9)在坐标(304,271)(1226,870)处截取140个小图,小图的高是60.2,宽是65.9

def cutPicture(left,top,right,botton,pWidth,pHeight,height,width):  # 截图
	size = (left, top, right, botton)
	img = ImageGrab.grab(size)
	# img.size(200,100)
	print(img.size)
	print(img)
	# img.save("D://cut.jpg")
	# img.show()
	# 2、分切小图
	image_list = {}
	for x in range(pWidth):
		image_list[x] = {}
		for y in range(pHeight):
			# print("show",x, y)
			# exit()
			top1 = x * height
			left1 = y * width
			right1 = (y + 1) * width
			botton1 = (x + 1) * height
			# 用crop函数切割成小图标,参数为图标的左上角和右下角左边
			im = img.crop((left1, top1, right1, botton1))
			# im.show()
			# time.sleep(1)
			# 将切割好的图标存入对应的位置
			image_list[x][y] = im
	print(image_list)
	return image_list

#创造数字矩阵
#传入矩阵的宽高和初始矩阵
def makeArray(pWidth,pHeight,image_list):
	image_type_list = []
	arr = np.zeros((pWidth + 2,pHeight + 2), dtype=np.int32)  # 创建矩阵以数字代替图片
	for i in range(len(image_list)):
		for j in range(len(image_list[0])):
			im = image_list[i][j]
			# 验证当前图标是否已存入
			index = getIndex(10,0.65,im, image_type_list)
			# 不存在image_type_list
			if index < 0:
				image_type_list.append(im)
				arr[i + 1][j + 1] = len(image_type_list)
			else:
				arr[i + 1][j + 1] = index + 1
	print("图标数:", len(image_type_list))
	# self.im2num_arr = arr
	return arr

# 检查数组中是否有图标,如果有则返回索引下表
#传入标准相似度similar1 similar2 和需要验证图片
def getIndex(similar1,similar2,im, im_list):
	global flag
	for i in range(len(im_list)):
		flag = 0
		# if self.compare_image_with_hash(im, im_list[i],6):
		#     return i
		val1 = calculate(im,im_list[i])
		val2 = classify_hist_with_split(im,im_list[i])
		val3 = cmpHash(pHash(im), pHash(im_list[i]))
		if val2 >= similar2:
			flag = 1
		if val1 >= similar2:
			flag = flag + 1
		if val3 <= similar1:
			flag = flag + 1
		if flag >= 2:
			return i
	return -1

以上就是程序所有的代码,程序难点在于两个图片之间相似度识别的精确性,不过对于练练看这个游戏来说,这个程序所使用的计算算法是够用的。喜欢的朋友可以一键三连!在评论区进行评论!

相关推荐

大文件传不动?WinRAR/7-Zip 入门到高手,这 5 个技巧让你效率翻倍

“这200张照片怎么传给女儿?微信发不了,邮箱附件又超限……”62岁的张阿姨对着电脑犯愁时,儿子只用了3分钟就把照片压缩成一个文件,还教她:“以后用压缩软件,比打包行李还方便!”职场人更懂这...

电脑解压缩软件推荐——7-Zip:免费、高效、简洁的文件管理神器

在日常工作中,我们经常需要处理压缩文件。无论是下载软件包、接收文件,还是存储大量数据,压缩和解压缩文件都成为了我们日常操作的一部分。而说到压缩解压软件,7-Zip绝对是一个不可忽视的名字。今天,我就来...

设置了加密密码zip文件要如何打开?这几个方法可以试试~

Zip是一种常见的压缩格式文件,文件还可以设置密码保护。那设置了密码的Zip文件要如何打开呢?不清楚的小伙伴一起来看看吧。当我们知道密码想要打开带密码的Zip文件,我们需要用到适用于Zip格式的解压缩...

大文件想要传输成功,怎么把ZIP文件分卷压缩

不知道各位小伙伴有没有这样的烦恼,发送很大很大的压缩包会受到限制,为此,想要在压缩过程中将文件拆分为几个压缩包并且同时为所有压缩包设置加密应该如何设置?方法一:使用7-Zip免费且强大的文件管理工具7...

高效处理 RAR 分卷压缩包:合并解压操作全攻略

在文件传输和存储过程中,当遇到大文件时,我们常常会使用分卷压缩的方式将其拆分成多个较小的压缩包,方便存储和传输。RAR作为一种常见的压缩格式,分卷压缩包的使用频率也很高。但很多人在拿到RAR分卷...

2个方法教你如何删除ZIP压缩包密码

zip压缩包设置了加密密码,每次解压文件都需要输入密码才能够顺利解压出文件,当压缩包文件不再需要加密的时候,大家肯定想删除压缩包密码,或是忘记了压缩包密码,想要通过删除操作将压缩包密码删除,就能够顺利...

速转!漏洞预警丨压缩软件Winrar目录穿越漏洞

WinRAR是一款功能强大的压缩包管理器,它是档案工具RAR在Windows环境下的图形界面。该软件可用于备份数据,缩减电子邮件附件的大小,解压缩从Internet上下载的RAR、ZIP及其它类...

文件解压方法和工具分享_文件解压工具下载

压缩文件减少文件大小,降低文件失效的概率,总得来说好处很多。所以很多文件我们下载下来都是压缩软件,很多小伙伴不知道怎么解压,或者不知道什么工具更好,所以今天做了文件解压方法和工具的分享给大家。一、解压...

[python]《Python编程快速上手:让繁琐工作自动化》学习笔记3

1.组织文件笔记(第9章)(代码下载)1.1文件与文件路径通过importshutil调用shutil模块操作目录,shutil模块能够在Python程序中实现文件复制、移动、改名和删除;同时...

Python内置tarfile模块:读写 tar 归档文件详解

一、学习目标1.1学习目标掌握Python内置模块tarfile的核心功能,包括:理解tar归档文件的原理与常见压缩格式(gzip/bz2/lzma)掌握tar文件的读写操作(创建、解压、查看、过滤...

使用python展开tar包_python拓展

类Unix的系统,打包文件经常使用的就是tar包,结合zip工具,可以方便的打包并解压。在python的标准库里面有tarfile库,可以方便实现生成了展开tar包。使用这个库最大的好处,可能就在于不...

银狐钓鱼再升级:白文件脚本化实现GO语言后门持久驻留

近期,火绒威胁情报中心监测到一批相对更为活跃的“银狐”系列变种木马。火绒安全工程师第一时间获取样本并进行分析。分析发现,该样本通过阿里云存储桶下发恶意文件,采用AppDomainManager进行白利...

ZIP文件怎么打开?2个简单方法教你轻松搞定!

在日常工作和生活中,我们经常会遇到各种压缩文件,其中最常见的格式之一就是ZIP。ZIP文件通过压缩数据来减少文件大小,方便我们进行存储和传输。然而,对于初学者来说,如何打开ZIP文件可能会成为一个小小...

Ubuntu—解压多个zip压缩文件.zip .z01 .z02

方法将所有zip文件放在同一目录中:zip_file.z01,zip_file.z02,zip_file.z03,...,zip_file.zip。在Zip3.0版本及以上,使用下列命令:将所有zi...

如何使用7-Zip对文件进行加密压缩

7-Zip是一款开源的文件归档工具,支持多种压缩格式,并提供了对压缩文件进行加密的功能。使用7-Zip可以轻松创建和解压.7z、.zip等格式的压缩文件,并且可以通过设置密码来保护压缩包中的...

取消回复欢迎 发表评论: