电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

Java Socket+多線程實現多人聊天室功能

瀏覽:46日期:2022-08-09 08:30:23

本文實例為大家分享了Java Socket+多線程實現多人聊天室的具體代碼,供大家參考,具體內容如下

思路簡介

分為客戶端和服務器兩個類,所有的客戶端將聊的內容發(fā)送給服務器,服務器接受后,將每一條內容發(fā)送給每一個客戶端,客戶端再顯示在終端上。

客戶端設計

客戶端包含2個線程,1個用來接受服務器的信息,再顯示,1個用來接收鍵盤的輸入,發(fā)送給服務器。

import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.net.Socket;import java.nio.charset.StandardCharsets;import java.util.Scanner; public class WeChatClient { //WeChat的客戶端類 private Socket client; private String name; private InputStream in; private OutputStream out; private MassageSenter massageSenter; private MassageGeter massageGeter; class MassageGeter extends Thread{ //一個子線程類,用于客戶端接收消息MassageGeter() throws IOException{ in = client.getInputStream();}@Overridepublic void run() { int len; byte[] bytes = new byte[1024]; try {while ((len = in.read(bytes)) != -1) { //此函數是阻塞的 System.out.println(new String(bytes,0,len, StandardCharsets.UTF_8));} }catch (IOException e){System.out.println(e.toString()); } System.out.println('Connection interruption');} } class MassageSenter extends Thread{ //一個子線程類,用于發(fā)送消息給服務器MassageSenter() throws IOException{ out = client.getOutputStream();} @Overridepublic void run() { Scanner scanner = new Scanner(System.in); try {while (scanner.hasNextLine()) { //此函數為阻塞的函數 String massage = scanner.nextLine(); out.write((name + ' : ' + massage).getBytes(StandardCharsets.UTF_8)); if(massage.equals('//exit'))break;} }catch (IOException e){e.printStackTrace(); }} } WeChatClient(String name, String host, int port) throws IOException {//初始化,實例化發(fā)送和接收2個線程this.name = name;client = new Socket(host,port);massageGeter = new MassageGeter();massageSenter = new MassageSenter(); } void login() throws IOException{//登錄時,先發(fā)送名字給服務器,在接收到服務器的正確回應之后,啟動線程out.write(name.getBytes(StandardCharsets.UTF_8));byte[] bytes = new byte[1024];int len;len = in.read(bytes);String answer = new String(bytes,0,len, StandardCharsets.UTF_8);if(answer.equals('logined!')) { System.out.println('Welcome to WeChat! '+name); massageSenter.start(); massageGeter.start(); try {massageSenter.join();//join()的作用是等線程結束之后再繼續(xù)執(zhí)行主線程(main)massageGeter.join(); }catch (InterruptedException e){System.err.println(e.toString()); } }else{ System.out.println('Server Wrong');}client.close(); } public static void main(String[] args) throws IOException{//程序入口String host = '127.0.0.1';WeChatClient client = new WeChatClient('Uzi',host,7777);client.login(); } }服務器設計

服務器包含3個線程類,端口監(jiān)聽線程,客戶端接收信息線程,發(fā)送信息線程。

服務器類還包含并維護著一個已經連接的用戶列表,和一個待發(fā)送信息列表。

服務器有一個負責監(jiān)聽端口的線程,此線程在接收到客戶端的連接請求后,將連接的客戶端添加進用戶列表;并為每一個連接的客戶端實例化一個接受信息的線程類,從各個客戶端接收員信息,并存入待發(fā)送信息列表。

發(fā)送信息線程查看列表是否為空,若不為空,則將里面的信息發(fā)送給用戶列表的每一個用戶。

import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.net.ServerSocket;import java.net.Socket;import java.nio.charset.StandardCharsets;import java.util.ArrayList; public class WeChatServer { private ServerSocket server; private ArrayList<User> users;//用戶列表 private ArrayList<String> massages;//待發(fā)送消息隊列 private Listener listener; private MassageSenter massageSenter; class User{ //用戶類,包含用戶的登錄id和一個輸出流String name;OutputStream out;User(String name,OutputStream out){ this.name = name; this.out = out;} @Overridepublic String toString() { return name;} } private static String GetMassage(InputStream in) throws IOException{//從一個輸入流接收一個字符串int len;byte[] bytes = new byte[1024];len = in.read(bytes);return new String(bytes,0,len,StandardCharsets.UTF_8); } private void UserList(){ //列出當前在線用戶,調試用for(User user : users) System.out.println(user); } class Listener extends Thread{ //監(jiān)聽線程類,負則監(jiān)聽是否有客戶端連接@Overridepublic void run() { try {while (true) { Socket socket = server.accept();//此函數是阻塞的 InputStream in = socket.getInputStream(); String name = GetMassage(in);//獲取接入用戶的name System.out.println(name +' has connected'); massages.add(name+' has joined just now!!');//向聊天室報告用戶連入的信息 OutputStream out = socket.getOutputStream(); out.write('logined!'.getBytes(StandardCharsets.UTF_8));//發(fā)送成功建立連接的反饋 User user = new User(name,out); users.add(user);//添加至在線用戶列表 MassageListener listener = new MassageListener(user,in);//創(chuàng)建用于接收此用戶信息的線程 listener.start();} }catch (IOException e){e.printStackTrace(); }} } class MassageListener extends Thread{ //接收線程類,用于從一個客戶端接收信息,并加入待發(fā)送列表private User user;private InputStream in;MassageListener(User user,InputStream in){ this.user = user; this.in = in;} @Overridepublic void run() { try {while (true){ String massage = GetMassage(in); System.out.println('GET MASSAGE '+massage); if(massage.contains('//exit')){ // '/exit' 是退出指令break; } massages.add(massage);}//用戶退出有兩種形式,輸入 “//exit” 或者直接關閉程序in.close();user.out.close(); }catch (IOException e){//此異常是處理客戶端異常關閉,即GetMassage(in)調用會拋出異常,因為in出入流已經自動關閉e.printStackTrace(); }finally {System.out.println(user.name+' has exited!!');massages.add(user.name+' has exited!!');users.remove(user);//必須將已經斷開連接的用戶從用戶列表中移除,否則會在發(fā)送信息時產生異常System.out.println('Now the users has');UserList(); } } } private synchronized void SentToAll(String massage)throws IOException{//將信息發(fā)送給每一個用戶,加入synchronized修飾,保證在發(fā)送時,用戶列表不會被其他線程更改if(users.isEmpty()) return;for(User user : users){ user.out.write(massage.getBytes(StandardCharsets.UTF_8));} } class MassageSenter extends Thread{//消息發(fā)送線程 @Overridepublic void run() { while(true){try{ sleep(1);//此線程中沒有阻塞的函數,加入沉睡語句防止線程過多搶占資源}catch (InterruptedException e){ e.printStackTrace();}if(!massages.isEmpty()){ String massage = massages.get(0); massages.remove(0); try {SentToAll(massage); }catch (IOException e){e.printStackTrace(); } } }} } WeChatServer(int port) throws IOException { //初始化server = new ServerSocket(port);users = new ArrayList<>();massages = new ArrayList<>();listener = new Listener();massageSenter = new MassageSenter(); } private void start(){ //線程啟動listener.start();massageSenter.start(); } public static void main(String[] args) throws IOException{WeChatServer server = new WeChatServer(7777);server.start(); } }總結

之所以需要多線程編程,是因為有的函數是阻塞的,例如

while ((len = in.read(bytes)) != -1) { //此函數是阻塞的 System.out.println(new String(bytes,0,len, StandardCharsets.UTF_8));}

while (scanner.hasNextLine()) { //此函數為阻塞的函數String massage = scanner.nextLine();out.write((name + ' : ' + massage).getBytes(StandardCharsets.UTF_8));if(massage.equals('//exit')) break; }

Socket socket = server.accept();//此函數是阻塞的

這些阻塞的函數是需要等待其他的程序,例如scanner.hasNextLine()需要等待程序員的輸入才會返回值,in.read需要等待流的另一端傳輸數據,使用多線程就可以在這些函數處于阻塞狀態(tài)時,去運行其他的線程。

所以,多線程編程的關鍵便是那些阻塞的函數。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Java
相關文章:
主站蜘蛛池模板: 蓝莓施肥机,智能施肥机,自动施肥机,水肥一体化项目,水肥一体机厂家,小型施肥机,圣大节水,滴灌施工方案,山东圣大节水科技有限公司官网17864474793 | 优宝-汽车润滑脂-轴承润滑脂-高温齿轮润滑油脂厂家 | 储能预警-储能消防系统-电池舱自动灭火装置-四川千页科技股份有限公司官网 | 并网柜,汇流箱,电控设备,中高低压开关柜,电气电力成套设备,PLC控制设备订制厂家,江苏昌伟业新能源科技有限公司 | 间苯二酚,间苯二酚厂家-淄博双和化工 | 学生作文网_中小学生作文大全与写作指导 | 电动高尔夫球车|电动观光车|电动巡逻车|电动越野车厂家-绿友机械集团股份有限公司 | ★店家乐|服装销售管理软件|服装店收银系统|内衣店鞋店进销存软件|连锁店管理软件|收银软件手机版|会员管理系统-手机版,云版,App | 吸音板,隔音板,吸音材料,吸音板价格,声学材料 - 佛山诺声吸音板厂家 | 澳门精准正版免费大全,2025新澳门全年免费,新澳天天开奖免费资料大全最新,新澳2025今晚开奖资料,新澳马今天最快最新图库-首页-东莞市傲马网络科技有限公司 | 佛山市钱丰金属不锈钢蜂窝板定制厂家|不锈钢装饰线条|不锈钢屏风| 电梯装饰板|不锈钢蜂窝板不锈钢工艺板材厂家佛山市钱丰金属制品有限公司 | 辊道窑炉,辊道窑炉厂家-山东艾希尔| 真空冷冻干燥机_国产冻干机_冷冻干燥机_北京四环冻干 | 上海单片机培训|重庆曙海培训分支机构—CortexM3+uC/OS培训班,北京linux培训,Windows驱动开发培训|上海IC版图设计,西安linux培训,北京汽车电子EMC培训,ARM培训,MTK培训,Android培训 | 二氧化碳/活性炭投加系统,次氯酸钠发生器,紫外线消毒设备|广州新奥 | 华中线缆有限公司-电缆厂|电缆厂家|电线电缆厂家 | 引领中高档酒店加盟_含舍·美素酒店品牌官网| 净水器代理,净水器招商,净水器加盟-FineSky德国法兹全屋净水 | 电气控制系统集成商-PLC控制柜变频控制柜-非标自动化定制-电气控制柜成套-NIDEC CT变频器-威肯自动化控制 | 智能电表|预付费ic卡水电表|nb智能无线远传载波电表-福建百悦信息科技有限公司 | 深圳富泰鑫五金_五金冲压件加工_五金配件加工_精密零件加工厂 | 热风机_工业热风机生产厂家上海冠顶公司提供专业热风机图片价格实惠 | 上海小程序开发-小程序制作-上海小程序定制开发公司-微信商城小程序-上海咏熠 | 食品机械专用传感器-落料放大器-低价接近开关-菲德自控技术(天津)有限公司 | 塑料瓶罐_食品塑料瓶_保健品塑料瓶_调味品塑料瓶–东莞市富慷塑料制品有限公司 | 高低温万能试验机-复合材料万能试验机-馥勒仪器 | 道康宁消泡剂-瓦克-大川进口消泡剂供应商| 水性绝缘漆_凡立水_绝缘漆树脂_环保绝缘漆-深圳维特利环保材料有限公司 | 阴离子聚丙烯酰胺价格_PAM_高分子聚丙烯酰胺厂家-河南泰航净水材料有限公司 | 碳钢法兰厂家,非标法兰,定制异型,法兰生产厂家-河北九瑞管道 | 悬浮拼装地板_篮球场木地板翻新_运动木地板价格-上海越禾运动地板厂家 | 上海办公室装修公司_办公室设计_直营办公装修-羚志悦装 | 天津货架厂_穿梭车货架_重型仓储货架_阁楼货架定制-天津钢力仓储货架生产厂家_天津钢力智能仓储装备 | 断桥铝破碎机_铝合金破碎机_废铁金属破碎机-河南鑫世昌机械制造有限公司 | 精密钢管,冷拔精密无缝钢管,精密钢管厂,精密钢管制造厂家,精密钢管生产厂家,山东精密钢管厂家 | 洛阳装修公司-洛阳整装一站式品牌-福尚云宅装饰 | arch电源_SINPRO_开关电源_模块电源_医疗电源-东佑源 | 电动葫芦|手拉葫芦|环链电动葫芦|微型电动葫芦-北京市凌鹰起重机械有限公司 | 伸缩节_伸缩器_传力接头_伸缩接头_巩义市联通管道厂 | 电缆接头_防水接头_电缆防水接头_防水电缆接头_上海闵彬 | 首页|光催化反应器_平行反应仪_光化学反应仪-北京普林塞斯科技有限公司 |