一.本案例基于UDP的socket编程方法来制作五子棋程序,网络五子棋采用C/S架构,分为服务器端和客户端,游戏时服务端首先启动,当客户端启动连接后,服务器端可以走棋,轮到自己棋才可以在棋盘上落子,同时下方标签会显示对方走起信息,服务器端用户可以通过“退出游戏”按钮可以结束游戏;
1.数据通信协议
网络五子棋游戏设计的难点在于对方进行通信,这里使用面向非连接的Socket编程,Socket编程用于C/S开发,在这类应用中,客户端和服务器端通常需要先建立连接,然后发送和接收数据,交互完成后需要断开连接,本章采用基于UDP的Socket编程实现,这里虽然两台计算机不分主次,但涉及时候假设一台作为服务器端,等待其他方加入,其他想加入必须输入服务器端主机的IP;
下面展示一些 数据通信协议代码。
def receiveMessage(): global s while True: global addr data,addr=s.recvfrom(1024) data=data.decode('utf-8') a=data.split("|") if not data: print("client has exited!") break elif a[0]=='join': #连接服务器请求 print('client 连接服务器!') label1["text"]='client连接服务器成功,请你走棋!' elif a[0]=='exit': print('client 对方退出!') label1["text"]='client对方退出,游戏结束!' elif a[0]=='over': print('对方赢信息!') label1["text"]=data.split("|")[0] showinfo(title="提示",message=data.split("|")[1]) elif a[0]=='move': print('received:',data,'from',addr) p=a[1].split(",") x=int(p[0]) y=int(p[1]) print(p[0],p[1]) label1["text"]="客户端走的位置"+p[0]+p[1] drawOtherChess(x,y) s.close()
2.判断输赢的算法
本游戏关键技术就是判断输赢的算法,对于算法实现大致可以分为以下几个部分:
(1)判断X=Y轴上是否形成五子连珠;
(2)判断X=-Y轴上是否形成五子连珠;
(3)判断X轴上是否形成五子连珠;
(4)判断Y轴上是否形成五子连珠;
#输赢判断 def win_lose(): a=str(turn) print("a=",a) for i in range(0,11): for j in range(0,11): if map[i][j]==a and map[i+1][j+1]==a and map[i+2][j+2]==a and map[i+3][j+3]==a and map[i+4][j+4]==a: print("x=y轴上形成五子连珠") return True for i in range(4,15): for j in range(0,11): if map[i][j]==a and map[i-1][j+1]==a and map[i-2][j+2]==a and map[i-3][j+3]==a and map[i-4][j+4]==a: print("x=-y轴上形成五子连珠") return True for i in range(0,15): for j in range(4,15): if map[i][j]==a and map[i][j-1]==a and map[i][j-2]==a and map[i][j-2]==a and map[i][j-4]==a: print("Y轴上形成了五子连珠") return True for i in range(0,11): for j in range(0,15): if map[i][j]==a and map[i+1][j]==a and map[i+2][j]==a and map[i+3][j]==a and map[i+4][j]==a: print("X轴形成五子连珠") return True return False
二. 源代码:
1.客户端编程代码如下:
from tkinter import * from tkinter.messagebox import * import socket import threading import os #主程序 root=Tk() root.title("网络五子棋v2.0--UDP客户端") imgs=[PhotoImage(file='E:\\game\\BlackStone.gif'),PhotoImage(file='E:\\game\\WhiteStone.gif')] turn=0 Myturn=-1 #画对方棋子 def drawOtherChess(x,y): global turn img1=imgs[turn] cv.create_image((x*40+20,y*40+20),image=img1) cv.pack() map[x][y]=str(turn) #换下一方走棋 if turn==0: turn=1 else: turn=0 #发送消息 def sendMessage(pos): global s s.sendto(pos.encode(),(host,port)) #退出函数 def callexit(event): pos="exit|" sendMessage(pos) os._exit(0) #走棋函数 def callback(event): global turn global Myturn if Myturn==-1: Myturn=turn else: if(Myturn!=turn): showinfo(title="提示",message="还没轮到自己走棋") return #print("clicked at",event.x,event.y) x=(event.x)//40 y=(event.y)//40 print("clicked at",x,y,turn) if map[x][y]!=" ": showinfo(title="提示",message="已有棋子") else: img1=imgs[turn] cv.create_image((x*40+20,y*40+20),image=img1) cv.pack() map[x][y]=str(turn) pos=str(x)+','+str(y) sendMessage("move|"+pos) print("客户端走的位置",pos) label1["text"]="客户端走的位置"+pos #输出输赢信息 if win_lose( )==True: if turn==0: showinfo(title="提示",message="黑方你赢了") sendMessage("over|黑方你赢了!") else: showinfo(title="提示",message="白方你赢了!") sendMessage("over|白方你赢了!") #换下一方走棋: if turn==0: turn=1 else: turn=0 #画棋盘 def drawQiPan( ): #画棋盘 for i in range(0,15): cv.create_line(20,20+40*i,580,20+40*i,width=2) for i in range(0,15): cv.create_line(20+40*i,20,20+40*i,580,width=2) cv.pack() #输赢判断 def win_lose(): a=str(turn) print("a=",a) for i in range(0,11): for j in range(0,11): if map[i][j]==a and map[i+1][j+1]==a and map[i+2][j+2]==a and map[i+3][j+3]==a and map[i+4][j+4]==a: print("x=y轴上形成五子连珠") return True for i in range(4,15): for j in range(0,11): if map[i][j]==a and map[i-1][j+1]==a and map[i-2][j+2]==a and map[i-3][j+3]==a and map[i-4][j+4]==a: print("x=-y轴上形成五子连珠") return True for i in range(0,15): for j in range(4,15): if map[i][j]==a and map[i][j-1]==a and map[i][j-2]==a and map[i][j-2]==a and map[i][j-4]==a: print("Y轴上形成了五子连珠") return True for i in range(0,11): for j in range(0,15): if map[i][j]==a and map[i+1][j]==a and map[i+2][j]==a and map[i+3][j]==a and map[i+4][j]==a: print("X轴形成五子连珠") return True return False #接受消息 def receiveMessage(): #接受消息 global s while True: data = s.recv(1024).decode('utf-8') a = data.split("|") if not data: print('server has exited!') break elif a[0] == 'exit': print('对方退出!') lanel1["text"] = '对方退出!游戏结束!' elif a[0] == 'over': print('对方赢信息!') label1["text"] = data.split("|")[0] showinfo(title="提示", message=data.split("|")[1]) elif a[0] == 'move': print('received:', data) p = a[1].split(",") x = int(p[0]) y = int(p[1]) print(p[0], p[1]) label1["text"] = "服务器走的位置" + p[0] + p[1] drawOtherChess(x,y) s.close() #启动线程接受客户端消息 def startNewThread(): thread=threading.Thread(target=receiveMessage,args=()) thread.setDaemon(True) thread.start() #主程序 map=[[" "," "," "," "," "," "," "," "," "," "," "," "," "," "," "] for y in range(15)] cv=Canvas(root,bg='green',width=610,height=610) drawQiPan() cv.bind("<Button-1>",callback) cv.pack() label1=Label(root,text="客户端...") label1.pack() button1=Button(root,text="退出游戏") button1.bind("<Button-1>",callexit) button1.pack() #创建UDP s=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) port=8000 host='localhost' pos='join|' sendMessage(pos) startNewThread() root.mainloop()
2.数据通信协议源代码:
下面展示一些 数据通信协议源代码。
def receiveMessage(): global s while True: global addr data,addr=s.recvfrom(1024) data=data.decode('utf-8') a=data.split("|") if not data: print("client has exited!") break elif a[0]=='join': #连接服务器请求 print('client 连接服务器!') label1["text"]='client连接服务器成功,请你走棋!' elif a[0]=='exit': print('client 对方退出!') label1["text"]='client对方退出,游戏结束!' elif a[0]=='over': print('对方赢信息!') label1["text"]=data.split("|")[0] showinfo(title="提示",message=data.split("|")[1]) elif a[0]=='move': print('received:',data,'from',addr) p=a[1].split(",") x=int(p[0]) y=int(p[1]) print(p[0],p[1]) label1["text"]="客户端走的位置"+p[0]+p[1] drawOtherChess(x,y) s.close() #输赢判断 def win_lose(): a=str(turn) print("a=",a) for i in range(0,11): for j in range(0,11): if map[i][j]==a and map[i+1][j+1]==a and map[i+2][j+2]==a and map[i+3][j+3]==a and map[i+4][j+4]==a: print("x=y轴上形成五子连珠") return True for i in range(4,15): for j in range(0,11): if map[i][j]==a and map[i-1][j+1]==a and map[i-2][j+2]==a and map[i-3][j+3]==a and map[i-4][j+4]==a: print("x=-y轴上形成五子连珠") return True for i in range(0,15): for j in range(4,15): if map[i][j]==a and map[i][j-1]==a and map[i][j-2]==a and map[i][j-2]==a and map[i][j-4]==a: print("Y轴上形成了五子连珠") return True for i in range(0,11): for j in range(0,15): if map[i][j]==a and map[i+1][j]==a and map[i+2][j]==a and map[i+3][j]==a and map[i+4][j]==a: print("X轴形成五子连珠") return True return False def checkwin(x,y): flag=False count=1 color=map[x][y] i=1 #横向判断 while color==map[x+i][y]: count=count+1 i=i+1 i=1 while color==map[x-i][y]: count=count+1 i=i+1 if count>=5: flag=True #竖向判断 i=1 while color==map[x][y+i]: count=count+1 i=i+1 i=1 while color==map[x][y-i]: count=count+1 i=i+1 if count>=5: flag=True #x=y判断 i=1 j=1 while color==map[x+i][y+i]: count=count+1 i=i+1 j=j+1 if count>=5: flag=True j=1 i=1 while color==map[x-i][y-i]: count=count+1 i=i+1 j=j+1 if count>=5: flag=True
3.服务器端源代码
客户端源代码如下:
from tkinter import * from tkinter.messagebox import * import socket,threading,os def drawQiPan(): for i in range(0,15): cv.create_line(20,20+40*i,580,20+40*i,width=2) for i in range(0,15): cv.create_line(20+40*i,20,20+40*i,580,width=2) cv.pack() #走棋函数 def callpos(event): global turn global Myturn if Myturn==-1: #第一次确认自己的角色 Myturn=turn else: if(Myturn!=turn): showinfo(title="提示",message="还没轮到自己下棋") return #print("clicked at",event.x,event.y,true) x=(event.x)//40 y=(event.y)//40 print("clicked at",x,y,turn) if map[x][y]!=" ": showinfo(title="提示",message="已有棋子") else: img1=imgs[turn] cv.create_image((x*40+20,y*40+20),image=img1) cv.pack() map[x][y]=str(turn) pos=str(x)+","+str(y) sendMessage("move|"+pos) print("服务器走的位置",pos) label1["text"]="服务器走的位置"+pos #输出输赢信息 if win_lose( )==True: if turn==0: showinfo(title="提示",message="黑方你赢了") sendMessage("over|黑方你赢了") else: showinfo(title="提示", message="白方你赢了") sendMessage("over|白方你赢了") #换下一方走棋 if turn==0: turn=1 else: turn=0 #发送消息 def sendMessage(pos): global s global addr s.sendto(pos.encode(),addr) #退出函数 def callexit(event): pos="exit|" sendMessage(pos) os._exit(0) #画对方棋子 def drawOtherChess(x,y): global turn img1=imgs[turn] cv.create_image((x*40+20,y*40+20),image=img1) cv.pack() map[x][y]=str(turn) #换下一方走棋 if turn==0: turn=1 else: turn=0 #判断整个棋盘的输赢 def win_lose(): a=str(turn) print("a=",a) for i in range(0,11): for j in range(0,11): if map[i][j]==a and map[i+1][j+1]==a and map[i+2][j+2]==a and map[i+3][j+3]==a and map[i+4][j+4]==a: print("x=y轴上形成五子连珠") return True for i in range(4,15): for j in range(0,11): if map[i][j]==a and map[i-1][j+1]==a and map[i-2][j+2]==a and map[i-3][j+3]==a and map[i-4][j+4]==a: print("x=-y轴上形成五子连珠") return True for i in range(0,15): for j in range(4,15): if map[i][j]==a and map[i][j-1]==a and map[i][j-2]==a and map[i][j-2]==a and map[i][j-4]==a: print("Y轴上形成了五子连珠") return True for i in range(0,11): for j in range(0,15): if map[i][j]==a and map[i+1][j]==a and map[i+2][j]==a and map[i+3][j]==a and map[i+4][j]==a: print("X轴形成五子连珠") return True return False #输出map地图 def print_map(): for j in range(0,15): for i in range(0,15): print(map[i][j],end=' ') print('w') #接受消息 def receiveMessage(): global s while True:#接受客户端发送的消息 global addr data,addr=s.recvfrom(1024) data=data.decode('utf-8') a=data.split("|") if not data: print('client has exited!') break elif a[0]=='join':#连接服务器的请求 print('client 连接服务器!') label1["text"]='client连接服务器成功,请你走棋!' elif a[0]=='exit': print('client对方退出!') label1["text"]='client对方退出,游戏结束!' elif a[0]=='over': print('对方赢信息!') labl1["text"]==data.split("|")[0] showinfo(title="提示",message=data.split("1")[1]) elif a[0]=='move': print('received:',data,'from',addr) p=a[1].split(",") x=int(p[0]) y=int(p[1]) print(p[0],p[1]) label1["text"]="客户端走的位置"+p[0]+p[1] drawOtherChess(x,y) s.close() def startNewThread( ):#启动新线程来接受客户端消息 thread=threading.Thread(target=receiveMessage,args=()) thread.setDaemon(True) thread.start() root=Tk() root.title("网络五子棋v2.0-服务器端") imgs=[PhotoImage(file='E:\\game\\BlackStone.gif'),PhotoImage(file='E:\\game\\WhiteStone.gif')] turn=0 Myturn=-1 map=[[" "," "," "," "," "," "," "," "," "," "," "," "," "," "," "] for y in range(15)] cv=Canvas(root,bg='green',width=610,height=610) drawQiPan() cv.bind("<Button-1>",callpos) cv.pack() label1=Label(root,text="服务器端...") label1.pack() button1=Button(root,text="退出游戏") button1.bind("<Button-1>",callexit) button1.pack() #创建UDP SOCKET s=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) s.bind(('localhost',8000)) addr=('localhost',8000) startNewThread() root.mainloop()
运行结果展示:
所有的都在这里了。
更多有趣的经典小游戏实现专题,也分享给大家:
C++经典小游戏汇总
python经典小游戏汇总
python俄罗斯方块游戏集合
JavaScript经典游戏 玩不停
java经典小游戏汇总
javascript经典小游戏汇总
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 凤飞飞《我们的主题曲》飞跃制作[正版原抓WAV+CUE]
- 刘嘉亮《亮情歌2》[WAV+CUE][1G]
- 红馆40·谭咏麟《歌者恋歌浓情30年演唱会》3CD[低速原抓WAV+CUE][1.8G]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[320K/MP3][193.25MB]
- 【轻音乐】曼托凡尼乐团《精选辑》2CD.1998[FLAC+CUE整轨]
- 邝美云《心中有爱》1989年香港DMIJP版1MTO东芝首版[WAV+CUE]
- 群星《情叹-发烧女声DSD》天籁女声发烧碟[WAV+CUE]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[FLAC/分轨][748.03MB]
- 理想混蛋《Origin Sessions》[320K/MP3][37.47MB]
- 公馆青少年《我其实一点都不酷》[320K/MP3][78.78MB]
- 群星《情叹-发烧男声DSD》最值得珍藏的完美男声[WAV+CUE]
- 群星《国韵飘香·贵妃醉酒HQCD黑胶王》2CD[WAV]
- 卫兰《DAUGHTER》【低速原抓WAV+CUE】
- 公馆青少年《我其实一点都不酷》[FLAC/分轨][398.22MB]
- ZWEI《迟暮的花 (Explicit)》[320K/MP3][57.16MB]