久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術(shù)文章
文章詳情頁

Java Socket+多線程實(shí)現(xiàn)多人聊天室功能

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

本文實(shí)例為大家分享了Java Socket+多線程實(shí)現(xiàn)多人聊天室的具體代碼,供大家參考,具體內(nèi)容如下

思路簡(jiǎn)介

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

客戶端設(shè)計(jì)

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

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{ //一個(gè)子線程類,用于客戶端接收消息MassageGeter() throws IOException{ in = client.getInputStream();}@Overridepublic void run() { int len; byte[] bytes = new byte[1024]; try {while ((len = in.read(bytes)) != -1) { //此函數(shù)是阻塞的 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{ //一個(gè)子線程類,用于發(fā)送消息給服務(wù)器MassageSenter() throws IOException{ out = client.getOutputStream();} @Overridepublic void run() { Scanner scanner = new Scanner(System.in); try {while (scanner.hasNextLine()) { //此函數(shù)為阻塞的函數(shù) 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 {//初始化,實(shí)例化發(fā)送和接收2個(gè)線程this.name = name;client = new Socket(host,port);massageGeter = new MassageGeter();massageSenter = new MassageSenter(); } void login() throws IOException{//登錄時(shí),先發(fā)送名字給服務(wù)器,在接收到服務(wù)器的正確回應(yīng)之后,啟動(dòng)線程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()的作用是等線程結(jié)束之后再繼續(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(); } }服務(wù)器設(shè)計(jì)

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

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

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

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

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ā)送消息隊(duì)列 private Listener listener; private MassageSenter massageSenter; class User{ //用戶類,包含用戶的登錄id和一個(gè)輸出流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{//從一個(gè)輸入流接收一個(gè)字符串int len;byte[] bytes = new byte[1024];len = in.read(bytes);return new String(bytes,0,len,StandardCharsets.UTF_8); } private void UserList(){ //列出當(dāng)前在線用戶,調(diào)試用for(User user : users) System.out.println(user); } class Listener extends Thread{ //監(jiān)聽線程類,負(fù)則監(jiān)聽是否有客戶端連接@Overridepublic void run() { try {while (true) { Socket socket = server.accept();//此函數(shù)是阻塞的 InputStream in = socket.getInputStream(); String name = GetMassage(in);//獲取接入用戶的name System.out.println(name +' has connected'); massages.add(name+' has joined just now!!');//向聊天室報(bào)告用戶連入的信息 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{ //接收線程類,用于從一個(gè)客戶端接收信息,并加入待發(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” 或者直接關(guān)閉程序in.close();user.out.close(); }catch (IOException e){//此異常是處理客戶端異常關(guān)閉,即GetMassage(in)調(diào)用會(huì)拋出異常,因?yàn)閕n出入流已經(jīng)自動(dòng)關(guān)閉e.printStackTrace(); }finally {System.out.println(user.name+' has exited!!');massages.add(user.name+' has exited!!');users.remove(user);//必須將已經(jīng)斷開連接的用戶從用戶列表中移除,否則會(huì)在發(fā)送信息時(shí)產(chǎn)生異常System.out.println('Now the users has');UserList(); } } } private synchronized void SentToAll(String massage)throws IOException{//將信息發(fā)送給每一個(gè)用戶,加入synchronized修飾,保證在發(fā)送時(shí),用戶列表不會(huì)被其他線程更改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);//此線程中沒有阻塞的函數(shù),加入沉睡語句防止線程過多搶占資源}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(){ //線程啟動(dòng)listener.start();massageSenter.start(); } public static void main(String[] args) throws IOException{WeChatServer server = new WeChatServer(7777);server.start(); } }總結(jié)

之所以需要多線程編程,是因?yàn)橛械暮瘮?shù)是阻塞的,例如

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

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

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

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

所以,多線程編程的關(guān)鍵便是那些阻塞的函數(shù)。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 成人午夜 | 亚洲二区在线视频 | 久久精品网 | 国产精品久久久久久久久岛 | 91精品国产一区二区三区四区在线 | 国产精品一区二区无线 | 黄色网址大全在线观看 | 99精品电影| 亚洲欧美一区二区三区久久 | 中文字字幕一区二区三区四区五区 | 午夜视频在线免费观看 | 欧美黄色片 | 欧美日韩一 | 欧美激情网站 | 日韩在线观看视频免费 | 成人免费视频视频在线观看 免费 | 精品av| 国产成人一区二区三区影院在线 | 日本久久久久久久久 | h免费在线观看 | 日韩成人影院在线观看 | 一区二区视频在线 | 久久手机免费视频 | 欧美日韩美女 | 精国产品一区二区三区 | 国产精品一区二区三区在线 | a国产在线观看 | 亚洲一区 国产 | 日韩在线播放一区 | 国产一区二区视频在线观看 | 日韩欧美国产一区二区 | 天天摸夜夜摸爽爽狠狠婷婷97 | 一区二区三区观看视频 | 亚洲精品一区二区在线观看 | 黄色片免费看 | 五月婷婷综合网 | 久久亚洲一区二区 | 精品久久久网站 | 爽死777影院 | 国产精品视频一区二区三区 | a级片在线观看 |