一.目的

  以实现小项目的方式,来巩固之前学过的Python基本语法以及相关的知识。

二.相关技术:

1.wxpython GUI编程

2.网络编程

3.多线程编程

4.数据库编程

5.简单的将数据导出到Excel表

三.存在的漏洞以及不足

  1.由于数据库编码的问题,无法使用中文。

  2.在客户端关闭后,其相关的线程仍然存在于服务器的用户线程队列中,所以服务器会错误地往已关闭的客户端传送信息。

  3.客户端初始登录并加载历史记录时,会出现每条历史消息后面的回车键丢失的现象,解决的方法是:在加载相邻两条消息之间加个时间间隔,但效果不佳。

四.源码

服务器Server:

  1 # -*- coding: UTF-8 -*-
  2
  3 from socket import *
  4 import time
  5 import threading
  6 import wx
  7 import MySQLdb
  8 import xlwt
  9 from clientthread import ClientThread
 10
 11 class Server(wx.Frame):
 12     def __init__(self,parent=None,id=-1,title='服务器',pos=wx.DefaultPosition,size=(500,300)):
 13
 14         '''窗口'''
 15         wx.Frame.__init__(self,parent,id,title,pos,size=(400,470))
 16         pl = wx.Panel(self)
 17         con = wx.BoxSizer(wx.VERTICAL)
 18         subcon = wx.FlexGridSizer(wx.HORIZONTAL)
 19         sta = wx.Button(pl , size=(133, 40),label='启动服务器')
 20         end = wx.Button(pl, size=(133, 40), label='关闭服务器')
 21         hist = wx.Button(pl,size=(133,40),label='导出聊天记录')
 22         subcon.Add(sta, 1, wx.BOTTOM)
 23         subcon.Add(hist, 1, wx.BOTTOM)
 24         subcon.Add(end, 1, wx.BOTTOM)
 25         con.Add(subcon,1,wx.ALIGN_CENTRE|wx.BOTTOM)
 26         self.Text = wx.TextCtrl(pl, size=(400,250),style = wx.TE_MULTILINE|wx.TE_READONLY)
 27         con.Add(self.Text, 1, wx.ALIGN_CENTRE)
 28         self.ttex = wx.TextCtrl(pl, size=(400,100),style=wx.TE_MULTILINE)
 29         con.Add(self.ttex, 1, wx.ALIGN_CENTRE)
 30         sub2 = wx.FlexGridSizer(wx.HORIZONTAL)
 31         clear = wx.Button(pl, size=(200, 40), label='清空')
 32         send = wx.Button(pl, size=(200, 40), label='发送')
 33         sub2.Add(clear, 1, wx.TOP | wx.LEFT)
 34         sub2.Add(send, 1, wx.TOP | wx.RIGHT)
 35         con.Add(sub2, 1, wx.ALIGN_CENTRE)
 36         pl.SetSizer(con)
 37         '''窗口'''
 38
 39         '''绑定'''
 40         self.Bind(wx.EVT_BUTTON, self.EditClear, clear)
 41         self.Bind(wx.EVT_BUTTON, self.SendMessage, send)
 42         self.Bind(wx.EVT_BUTTON, self.Start, sta)
 43         self.Bind(wx.EVT_BUTTON, self.Break, end)
 44         self.Bind(wx.EVT_BUTTON, self.WriteToExcel, hist)
 45         '''绑定'''
 46
 47         '''服务器准备工作'''
 48         self.UserThreadList = []
 49         self.onServe = False
 50         addr = ('', 21567)
 51         self.ServeSock = socket(AF_INET, SOCK_STREAM)
 52         self.ServeSock.bind(addr)
 53         self.ServeSock.listen(10)
 54         '''服务器准备工作'''
 55
 56         '''数据库准备工作,用于存储聊天记录'''
 57         self.db = MySQLdb.connect('localhost', 'root', '123456', 'user_info')
 58         self.cursor = self.db.cursor()
 59         self.cursor.execute("select * from history order by time")
 60         self.Text.SetValue('')
 61         for data in self.cursor.fetchall():   #加载历史聊天记录
 62             self.Text.AppendText('%s said:\n%s\nwhen %s\n\n' % (data[0], data[2], data[1]))
 63         '''数据库准备工作,用于存储聊天记录'''
 64
 65
 66     #将聊天记录导出到EXCEl表中
 67     def WriteToExcel(self,event):
 68         wbk = xlwt.Workbook()
 69         sheet = wbk.add_sheet('sheet 1')
 70         self.cursor.execute("select * from history order by time")
 71         sheet.write(0, 0, "User")
 72         sheet.write(0, 1, "Datetime")
 73         sheet.write(0, 5, "Message")
 74         index = 0
 75         for data in self.cursor.fetchall():
 76             index = index + 1
 77             Time = '%s'%data[1] #将datetime转成字符形式,否则直接写入Excel会变成时间戳
 78             sheet.write(index,0,data[0])
 79             sheet.write(index,1,Time)    #写进EXCEL会变成时间戳
 80             sheet.write(index,5,data[2])
 81         wbk.save(r'D:\History_Dialog.xls')
 82
 83
 84     #启动服务器的服务线程
 85     def Start(self,event):
 86         if not self.onServe:
 87             '''启动服务线程'''
 88             self.onServe = True
 89             mainThread = threading.Thread(target=self.on_serving, args=())
 90             mainThread.setDaemon(True)  # 解决父线程结束,子线程还继续运行的问题
 91             mainThread.start()
 92             '''启动服务线程'''
 93
 94     #关闭服务器
 95     def Break(self,event):
 96         self.onServe = False
 97
 98     #服务器主循环
 99     def on_serving(self):
100         print '...On serving...'
101         while self.onServe:
102             UserSocket, UserAddr = self.ServeSock.accept()
103             username = UserSocket.recv(1024).decode(encoding='utf-8')   #接收用户名
104             userthread = ClientThread(UserSocket, username,self)
105             self.UserThreadList.append(userthread)  #将用户线程加到队列中
106             userthread.start()
107         self.ServeSock.close()
108
109     #绑定发送按钮
110     def SendMessage(self,event):
111         if self.onServe and cmp(self.ttex.GetValue(),''):
112             data = self.ttex.GetValue()
113             self.AddText('Server',data,time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()))
114             self.ttex.SetValue('')
115
116
117     # 向所有客户端(包括自己)发送信息,同时更新到数据库
118     def AddText(self, source, data,Time):
119         self.cursor.execute("insert into history values(\"%s\",\"%s\",\"%s\")" % (source,Time,data))    #双引号里面有双引号,bug:句子不能有双引号、以及中文
120         self.db.commit()
121         sendData = '%s said:\n%s\nwhen %s\n' % (source,data,Time)
122         self.Text.AppendText('%s\n'%sendData)
123         for user in self.UserThreadList:        #bug:客户端关闭了仍然在队列中。如果客户端关闭了,那怎么在服务器判断是否已经关闭了?客户端在关闭之前发一条信息给服务器?
124             user.UserSocket.send(sendData.encode(encoding='utf-8'))
125
126     #绑定清空按钮
127     def EditClear(self,event):
128         self.ttex.Clear()
129
130
131 def main():
132     app = wx.App(False)
133     Server().Show()
134     app.MainLoop()
135
136 if __name__ == '__main__':
137     main()

View Code

服务器的客户线程Clientthread:

 1 # -*- coding: UTF-8 -*-
 2
 3 import threading
 4 import time
 5
 6 class ClientThread(threading.Thread):
 7
 8     def __init__(self,UserSocket, Username,server):
 9         threading.Thread.__init__(self)
10         self.UserSocket = UserSocket
11         self.Username = Username
12         self.server = server
13         self.Loadhist()
14
15     # 加载历史聊天记录
16     def Loadhist(self):
17         self.server.cursor.execute("select * from history order by time")
18         for data in self.server.cursor.fetchall():
19             time.sleep(0.6)                 #几条信息同时发,会造成末尾回车键的丢失,所以要有时间间隔
20             sendData = '%s said:\n%s\nwhen %s\n'%(data[0], data[2], data[1])
21             self.UserSocket.send(sendData.encode(encoding='utf-8'))
22
23
24     #方法重写,线程的入口
25     def run(self):
26         size = 1024
27         while True:
28             data = self.UserSocket.recv(size)   #未解决:客户端断开连接后这里会报错
29             self.server.AddText(self.Username,data.decode(encoding='utf-8'),time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()))
30         self.UserSocket.close() #这里都执行不到

View Code

客户登录界面Logframe:

  1 # -*- coding: UTF-8 -*-
  2
  3 from socket import *
  4 import wx
  5 import MySQLdb
  6 from client import Client
  7
  8 class LogFrame(wx.Frame):
  9     def __init__(self,parent=None,id=-1,title='登录窗口',pos=wx.DefaultPosition,size=(500,300)):
 10
 11         '''窗口'''
 12         wx.Frame.__init__(self,parent,id,title,pos,size=(400,280))
 13         self.pl = wx.Panel(self)
 14         con = wx.BoxSizer(wx.VERTICAL)
 15         subcon = wx.FlexGridSizer(2,2,10,10)
 16         username =  wx.StaticText(self.pl, label="Username:",style=wx.ALIGN_LEFT)
 17         password = wx.StaticText(self.pl, label="Password:",style=wx.ALIGN_LEFT)
 18         self.tc1 = wx.TextCtrl(self.pl,size=(180,20))
 19         self.tc2 = wx.TextCtrl(self.pl,size=(180,20),style=wx.TE_PASSWORD)
 20         subcon.Add(username,wx.TE_LEFT)
 21         subcon.Add(self.tc1,1,wx.EXPAND)
 22         subcon.Add(password)
 23         subcon.Add(self.tc2,1,wx.EXPAND)
 24         con.Add(subcon,1,wx.ALIGN_CENTER)
 25         subcon2 = wx.FlexGridSizer(1,2,10,10)
 26         register = wx.Button(self.pl,label='Register')
 27         login = wx.Button(self.pl,label='Login')
 28         subcon2.Add(register,1, wx.TOP)
 29         subcon2.Add(login,1, wx.TOP)
 30         con.Add(subcon2,1,wx.ALIGN_CENTRE)
 31         self.pl.SetSizer(con)
 32         self.Bind(wx.EVT_BUTTON,self.Register,register)
 33         self.Bind(wx.EVT_BUTTON,self.Login,login)
 34         '''窗口'''
 35         self.isConnected = False
 36         self.userSocket = None
 37
 38     #连接到服务器
 39     def ConnectToServer(self):
 40         if not self.isConnected:
 41             ADDR = ('localhost', 21567)
 42             self.userSocket = socket(AF_INET, SOCK_STREAM)
 43             try:
 44                 self.userSocket.connect(ADDR)
 45                 self.userSocket.send(self.tc1.GetValue().encode(encoding='utf-8'))
 46                 self.isConnected = True
 47                 return True
 48             except Exception:
 49                 return False
 50         else:
 51             return True
 52
 53     #登录
 54     def Login(self,event):
 55         if not self.ConnectToServer():
 56             err = wx.MessageDialog(None, '服务器未启动', 'ERROR!', wx.OK)
 57             err.ShowModal()
 58             err.Destroy()
 59         else:
 60             username = self.tc1.GetValue()
 61             password = self.tc2.GetValue()
 62             db = MySQLdb.connect('localhost', 'root', '123456', 'user_info')
 63             cursor = db.cursor()
 64             cursor.execute("select * from user_list where username='%s' and password='%s'"%(username,password))
 65             if not cursor.fetchone():
 66                 err = wx.MessageDialog(None,'用户不存在或密码错误','ERROR!',wx.OK)
 67                 err.ShowModal()
 68             else:
 69                 self.Close()
 70                 Client(opSock=self.userSocket, username=username).Show()
 71             db.commit()
 72             db.close()
 73
 74     #注册
 75     def Register(self,event):
 76         if not self.ConnectToServer():
 77             err = wx.MessageDialog(None, '服务器未启动', 'ERROR!', wx.OK)
 78             err.ShowModal()
 79             err.Destroy()
 80         else:
 81             username = self.tc1.GetValue()
 82             password = self.tc2.GetValue()
 83             db = MySQLdb.connect('localhost', 'root', '123456', 'user_info')
 84             cursor = db.cursor()
 85             cursor.execute("select * from user_list where username='%s'"%username)
 86             if not cursor.fetchone():
 87                 cursor.execute("insert into user_list(username,password) values('%s','%s')"%(username,password))
 88             else:
 89                 err = wx.MessageDialog(None, '用户已存在', 'ERROR!', wx.OK)
 90                 err.ShowModal()
 91             db.commit()
 92             db.close()
 93
 94
 95 def main():
 96     app = wx.App(False)
 97     LogFrame().Show()
 98     app.MainLoop()
 99
100 if __name__ == '__main__':
101     main()

View Code

客户端Client:

 1 #/usr/bin/env python
 2 # -*- coding: UTF-8 -*-
 3
 4 import wx
 5 import threading
 6 from time import ctime
 7
 8 class Client(wx.Frame):
 9     def __init__(self,opSock,username,parent=None,id=-1,title='客户端',pos=wx.DefaultPosition,size=(500,300)):
10
11         '''窗口'''
12         wx.Frame.__init__(self,parent,id,title,pos,size=(400,470))
13         self.opSock = opSock
14         self.username = username
15         pl = wx.Panel(self)
16         con = wx.BoxSizer(wx.VERTICAL)
17         subcon = wx.FlexGridSizer(wx.HORIZONTAL)
18         sta = wx.Button(pl, size=(200, 40),label='连接')
19         end = wx.Button(pl, size=(200, 40),label='断开')
20         subcon.Add(sta, 1, wx.TOP|wx.LEFT)
21         subcon.Add(end, 1, wx.TOP|wx.RIGHT)
22         con.Add(subcon,1,wx.ALIGN_CENTRE)
23         self.Text = wx.TextCtrl(pl, size=(400,250),style = wx.TE_MULTILINE|wx.TE_READONLY)
24         con.Add(self.Text, 1, wx.ALIGN_CENTRE)
25         self.ttex = wx.TextCtrl(pl, size=(400,100),style=wx.TE_MULTILINE)
26         con.Add(self.ttex, 1, wx.ALIGN_CENTRE)
27         sub2 = wx.FlexGridSizer(wx.HORIZONTAL)
28         clear = wx.Button(pl, size=(200, 40), label='清空')
29         send = wx.Button(pl, size=(200, 40), label='发送')
30         sub2.Add(clear, 1, wx.TOP | wx.LEFT)
31         sub2.Add(send, 1, wx.TOP | wx.RIGHT)
32         con.Add(sub2, 1, wx.ALIGN_CENTRE)
33         pl.SetSizer(con)
34         '''窗口'''
35
36         '''绑定'''
37         self.Bind(wx.EVT_BUTTON, self.EditClear, clear)
38         self.Bind(wx.EVT_BUTTON, self.Send, send)
39         self.Bind(wx.EVT_BUTTON, self.Login, sta)
40         self.Bind(wx.EVT_BUTTON, self.Logout, end)
41         '''绑定'''
42         self.isConnected = False
43
44     #登录
45     def Login(self,event):
46         '''客户端准备工作'''
47         self.isConnected = True
48         t = threading.Thread(target=self.Receive, args=())
49         t.setDaemon(True)
50         t.start()
51         '''客户端准备工作'''
52
53     #退出
54     def Logout(self,event):
55         self.isConnected = False
56
57     #绑定发送按钮
58     def Send(self,event):
59         if self.isConnected and cmp(self.ttex.GetValue(),''):
60             self.opSock.send(self.ttex.GetValue().encode(encoding='utf-8'))
61             self.ttex.SetValue('')
62
63     #绑定清空按钮
64     def EditClear(self,event):
65         self.ttex.Clear()
66
67     #接收客户端的信息(独立一个线程)
68     def Receive(self):
69         while self.isConnected:
70             data = self.opSock.recv(1024).decode(encoding='utf-8')
71             self.Text.AppendText('%s\n'%data)

View Code

转载于:https://www.cnblogs.com/DOLFAMINGO/p/9097393.html

Python多人聊天室相关推荐

  1. python多人聊天室_python实现简单多人聊天室

    本文实例为大家分享了python实现多人聊天室的具体代码,供大家参考,具体内容如下 刚开始学习python,写了一个聊天室练练手. Server.py import socket,select,thr ...

  2. python多人聊天室 跨主机_python+tcp实现多人聊天室

    tcp介绍 引用百度百科的介绍 传输控制协议(TCP,Transmission Control Protocol)是一种面向连接的.可靠的.基于字节流的传输层通信协议,由IETF的RFC 793 [1 ...

  3. python多人聊天室_Python基于Socket实现简易多人聊天室

    前言 套接字(Sockets)是双向通信信道的端点. 套接字可以在一个进程内,在同一机器上的进程之间,或者在不同主机的进程之间进行通信,主机可以是任何一台有连接互联网的机器. 套接字可以通过多种不同的 ...

  4. Python多人聊天室-基于socket UDP协议

    简介 使用Python编写的基于socket UDP通信的多功能即时聊天室,包含Tkinter编写的图形化聊天界面,功能包括有账号注册和登录,登录成功后可以查看在线用户,并和聊天室内的其他在线用户聊天 ...

  5. Python编写多线程多人聊天室系统(Socket编程、tkinter组件使用)

    是Python作业的说,也是自己编写了好长时间,有几个地方实现的方法非常的傻,请见谅QwQ 代码包含详细注释. 与此同时如果想让这个聊天室也能被全国各地的同学使用的话也可以将这个程序部署到云服务器上! ...

  6. python实现简易聊天需要登录博客园zip下载_Python基于Socket实现简易多人聊天室的示例代码...

    前言 套接字(Sockets)是双向通信信道的端点. 套接字可以在一个进程内,在同一机器上的进程之间,或者在不同主机的进程之间进行通信,主机可以是任何一台有连接互联网的机器. 套接字可以通过多种不同的 ...

  7. Python实现网络多人聊天室

    网络多人聊天室 相关连接:Python实现网络图形化界面多人聊天室 文件结构: chatroom ├── client.py  # 客户端代码 ├── language.py  # 语言文件 ├── ...

  8. 基于python面向对象多人聊天室

    基于python面向对象多人聊天室 1.项目环境 项目名称:多人聊天室 项目模式:C/S 开发环境:win10+python3.8+pycharm 所需知识:python GUI编程,多线程编程,网络 ...

  9. 基于Python Tkiner、thread与socket实现的简单多人聊天室,在Python中创建TCP服务器与客户端进行通信

    基于Python Tkiner.thread与socket实现的简单多人聊天室,在Python中创建TCP服务器与客户端进行通信 完整代码下载地址:基于Python Tkiner.thread与soc ...

  10. Python+Socket实现多人聊天室,功能:好友聊天、群聊、图片、表情、文件等

    一.项目简介 本项目主要基于python实现的多人聊天室,主要的功能如下: 登录注册 添加好友 与好友进行私聊 创建群聊 邀请/申请加入群聊 聊天发送图片 聊天发送表情 聊天发送文件 聊天记录保存在本 ...

最新文章

  1. proget Android代码混淆
  2. 框架退出的方法_AOP日志框架实现
  3. alpha冲刺day12
  4. 隐式马可夫模型(hidden markov model,HMM)
  5. python读取rar文件_在 python 中,如何读取由 7z 压缩的文本文件_python_酷徒编程知识库...
  6. tomcat jsp导入java_[导入]Tomcat JSP Web 开发中的乱码问题小姐
  7. 移动端html搜索怎么写,移动端实现搜索功能
  8. 使用MATLAB绘制周期信号的,周期信号频域分析及MATLAB实现.ppt
  9. 鸿蒙硬件HI3861-INA226-电压测量(外挂方案)
  10. 关于各种地图(百度、高德等等)的坐标类型以及相互转换
  11. Docker离线安装教程(Centos7) 转帖
  12. pthread_key_t和pthread_key_create()详解
  13. 几种降维思想方法总结
  14. spark根据key输出到多个目录
  15. a:link,a:visited,a:hover,a:active
  16. python深度学习——案例讲解
  17. windows做软件界面
  18. 挑战杯创业计划书_基于云计算的下料优化软件_1. 执行总结
  19. 在RStudio里面部署Python
  20. 有限公司清算组成员怎样构成

热门文章

  1. 解决git克隆项目出现fatal无法访问‘https://github.com/xxx/xxx‘Unsupported proxy syntax in‘127.0.0.1:8118‘
  2. python中文相似度_python比较两个文本的相似性
  3. mac IDEA java 如何进入到方法跳转 快捷键
  4. linux cpu 个数、核心数、线程数
  5. 资产负债表 公式总结
  6. 安装 EoLinker_4.0 开源版
  7. JDBC06 其他操作及批处理Batch
  8. JDBC02 加载JDBC驱动 建立连接
  9. guava 工具类及代码
  10. Servlet chapter 2