python实现多人聊天室

本文实例为大家分享了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', '123456', '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', '123456', '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', '123456', '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)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 教你用Python创建微信聊天机器人

    最近研究微信API,发现个非常好用的python库:wxpy.wxpy基于itchat,使用了 Web 微信的通讯协议,实现了微信登录.收发消息.搜索好友.数据统计等功能. 这里我们就来介绍一下这个库,并在最后实现一个聊天机器人. 有没有很兴奋?有没有很期待? 好了,接下来,开始我们的正题. 准备工作 安装非常简单,从官方源下载安装 pip install -U wxpy 或者从豆瓣源安装 pip install -U wxpy -i "https://pypi.doubanio.com/sim

  • Python实现基于C/S架构的聊天室功能详解

    本文实例讲述了Python实现基于C/S架构的聊天室功能.分享给大家供大家参考,具体如下: 一.课程介绍 1.简介 本次项目课是实现简单聊天室程序的服务器端和客户端. 2.知识点 服务器端涉及到asyncore.asynchat和socket这几个模块,客户端用到了telnetlib.wx.time和thread这几个模块. 3.所需环境 本次课中编写客户端需要用到wxPython,它是一个GUI工具包,请先使用下面的命令安装: $ sudo apt-get install python-wxt

  • 使用Python AIML搭建聊天机器人的方法示例

    AIML全名为Artificial Intelligence Markup Language(人工智能标记语言),是一种创建自然语言软件代理的XML语言,是由RichardS. Wallace 博士和Alicebot开源软件组织于1995-2000年间发明创造的.AIML是一种为了匹配模式和确定响应而进行规则定义的 XML 格式. AIML的设计目标如下: AIML应当为大众所易学易会. AIML应当使最小的概念得以编码使之基于L.I.C.E支持一种刺激-响应学科系统组件. AIML应当兼容XM

  • python使用udp实现聊天器功能

    聊天器简易版 使用udp实现一个简单的聊天器程序,要求如下: •在一个电脑中编写1个程序,有2个功能 •1.获取键盘数据,并将其发送给对方 •2.接收数据并显示 •并且功能数据进行选择以上的2个功能调用 例子程序如下: import socket def send_message(udp_socket): # 输入对方的ip/port dest_ip = input("请输入对方的ip:") dest_port = int(input("请输入对象的port:"))

  • Python+Socket实现基于TCP协议的客户与服务端中文自动回复聊天功能示例

    本文实例讲述了Python+Socket实现基于TCP协议的客户与服务端中文自动回复聊天功能.分享给大家供大家参考,具体如下: [吐槽] 网上的代码害死人,看着都写的言之凿凿,可运行就是有问题. 有些爱好代码.喜欢收藏代码的朋友,看到别人的代码就粘贴复制过来.可是起码你也试试运行看啊大哥 [正文] 昨日修改运行了UDP协议的C/S聊天程序,可是TCP协议的怎么都不行.各种试,各种坑. 做了下面几个修改后,终于可以了: 1.对发送.接收的信息,分别进行编码和解码 2.客户端的第10行bind改为c

  • 快速实现基于Python的微信聊天机器人示例代码

    最近听说一个很好玩的图灵机器人api,正好可以用它做一个微信聊天机器人,下面是实现 # test.py import requests import itchat #这是一个用于微信回复的库 KEY = '8edce3ce905a4c1dbb965e6b35c3834d' #这个key可以直接拿来用 # 向api发送请求 def get_response(msg): apiUrl = 'http://www.tuling123.com/openapi/api' data = { 'key' :

  • python实现的多任务版udp聊天器功能案例

    本文实例讲述了python实现的多任务版udp聊天器.分享给大家供大家参考,具体如下: 说明 编写一个有2个线程的程序 线程1用来接收数据然后显示 线程2用来检测键盘数据然后通过udp发送数据 要求 实现上述要求 总结多任务程序的特点 参考代码: import socket import threading def send_msg(udp_socket): """获取键盘数据,并将其发送给对方""" while True: # 1. 从键盘输入

  • Python socket C/S结构的聊天室应用实现

    Python socket C/S结构的聊天室应用 服务端: #!/usr/bin/env python #coding:utf8 import socket,select def broadcast_data (sock,message): for socket in conn_list: if socket != server_socket and socket != sock : try : socket.send(message) except : socket.close() conn

  • Python基于Socket实现的简单聊天程序示例

    本文实例讲述了Python基于Socket实现的简单聊天程序.分享给大家供大家参考,具体如下: 需求:SCIENCE 和MOOD两个人软件专业出身,厌倦了大众化的聊天软件,想着自己开发一款简易的聊天软件,满足他们的个性化需求,又不失"专业水准",Talk is easy, try to code it. 技术:socket,详细可参考前文:Python Socket实现简单TCP Server/client功能 语言:python 尽管socket区分服务器和客户端,但是在聊天程序中两

  • python实现多人聊天室

    本文实例为大家分享了python实现多人聊天室的具体代码,供大家参考,具体内容如下 一.目的 以实现小项目的方式,来巩固之前学过的Python基本语法以及相关的知识. 二.相关技术 1.wxpython GUI编程 2.网络编程 3.多线程编程 4.数据库编程 5.简单的将数据导出到Excel表 三.存在的漏洞以及不足 1.由于数据库编码的问题,无法使用中文. 2.在客户端关闭后,其相关的线程仍然存在于服务器的用户线程队列中,所以服务器会错误地往已关闭的客户端传送信息. 3.客户端初始登录并加载

  • python实现简单多人聊天室

    本文实例为大家分享了python实现多人聊天室的具体代码,供大家参考,具体内容如下 刚开始学习python,写了一个聊天室练练手. Server.py import socket,select,thread; host=socket.gethostname() port=5963 addr=(host,port) inputs=[] fd_name={} def who_in_room(w): name_list=[] for k in w: name_list.append(w[k]) ret

  • Python基于Socket实现简易多人聊天室的示例代码

    前言 套接字(Sockets)是双向通信信道的端点. 套接字可以在一个进程内,在同一机器上的进程之间,或者在不同主机的进程之间进行通信,主机可以是任何一台有连接互联网的机器. 套接字可以通过多种不同的通道类型实现:Unix域套接字,TCP,UDP等. 套接字库提供了处理公共传输的特定类,以及一个用于处理其余部分的通用接口. socket模块: 要创建套接字,必须使用套接字模块中的socket.socket()函数,该函数具有一般语法 s = socket.socket (socket_famil

  • 利用Python打造一个多人聊天室的示例详解

    一.实验名称 建立聊天工具 二.实验目的 掌握Socket编程中流套接字的技术,实现多台电脑之间的聊天. 三.实验内容和要求 vii.掌握利用Socket进行编程的技术 viii.必须掌握多线程技术,保证双方可以同时发送 ix.建立聊天工具 x.可以和多个人同时进行聊天 xi.必须使用图形界面,显示双方的语录 四.实验环境 PC多台,操作系统Win7,win10(32位.64位) 具备软件python3.6 . 五.操作方法与实验步骤 服务端 1.调入多线程.与scoket包,用于实现多线程连接

  • 使用Angular和Nodejs、socket.io搭建聊天室及多人聊天室

    一,利用Node搭建静态服务器 这个是这个项目的底层支撑部分.用来支持静态资源文件像html, css, gif, jpg, png, javascript, json, plain text等等静态资源的访问.这里面是有一个mime类型的文件映射. mime.js /** * mime类型的 map * @ author Cheng Liufeng * @ date 2014/8/30 * 当请求静态服务器文件的类型 html, css, gif, jpg, png, javascript,

  • 基于Nodejs利用socket.io实现多人聊天室

    socket.io简介 在Html5中存在着这样的一个新特性,引入了websocket,关于websocket的内部实现原理可以看这篇文章,这篇文章讲述了websocket无到有,根据协议,分析数据帧的头,进行构建websocket.虽然代码短,但可以很好地体现websocket的原理. ,这个特性提供了浏览器端和服务器端的基于TCP连接的双向通道.但是并不是所有的浏览器都支持websocket特性,故为了磨平浏览器间的差异,为开发者提供统一的接口,引入了socket.io模块.在不支持webs

  • nodejs+express搭建多人聊天室步骤

    前言 本文主要是笔者在学习node的时候,作为练手的一个小项目,花了几天空余时间,边码边写教程的一个过程.适用于对node理论知识看的多,实战少的同学,那么现在就让我们开始吧! 准备工作 新建一个文件夹 chatroom 在终端输入以下命令,按照步骤npm(没装过的去官网安装下node和npm)会自动给你生成一个package.json文件 安装express和socket.io package.json文件如下: //package.json { "name": "chat

  • 利用GO语言实现多人聊天室实例教程

    前言 运用go里面的net包中的相关方法来实现一个基于tcp的简单多人聊天室,用一个服务器来管理,主要反馈客户端是否连接成功并显示客户端输入的内容,并且发送给每一个在服务器上连接的客服端,下面话不多说了,来一起看看详细的介绍吧. 示例代码 服务器代码 // server package main import ( "fmt" "net" ) var ConnMap map[string]*net.TCPConn func checkErr(err error) in

  • Python socket实现简单聊天室

    本文实例为大家分享了Python socket实现简单聊天室的具体代码,供大家参考,具体内容如下 服务端使用了select模块,实现了对多个socket的监控.客户端由于select在Windows下只能对socket使用,所以使用了多线程来实现对客户端输入和socket连接的同时监控.注意这里的socket设置为了非阻塞.这样就实现了在一个线程中同时进行socket的接收和发送. 服务器代码: # -*- coding: utf-8 -*- import socket,select conne

  • Java基于中介者模式实现多人聊天室功能示例

    本文实例讲述了Java基于中介者模式实现多人聊天室功能.分享给大家供大家参考,具体如下: 一 模式定义 中介者模式,用一个中介对象来封装一系列对象之间的交互,使各个对象中不需要显示地引用其他对象实例,从而降低各个对象之间的耦合度,并且可以独立地改变对象间的交互关系. 二 模式举例 1 模式分析 我们借用多人聊天室来说明这一模式 2 中介模式静态类图 3 代码示例 3.1中介者接口--IMediator package com.demo.mediator; import com.demo.coll

随机推荐