소규모 LAN 채팅 애플리케이션을 구현하는 Java 프로그래밍 단계
开发环境:
IDEA 2018.2 集成开发工具。
实现功能:
1、用户上线,向服务器通知并注册。
2、同局域网下,所有注册用户可以进行群聊。
3、同局域网下,所有用户可与任意已注册用户进行私聊。
4、用户下线,通知服务器,服务器更新信息。
实现原理:
1、服务器端实例化一个ServerSocket对象,调用accept方法等待客户端连接到服务器。
2、客户端实例化 Socket 对象,并使用构造方法与服务器建立链接。
3、服务器端根据客户端输入信息,辨别客户端请求的功能从而做出相应响应。
实用技术:
为了能够高效的处理客户端的请求,在服务器端使用多线程处理客户端请求。并且使用 ConcurrentHashMap 来存储所有注册过的客户端。
项目源码(解释写在代码的注释当中):
服务器端:
import java.io.IOException; import java.io.PrintStream; import java.net.ServerSocket; import java.net.Socket; import java.util.Map; import java.util.Scanner; import java.util.Set; import java.util.concurrent.*; import java.util.regex.Matcher; import java.util.regex.Pattern; public class ManyThreadServer { //存储所有注册的客户端 private static Map<String, Socket> clientMap = new ConcurrentHashMap<String,Socket>(); //具体的处理每个客户端的请求 private static class ExcuteClient implements Runnable{ private Socket client; public ExcuteClient(Socket client) { this.client = client; } @Override public void run() { try { //获取客户端的输出流,读取客户端消息,并处理 Scanner in = new Scanner(client.getInputStream()); String strFromClient; while(true){ if(in.hasNextLine()){ strFromClient = in.nextLine(); //在Windows下默认换行是:\r\n,所以把\r要转换为空字符串 Pattern pattern = Pattern.compile("\r"); Matcher matcher = pattern.matcher(strFromClient); strFromClient = matcher.replaceAll(""); //注册流程 if(strFromClient.startsWith("useName")){ String useName = strFromClient.split("\\:")[1]; registerUser(useName,client); continue; } //群聊功能 if(strFromClient.startsWith("G")){ String msg = strFromClient.split("\\:")[1]; groupChat(msg,client); continue; } //私聊功能 if(strFromClient.startsWith("P")){ String userName = strFromClient.split("\\:")[1].split("-")[0]; String msg = strFromClient.split("\\:")[1].split("-")[1]; privateChat(userName,msg,client); continue; } //用户退出 if(strFromClient.startsWith("B")){ String userName = null; //根据Socket找到UserName for(String keyName : clientMap.keySet()){ if(clientMap.get(keyName).equals(client)){ userName = keyName; } } System.out.println("用户" + userName + "下线了。。。"); clientMap.remove(userName); System.out.println("当前共有用户" + clientMap.size() + "人"); continue; } else{ PrintStream out = new PrintStream(client.getOutputStream(),true,"UTF-8"); out.println("输入错误。。。"); } } } } catch (IOException e) { e.printStackTrace(); } } private void registerUser(String name,Socket client){ System.out.println("用户:" + name + "已上线!"); clientMap.put(name,client); System.out.println("当前在线人数:" + clientMap.size() + "人!"); //既然是用户在注册,所以这里服务器通知用户注册结果 try { PrintStream out = new PrintStream(client.getOutputStream(),true,"UTF-8"); out.println("用户注册成功!"); } catch (IOException e) { e.printStackTrace(); } } private void groupChat(String msg,Socket client){ //取出clientMap中所有的Entry对象,遍历每个用户,并且发送消息 Set<Map.Entry<String,Socket>> clientSet = clientMap.entrySet(); for(Map.Entry<String,Socket> entry:clientSet){ try { Socket socket = entry.getValue(); //取得输出流,向客户端发送消息 PrintStream out = new PrintStream(socket.getOutputStream(),true,"UTF-8"); out.println("由端口号为"+ client.getPort() + "发来的群聊消息:" + msg); } catch (IOException e) { e.printStackTrace(); } } } private void privateChat(String userName,String msg,Socket client){ Socket privateSocket = clientMap.get(userName); try { PrintStream out = new PrintStream(privateSocket.getOutputStream(),true,"UTF-8"); out.println("由端口号为:" + client.getPort() + "发来的消息:" + msg); } catch (IOException e) { e.printStackTrace(); } } } public static void main(String[] args)throws Exception{ //为了提高效率,这里使用多线程进行处理 ExecutorService executorService = Executors.newFixedThreadPool(30); //实例化ServerSocket对象,并指定IP为本地主机,端口号为6666 ServerSocket serverSocket = new ServerSocket(6666); for(int i = 0; i < 30;i++){ System.out.println("等待用户连接。。。"); //等待客户端连接服务器 Socket client = serverSocket.accept(); System.out.println("有客户端连接,端口号为:" + client.getPort()); //启动线程,并处理客户端请求 executorService.submit(new ExcuteClient(client)); } //关闭线程,关闭服务器 executorService.shutdown(); serverSocket.close(); } }
客户端:
import java.io.IOException; import java.io.PrintStream; import java.net.Socket; import java.util.Scanner; /** * 接收服务端发来的消息 */ class FromServer implements Runnable{ Socket client; public FromServer(Socket client){ this.client = client; } @Override public void run() { try { Scanner in = new Scanner(client.getInputStream()); while (true) { if (in.hasNextLine()) { System.out.println("服务器:" + in.nextLine()); } //判断客户端是否退出,如果推出,跳出循环,并关闭流 if (client.isClosed()) { System.out.println("客户端关闭。。。"); break; } } in.close(); } catch (IOException e) { e.printStackTrace(); } } } /** * 向服务端发出消息 */ class ToServer implements Runnable{ Socket client; public ToServer(Socket client){ this.client = client; } @Override public void run() { try { Scanner scanner = new Scanner(System.in); PrintStream out = new PrintStream(client.getOutputStream(),true,"UTF-8"); while (true) { System.out.println("请输入信息:"); String strToserver; if(scanner.hasNextLine()){ strToserver = scanner.nextLine().trim(); out.println(strToserver); //客户端退出标志:B if(strToserver.startsWith("B")){ System.out.println("客户端退出。。。"); scanner.close(); out.close(); client.close(); break; } } } } catch (IOException e) { e.printStackTrace(); } } } public class ManyThreadClient { public static void main(String[] args){ try { //实例化Socket对象,与服务器建立连接 Socket client = new Socket("127.0.0.1",6666); //为了发送消息和接收消息可以同时进行,使用多线程进行处理 Thread thread1 = new Thread(new FromServer(client)); Thread thread2 = new Thread(new ToServer(client)); thread1.start(); thread2.start(); } catch (IOException e) { e.printStackTrace(); } } }
위 내용은 소규모 LAN 채팅 애플리케이션을 구현하는 Java 프로그래밍 단계의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

핫 AI 도구

Undress AI Tool
무료로 이미지를 벗다

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Clothoff.io
AI 옷 제거제

Video Face Swap
완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

ArgParse 모듈을 사용하는 경우 필요한 = true를 설정하여 제공 해야하는 매개 변수를 달성 할 수 있습니다. 1. reque yewing = true 옵션 매개 변수 (예 : -입력)를 설정하려면 true가 필요합니다. 스크립트를 실행할 때 제공되지 않으면 오류 가보고됩니다. 2. 위치 매개 변수는 기본적으로 필요하며, 필요한 = true를 설정할 필요가 없습니다. 3. 필요한 매개 변수에 위치 매개 변수를 사용하는 것이 좋습니다. 때때로, 필요한 = true의 선택적 매개 변수는 유연성을 유지하는 데 사용됩니다. 4. 필수 = true는 매개 변수를 제어하는 가장 직접적인 방법입니다. 사용 후 사용자는 스크립트를 호출 할 때 해당 매개 변수를 제공해야합니다. 그렇지 않으면 프로그램에 오류가 발생하고 종료됩니다.

CommentSinjavaIgNEDBYTHECOMPILERDOUDSTROXPLANATION, NOTES, ORDISABLINGCODE.THEREARETHREETYPES : 1) 단일 LINECOMMENTSSTARTWITH // andlastuntilTheEndoftheline; 2) Multi-Linecommentsbeginwith/andcanspanmultiplelines; 3) 문서 작성

javahttpclientapi를 사용하는 핵심은 httpclient를 만들고, httprequest를 구축하고, httpresponse를 처리하는 것입니다. 1. httpclient.newhttpclient () 또는 httpclient.newbuilder ()를 사용하여 시간 초과, 프록시 등을 구성하여 클라이언트를 생성합니다. 2. httprequest.newbuilder ()를 사용하여 URI, Method, Header 및 Body를 설정하여 요청을 구축하십시오. 3. client.send ()를 통해 동기 요청을 보내거나 client.sendasync ()를 통해 비동기 요청을 보냅니다. 4. BodyHandlers를 사용하십시오

thebestjavaidein2024DependsonyOURNEEDS : 1. chooseIntellijideaforProfectional, Enterprise, Orfull-StackdevelopmentDuetoitsSuperiorCodeIntelligence, Frameworkintegration 및 andtoompsepsececececepomececupongati

RETARTYOURROUTERANDOCPUTERESOLVETEMPORARYGLITCES.2.RUNTHENTERWORKTROUBLESHOTERVIATHESTEMTRAYTOAUAUTOMALICALLYFIXCOMMONISS.3.RENEWTHEIPADDRESSUSINGCOMMANDPROMPROMPTASADMINSTRATORBYRUNNINGIPCONFIG/RELEART, IPCONFIG/RENEET, NETSHWINSOCKRETSHONT, 및 rENTOKRETSHONTONT, and NETSHWINSOCKREST

.quals ()를 사용하여 문자열 내용을 비교하십시오. == 실제 문자보다는 객체 참조 만 비교하므로; 2. 사례를 무시할 때 .equalsignorecase ()를 사용하십시오. 3. 알파벳순으로 정렬 할 때 .compareto ()를 사용하고, .comparetoignorecase ()를 무시할 때; 4. 무일한 줄을 부르지 마십시오. equals ()는 "literal".equals (변수) 또는 Objects.equals (str1, str2)를 사용하여 널 값을 안전하게 처리하는 데 사용해야합니다. 요컨대, 항상 참조보다는 컨텐츠 비교에주의를 기울이십시오.

LinkedList는 Java의 양방향 링크 목록, 목록 및 Deque 인터페이스를 구현합니다. 요소가 자주 삽입되고 삭제되는 시나리오에 적합합니다. 특히 목록의 양쪽 끝에서 작동 할 때는 높은 효율이 있지만 랜덤 액세스 성능은 좋지 않으며 시간 복잡성은 O (N)입니다. 삽입 및 삭제는 알려진 위치에서 O (1)에 도달 할 수 있습니다. 따라서 구조물을 동적으로 수정 해야하는 스택, 큐 또는 상황을 구현하는 데 적합하며 인덱스별로 자주 액세스하는 읽기 집약적 인 작업에는 적합하지 않습니다. 마지막 결론은 LinkedList가 자주 수정되었지만 액세스가 적을 때 ArrayList보다 낫다는 것입니다.

cheecksearchsettingslike "matchEntirecellContents"및 "matchcase"exexpandingoptionsinfindandreplace, "tocorrectscope 내에서"lookin "issettovaluesand"를 보장합니다
