Python多人聊天室
一.目的
以实现小项目的方式,来巩固之前学过的Python基本语法以及相关的知识。
二.相关技术:
1.wxpython GUI编程
2.网络编程
3.多线程编程
4.数据库编程
5.简单的将数据导出到Excel表
三.存在的漏洞以及不足
1.由于数据库编码的问题,无法使用中文。
2.在客户端关闭后,其相关的线程仍然存在于服务器的用户线程队列中,所以服务器会错误地往已关闭的客户端传送信息。
3.客户端初始登录并加载历史记录时,会出现每条历史消息后面的回车键丢失的现象,解决的方法是:在加载相邻两条消息之间加个时间间隔,但效果不佳。
四.源码
服务器Server:
# -*- coding: UTF-8 -*- from socket import *
import time
import threading
import wx
import MySQLdb
import xlwt
from clientthread import ClientThread class Server(wx.Frame):
def __init__(self,parent=None,id=-1,title='服务器',pos=wx.DefaultPosition,size=(500,300)): '''窗口'''
wx.Frame.__init__(self,parent,id,title,pos,size=(400,470))
pl = wx.Panel(self)
con = wx.BoxSizer(wx.VERTICAL)
subcon = wx.FlexGridSizer(wx.HORIZONTAL)
sta = wx.Button(pl , size=(133, 40),label='启动服务器')
end = wx.Button(pl, size=(133, 40), label='关闭服务器')
hist = wx.Button(pl,size=(133,40),label='导出聊天记录')
subcon.Add(sta, 1, wx.BOTTOM)
subcon.Add(hist, 1, wx.BOTTOM)
subcon.Add(end, 1, wx.BOTTOM)
con.Add(subcon,1,wx.ALIGN_CENTRE|wx.BOTTOM)
self.Text = wx.TextCtrl(pl, size=(400,250),style = wx.TE_MULTILINE|wx.TE_READONLY)
con.Add(self.Text, 1, wx.ALIGN_CENTRE)
self.ttex = wx.TextCtrl(pl, size=(400,100),style=wx.TE_MULTILINE)
con.Add(self.ttex, 1, wx.ALIGN_CENTRE)
sub2 = wx.FlexGridSizer(wx.HORIZONTAL)
clear = wx.Button(pl, size=(200, 40), label='清空')
send = wx.Button(pl, size=(200, 40), label='发送')
sub2.Add(clear, 1, wx.TOP | wx.LEFT)
sub2.Add(send, 1, wx.TOP | wx.RIGHT)
con.Add(sub2, 1, wx.ALIGN_CENTRE)
pl.SetSizer(con)
'''窗口''' '''绑定'''
self.Bind(wx.EVT_BUTTON, self.EditClear, clear)
self.Bind(wx.EVT_BUTTON, self.SendMessage, send)
self.Bind(wx.EVT_BUTTON, self.Start, sta)
self.Bind(wx.EVT_BUTTON, self.Break, end)
self.Bind(wx.EVT_BUTTON, self.WriteToExcel, hist)
'''绑定''' '''服务器准备工作'''
self.UserThreadList = []
self.onServe = False
addr = ('', 21567)
self.ServeSock = socket(AF_INET, SOCK_STREAM)
self.ServeSock.bind(addr)
self.ServeSock.listen(10)
'''服务器准备工作''' '''数据库准备工作,用于存储聊天记录'''
self.db = MySQLdb.connect('localhost', 'root', '', 'user_info')
self.cursor = self.db.cursor()
self.cursor.execute("select * from history order by time")
self.Text.SetValue('')
for data in self.cursor.fetchall(): #加载历史聊天记录
self.Text.AppendText('%s said:\n%s\nwhen %s\n\n' % (data[0], data[2], data[1]))
'''数据库准备工作,用于存储聊天记录''' #将聊天记录导出到EXCEl表中
def WriteToExcel(self,event):
wbk = xlwt.Workbook()
sheet = wbk.add_sheet('sheet 1')
self.cursor.execute("select * from history order by time")
sheet.write(0, 0, "User")
sheet.write(0, 1, "Datetime")
sheet.write(0, 5, "Message")
index = 0
for data in self.cursor.fetchall():
index = index + 1
Time = '%s'%data[1] #将datetime转成字符形式,否则直接写入Excel会变成时间戳
sheet.write(index,0,data[0])
sheet.write(index,1,Time) #写进EXCEL会变成时间戳
sheet.write(index,5,data[2])
wbk.save(r'D:\History_Dialog.xls') #启动服务器的服务线程
def Start(self,event):
if not self.onServe:
'''启动服务线程'''
self.onServe = True
mainThread = threading.Thread(target=self.on_serving, args=())
mainThread.setDaemon(True) # 解决父线程结束,子线程还继续运行的问题
mainThread.start()
'''启动服务线程''' #关闭服务器
def Break(self,event):
self.onServe = False #服务器主循环
def on_serving(self):
print '...On serving...'
while self.onServe:
UserSocket, UserAddr = self.ServeSock.accept()
username = UserSocket.recv(1024).decode(encoding='utf-8') #接收用户名
userthread = ClientThread(UserSocket, username,self)
self.UserThreadList.append(userthread) #将用户线程加到队列中
userthread.start()
self.ServeSock.close() #绑定发送按钮
def SendMessage(self,event):
if self.onServe and cmp(self.ttex.GetValue(),''):
data = self.ttex.GetValue()
self.AddText('Server',data,time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()))
self.ttex.SetValue('') # 向所有客户端(包括自己)发送信息,同时更新到数据库
def AddText(self, source, data,Time):
self.cursor.execute("insert into history values(\"%s\",\"%s\",\"%s\")" % (source,Time,data)) #双引号里面有双引号,bug:句子不能有双引号、以及中文
self.db.commit()
sendData = '%s said:\n%s\nwhen %s\n' % (source,data,Time)
self.Text.AppendText('%s\n'%sendData)
for user in self.UserThreadList: #bug:客户端关闭了仍然在队列中。如果客户端关闭了,那怎么在服务器判断是否已经关闭了?客户端在关闭之前发一条信息给服务器?
user.UserSocket.send(sendData.encode(encoding='utf-8')) #绑定清空按钮
def EditClear(self,event):
self.ttex.Clear() def main():
app = wx.App(False)
Server().Show()
app.MainLoop() if __name__ == '__main__':
main()
服务器的客户线程Clientthread:
# -*- coding: UTF-8 -*- import threading
import time class ClientThread(threading.Thread): def __init__(self,UserSocket, Username,server):
threading.Thread.__init__(self)
self.UserSocket = UserSocket
self.Username = Username
self.server = server
self.Loadhist() # 加载历史聊天记录
def Loadhist(self):
self.server.cursor.execute("select * from history order by time")
for data in self.server.cursor.fetchall():
time.sleep(0.6) #几条信息同时发,会造成末尾回车键的丢失,所以要有时间间隔
sendData = '%s said:\n%s\nwhen %s\n'%(data[0], data[2], data[1])
self.UserSocket.send(sendData.encode(encoding='utf-8')) #方法重写,线程的入口
def run(self):
size = 1024
while True:
data = self.UserSocket.recv(size) #未解决:客户端断开连接后这里会报错
self.server.AddText(self.Username,data.decode(encoding='utf-8'),time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()))
self.UserSocket.close() #这里都执行不到
客户登录界面Logframe:
# -*- coding: UTF-8 -*- from socket import *
import wx
import MySQLdb
from client import Client class LogFrame(wx.Frame):
def __init__(self,parent=None,id=-1,title='登录窗口',pos=wx.DefaultPosition,size=(500,300)): '''窗口'''
wx.Frame.__init__(self,parent,id,title,pos,size=(400,280))
self.pl = wx.Panel(self)
con = wx.BoxSizer(wx.VERTICAL)
subcon = wx.FlexGridSizer(2,2,10,10)
username = wx.StaticText(self.pl, label="Username:",style=wx.ALIGN_LEFT)
password = wx.StaticText(self.pl, label="Password:",style=wx.ALIGN_LEFT)
self.tc1 = wx.TextCtrl(self.pl,size=(180,20))
self.tc2 = wx.TextCtrl(self.pl,size=(180,20),style=wx.TE_PASSWORD)
subcon.Add(username,wx.TE_LEFT)
subcon.Add(self.tc1,1,wx.EXPAND)
subcon.Add(password)
subcon.Add(self.tc2,1,wx.EXPAND)
con.Add(subcon,1,wx.ALIGN_CENTER)
subcon2 = wx.FlexGridSizer(1,2,10,10)
register = wx.Button(self.pl,label='Register')
login = wx.Button(self.pl,label='Login')
subcon2.Add(register,1, wx.TOP)
subcon2.Add(login,1, wx.TOP)
con.Add(subcon2,1,wx.ALIGN_CENTRE)
self.pl.SetSizer(con)
self.Bind(wx.EVT_BUTTON,self.Register,register)
self.Bind(wx.EVT_BUTTON,self.Login,login)
'''窗口'''
self.isConnected = False
self.userSocket = None #连接到服务器
def ConnectToServer(self):
if not self.isConnected:
ADDR = ('localhost', 21567)
self.userSocket = socket(AF_INET, SOCK_STREAM)
try:
self.userSocket.connect(ADDR)
self.userSocket.send(self.tc1.GetValue().encode(encoding='utf-8'))
self.isConnected = True
return True
except Exception:
return False
else:
return True #登录
def Login(self,event):
if not self.ConnectToServer():
err = wx.MessageDialog(None, '服务器未启动', 'ERROR!', wx.OK)
err.ShowModal()
err.Destroy()
else:
username = self.tc1.GetValue()
password = self.tc2.GetValue()
db = MySQLdb.connect('localhost', 'root', '', 'user_info')
cursor = db.cursor()
cursor.execute("select * from user_list where username='%s' and password='%s'"%(username,password))
if not cursor.fetchone():
err = wx.MessageDialog(None,'用户不存在或密码错误','ERROR!',wx.OK)
err.ShowModal()
else:
self.Close()
Client(opSock=self.userSocket, username=username).Show()
db.commit()
db.close() #注册
def Register(self,event):
if not self.ConnectToServer():
err = wx.MessageDialog(None, '服务器未启动', 'ERROR!', wx.OK)
err.ShowModal()
err.Destroy()
else:
username = self.tc1.GetValue()
password = self.tc2.GetValue()
db = MySQLdb.connect('localhost', 'root', '', 'user_info')
cursor = db.cursor()
cursor.execute("select * from user_list where username='%s'"%username)
if not cursor.fetchone():
cursor.execute("insert into user_list(username,password) values('%s','%s')"%(username,password))
else:
err = wx.MessageDialog(None, '用户已存在', 'ERROR!', wx.OK)
err.ShowModal()
db.commit()
db.close() def main():
app = wx.App(False)
LogFrame().Show()
app.MainLoop() if __name__ == '__main__':
main()
客户端Client:
#/usr/bin/env python
# -*- coding: UTF-8 -*- import wx
import threading
from time import ctime class Client(wx.Frame):
def __init__(self,opSock,username,parent=None,id=-1,title='客户端',pos=wx.DefaultPosition,size=(500,300)): '''窗口'''
wx.Frame.__init__(self,parent,id,title,pos,size=(400,470))
self.opSock = opSock
self.username = username
pl = wx.Panel(self)
con = wx.BoxSizer(wx.VERTICAL)
subcon = wx.FlexGridSizer(wx.HORIZONTAL)
sta = wx.Button(pl, size=(200, 40),label='连接')
end = wx.Button(pl, size=(200, 40),label='断开')
subcon.Add(sta, 1, wx.TOP|wx.LEFT)
subcon.Add(end, 1, wx.TOP|wx.RIGHT)
con.Add(subcon,1,wx.ALIGN_CENTRE)
self.Text = wx.TextCtrl(pl, size=(400,250),style = wx.TE_MULTILINE|wx.TE_READONLY)
con.Add(self.Text, 1, wx.ALIGN_CENTRE)
self.ttex = wx.TextCtrl(pl, size=(400,100),style=wx.TE_MULTILINE)
con.Add(self.ttex, 1, wx.ALIGN_CENTRE)
sub2 = wx.FlexGridSizer(wx.HORIZONTAL)
clear = wx.Button(pl, size=(200, 40), label='清空')
send = wx.Button(pl, size=(200, 40), label='发送')
sub2.Add(clear, 1, wx.TOP | wx.LEFT)
sub2.Add(send, 1, wx.TOP | wx.RIGHT)
con.Add(sub2, 1, wx.ALIGN_CENTRE)
pl.SetSizer(con)
'''窗口''' '''绑定'''
self.Bind(wx.EVT_BUTTON, self.EditClear, clear)
self.Bind(wx.EVT_BUTTON, self.Send, send)
self.Bind(wx.EVT_BUTTON, self.Login, sta)
self.Bind(wx.EVT_BUTTON, self.Logout, end)
'''绑定'''
self.isConnected = False #登录
def Login(self,event):
'''客户端准备工作'''
self.isConnected = True
t = threading.Thread(target=self.Receive, args=())
t.setDaemon(True)
t.start()
'''客户端准备工作''' #退出
def Logout(self,event):
self.isConnected = False #绑定发送按钮
def Send(self,event):
if self.isConnected and cmp(self.ttex.GetValue(),''):
self.opSock.send(self.ttex.GetValue().encode(encoding='utf-8'))
self.ttex.SetValue('') #绑定清空按钮
def EditClear(self,event):
self.ttex.Clear() #接收客户端的信息(独立一个线程)
def Receive(self):
while self.isConnected:
data = self.opSock.recv(1024).decode(encoding='utf-8')
self.Text.AppendText('%s\n'%data)
Python多人聊天室的更多相关文章
- Python实现网络多人聊天室
网络多人聊天室 文件结构: chatroom ├── client.py # 客户端代码 ├── language.py # 语言文件 ├── server.py # 服务端代码 └── set ...
- Python实现网络图形化界面多人聊天室 - Windows
Python实现网络图形化界面多人聊天室 - Windows 项目名称:网络多人聊天室图形界面版本 项目思路: server.py 服务端文件,主进程中,创建图形化界面,询问地址(主机名,端口),点击 ...
- Python实现网络图形化界面多人聊天室 - Linux
网络图形化界面多人聊天室 - Linux Windows版本:https://www.cnblogs.com/noonjuan/p/12078524.html 在Python实现网络多人聊天室基础上, ...
- Python实现网络多人聊天室 - Windows
项目名称:多人聊天室项目结构: client.py server.py settings.py项目思路:服务端接收客户端连接,客户端发送信息给服务端,服务端将信息发送给所有客户端.项目实现:主进程负责 ...
- Apache MiNa 实现多人聊天室
Apache MiNa 实现多人聊天室 开发环境: System:Windows JavaSDK:1.6 IDE:eclipse.MyEclipse 6.6 开发依赖库: Jdk1.4+.mina-c ...
- 与众不同 windows phone (31) - Communication(通信)之基于 Socket UDP 开发一个多人聊天室
原文:与众不同 windows phone (31) - Communication(通信)之基于 Socket UDP 开发一个多人聊天室 [索引页][源码下载] 与众不同 windows phon ...
- 与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室
原文:与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室 [索引页][源码下载] 与众不同 windows phon ...
- Spring整合DWR comet 实现无刷新 多人聊天室
用dwr的comet(推)来实现简单的无刷新多人聊天室,comet是长连接的一种.通常我们要实现无刷新,一般会使用到Ajax.Ajax 应用程序可以使用两种基本的方法解决这一问题:一种方法是浏览器每隔 ...
- 多人聊天室(Java)
第1部分 TCP和UDP TCP:是一种可靠地传输协议,是把消息按一个个小包传递并确认消息接收成功和正确才发送下一个包,速度相对于UDP慢,但是信息准确安全:常用于一般不要求速度和需要准确发送消息的场 ...
随机推荐
- struts2.16中文乱码问题解决
方法1.在struts.xml文件中添加<constant name="struts.i18n.encoding" value="GBK" /> 方 ...
- java把一个文件的内容复制到另外一个文件
/** * java把一个文件的内容复制到另外一个文件 */import java.io.File;import java.io.FileInputStream;import java.io.File ...
- An internal error occurred Exception caught during execution of commit command
在工程目录下找到 .git 文件夹 ,找到里面的 index.lock 文件,删掉再commit
- vue2.0 仿手机新闻站(四)axios
1.axios的配置 main.js import Vue from 'vue' import App from './App.vue' // 引入 路由 import VueRouter from ...
- 提高SharePoint2013服务器性能
一劳永逸,删除search services application,停止Windows服务:SharePoint Search Host Controller和SharePoint Server S ...
- Chrome禁用NPAPI插件(包含 Silverlight、Java 和 Unity)
过去,很多插件都是使用一种称为NPAPI 的旧系统开发的. 现在,仅仅有少量站点在使用NPAPI 插件,由于这些插件有时会给站点带来安全风险. 为了让用户获得更安全.更高速且更稳定的 Chrome 浏 ...
- jquery中的clone()方法
jquery中不能直接把选择到的元素如$('div')添加到其他地方,而需要使用$('div')[0]等 clone()方法直接复制HTML代码,所以可以直接用来添加元素.
- POJ 1787 Charlie's Change
多重背包 可行性+路径记录 题意是说你要用很多其它的零钱去买咖啡.最后输出你分别要用的 1,5 ,10 .25 的钱的数量. 多重背包二进制分解.然后记录下 这个状态.最后逆向推就可以. #inclu ...
- 如何利用hibernate3解决数据库丢失更新问题?
首先我们要明白什么叫丢失更新. 比如数据库有一个person表,里面有一条这样的数据 "5 zhangsan shenzhen"; 现在有两个事务A.B同时查找了这一条记录: A事 ...
- python 基础 5.1 python 构造器
一. 类的构造器 __init__ 构造函数,在生成对象时调用.由于类可以起到模板的作用,因此,可以在创建实例的时候,把一些我们认为必须绑定的属性强制填写进去.通过定义一个特殊的__init__方法, ...