websocket
from abc import ABC import tornado.web import tornado.httpserver import tornado.ioloop from tornado.options import define, options import tornado.web import tornado.websocket class IndexHandler(tornado.web.RequestHandler, ABC): # 定义首页视图处理类,提示用户登录 def get(self): self.render('index.html') class LoginHandler(tornado.web.RequestHandler, ABC): # 定义登录视图处理类 def get(self): # 获取用户登录的昵称 nickname = self.get_argument('nickname') # 将用户登录的昵称保存在cookie中,安全cookie self.set_secure_cookie('nickname', nickname) self.render('chat.html', nickname=nickname) class ChatHandler(tornado.websocket.WebSocketHandler, ABC): # 定义接收/发送聊天消息的视图处理类,继承自websocket的WebSocketHandler # 定义一个集合,用来保存在线的所有用户 online_users = set() # 从客户端获取cookie信息 # 重写open方法,当有新的聊天用户进入的时候自动触发该函数 def open(self): # 当有新的用户上线,将该用户加入集合中 self.online_users.add(self) # 将新用户加入的信息发送给所有的在线用户 for user in self.online_users: user.write_message('【%s】进入了聊天室' % self.request.remote_ip) # 重写on_message方法,当聊天消息有更新时自动触发的函数 def on_message(self, message): # 将在线用户发送的消息通过服务器转发给所有的在线用户 for user in self.online_users: user.write_message('%s:%s' % (self.request.remote_ip, message)) # 重写on_close方法,当有用户离开时自动触发的函数 def on_close(self): # 先将用户从列表中移除 self.online_users.remove(self) # 将该用户离开的消息发送给所有在线的用户 for user in self.online_users: user.write_message('【%s】离开了聊天室~' % self.request.remote_ip) # 重写check_origin方法, 解决WebSocket的跨域请求 def check_origin(self, origin): return True define("port", default=8180, help="run on the given port", type=int) def make_app(): return tornado.web.Application(handlers=[ (r'/', IndexHandler), (r'/login', LoginHandler), (r'/chat', ChatHandler), ], debug=True, cookie_secret='cqVJzSSjQgWzKtpHMd4NaSeEa6yTy0qRicyeUDIMSjo=' ) # 程序运行入口 if __name__ == '__main__': app = make_app() http_server = tornado.httpserver.HTTPServer(app) http_server.listen(options.port) tornado.ioloop.IOLoop.current().start()

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>聊天室登录首页</title> <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.js"></script> </head> <body> <div> <div style="width:60%;"> <div> 聊天室个人登录 </div> <div> <form method="get" action="/login" style="width:80%"> <p>昵称:<input type="text" placeholder="请输入昵称" name="nickname"></p> <button type="submit">登录</button> </form> </div> </div> </div> </body> </html>

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Tornado聊天室</title> <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.js"></script> </head> <body> <div> <div style="font-weight:bold; font-size:2em;"> 聊天室 </div> <div class="content"> <div class="receive"> </div> <div class="send"> <textarea id="send_content"> </textarea> <br> <button id="btn" >发送</button> </div> </div> </div> <script> // 创建一个websocket长连接对象 var _websocket= new WebSocket('ws://127.0.0.1:8180/chat') //发送消息 $('#btn').click(function(){ //获取消息内容 $msg=$('#send_content').val(); //发送消息 _websocket.send($msg); $("#msg").val(''); }) //接收消息,当消息更新时自动触发 _websocket.onmessage=function(e){ console.log(e) var $content=e.data; //重构date的Format属性 Date.prototype.Format = function (fmt) { var o = { "M+": this.getMonth() + 1, //月份 "d+": this.getDate(), //日 "H+": this.getHours(), //小时 "m+": this.getMinutes(), //分 "s+": this.getSeconds(), //秒 "q+": Math.floor((this.getMonth() + 3) / 3), //季度 "S": this.getMilliseconds() //毫秒 }; if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length)); for (var k in o) if (new RegExp("(" + k + ")").test(fmt)) fmt = fmt.replace(RegExp.$1, (RegExp.$1.length === 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length))); return fmt; } var date=new Date(); var $time=date.Format("yyyy年MM月dd HH:mm:ss"); // 添加内容到class为recceive的div框中 var $p1=$('<p style="color:red;">').text($content); var $p2=$('<p style="color:#000000;">').text($time); $('.receive').append($p2); $('.receive').append($p1) } </script> </body> </html>