先编写最简答的服务器

思路

1编写一个实现Runnable接口的静态内部类 ServerC,便于区分每个客户端
1.1 获取客户端数据函数 public String remsg()
1.2 转发消息给所有客户端(自己除外) public void allsend(String data)
1.3 释放资源 public void release()
1.4编写接口重写方法 public void run(),调用收消息和发消息方法。 2server主函数监听客户端的请求
每次的客户端请求都将对应的 ServerC 保存到一个集合当中
最后开启一个线程单独对应
package chat;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.CopyOnWriteArrayList; public class ServerChat {
private static CopyOnWriteArrayList<ServerC> all = new CopyOnWriteArrayList<>(); public static void main(String[] args) throws IOException {
System.out.println("##################Server##############");
ServerSocket server = new ServerSocket(8888);
while (true){
Socket socket = server.accept();
System.out.println("进入");
ServerC serverC = new ServerC(socket);
all.add(serverC);
//使用多线程分离各个客户端
new Thread(serverC).start();
}
} static class ServerC implements Runnable{
private Socket socket;
private DataOutputStream dos;
private DataInputStream dis; public ServerC(Socket socket) {
this.socket = socket;
try {
dis = new DataInputStream(socket.getInputStream());
dos = new DataOutputStream(socket.getOutputStream());
} catch (IOException e) {
e.printStackTrace();
release();
}
} /**
* 获取客户端data
* @return
*/
public String remsg(){
//接受消息
String data = null;
try {
data = dis.readUTF();
} catch (IOException e) {
e.printStackTrace();
}
return data; }
//消息转发给所有人函数
public void transaction(String data){
try{
dos.writeUTF(data);
dos.flush();
}catch (Exception e){
release();
}
} //群发给客户端
public void allsend(String data) {
for (ServerC serverC : all) {
if (this == serverC) {
continue;
} else {
serverC.transaction(data);
} }
} //释放资源函数
public void release(){
try {
dos.close();
dis.close();
socket.close();
} catch (IOException e) {
e.printStackTrace();
} } @Override
public void run() {
String data = remsg();
allsend(data);
}
} }

编写客户端的收消息类:

package chat;

import java.io.DataInputStream;
import java.io.IOException;
import java.net.Socket; public class ClientResevice implements Runnable{
private Socket socket;
private DataInputStream dis; public ClientResevice(Socket socket) {
this.socket = socket;
try {
this.dis = new DataInputStream(socket.getInputStream());
} catch (IOException e) {
e.printStackTrace();
release();
}
} @Override
public void run() {
while (true){
//接受消息
String msg = receive();
if(!msg.isEmpty()){
System.out.println(msg);
}
}
} /**
* 接受消息函数
* @return返回消息
*/
public String receive(){
String msg = null;
try {
msg = dis.readUTF();
} catch (IOException e) {
e.printStackTrace();
release();
}
return msg;
} //释放资源函数
public void release(){
try {
dis.close();
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}

客户端发消息类:

package chat;

import java.io.BufferedReader;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.Socket; /**
* 创建发送消息线程类,发送和接受分开
*/
public class ClientSend implements Runnable{
private BufferedReader console;
private DataOutputStream dos;
private Socket socket; /**
* 建立发送消息线程类的构造函数
* @param socket
*/
public ClientSend( Socket socket){
this.socket = socket;
this.console = new BufferedReader(new InputStreamReader(System.in));
try {
this.dos = new DataOutputStream(socket.getOutputStream());
} catch (IOException e) {
e.printStackTrace();
this.release();
}
} /**
* 实现Runnable里面的run方法
*/
@Override
public void run() {
while (true){
send();
}
} /**
*获取命令函输入函数
* @return
*/
public String conin(){
String msg = null;
try {
msg = console.readLine();
} catch (IOException e) {
e.printStackTrace();
}
return msg;
} /**
* 发送函数
*/
public void send(){
String msg = conin();
if(!msg.isEmpty()){
try {
dos.writeUTF(msg);
dos.flush();
} catch (IOException e) {
e.printStackTrace();
}
}
} //释放资源函数
public void release(){
try {
dos.close();
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}

客户端代码,主要怎么使用发收消息类:

package chat;

import java.io.*;
import java.net.Socket; public class FirstClient { public static void main(String[] args) throws IOException {
System.out.println("##############client################");
Socket socket = new Socket("127.0.0.1",8888);
// 启动发送消息线程
new Thread(new ClientSend(socket)).start();
// 启动接受线程
new Thread(new ClientResevice(socket)).start();
}
}

java-网络通信--socket实现多人聊天(基于命令行)的更多相关文章

  1. java swing+socket实现多人聊天程序

    swing+socket实现多人聊天程序 1.准备工作 先看效果: 客户端项目结构图: 服务端项目结构图: 2.运行原理 服务端 先开一个线程serverListerner,线程中开启一个Server ...

  2. Java使用socket实现两人聊天对话

    import java.io.*; import java.net.ServerSocket; import java.net.Socket; import java.util.Scanner; /* ...

  3. 如何创建一个基于命令行工具的跨平台的 NuGet 工具包

    命令行可是跨进程通信的一种非常方便的手段呢,只需启动一个进程传入一些参数即可完成一些很复杂的任务.NuGet 为我们提供了一种自动导入 .props 和 .targets 的方法,同时还是一个 .NE ...

  4. Linux下基于命令行的抓包方法

    大家可能都已经对著名的抓包工具Ethereal比较熟悉了,这里再介绍一种基于命令行的抓包工具tcpdump. 举例:抓本机1813端口上的数据,并将抓包结果保存在test.cap文件中 然后在本地可以 ...

  5. POP3:基于命令行的电子邮件(EMail)在线查看和批量下载工具

    使用该工具可以在不安装outlook和foxmail等邮件客户端的情况下快速下载指定邮箱的邮件,并将下载的邮件以eml格式进行保存. 附: 查看eml格式的邮件可使用 EmlReader 工具,该工具 ...

  6. java socket之多人聊天室Demo

    一.功能介绍 该功能实现了一个类似QQ的最简单多人聊天室,如下图所示. 二.目录结构 三.服务端 1)SocketServer类,该类是服务端的主类,主要负责创建聊天窗口,创建监听客户端的线程: pa ...

  7. 基于Socket实现多人聊天室

    当前支持: 1.仅文字 2.加入聊天室提醒 3.退出聊天室提醒 可能出现的BUG: 1.可能出现客户端发送信息后不能及时推送,需要下一个客户端发送信息后一起推送 服务端代码: 1 package co ...

  8. 多线程+socket实现多人聊天室

    最近在学习多线程的时候打算做一个简单的多线程socke聊天的程序,结果发现网上的代码都没有完整的实现功能,所以自己实现了一个demo: demo功能大致就是,有一个服务端负责信息转发,多个客户端发送消 ...

  9. Java 多线程Socket编程通讯--实现聊天室代码

    1.创建服务器类 import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import ja ...

随机推荐

  1. C++ 派生类函数重载与虚函数继承详解

    目录 一.作用域与名字查找 1.作用域的嵌套 2.在编译时进行名字查找 3.名字冲突与继承 4.通过作用域运算符来使用隐藏的成员 二.同名函数隐藏与虚函数覆盖 1.几种必须区分的情况 2.一个更复杂的 ...

  2. 为Visual Studio 2019设置 更改皮肤

    下载主题插件:Color Themes for Visual Studio 安装插件 下载完成后 关闭vs2019 完成初始化,初始化完成后,再次打开软件进行配置. 卸载插件 点击卸载,然后关闭vs2 ...

  3. linux – 使用find -name命令的通配符和双引号

    转至:https://www.icode9.com/content-3-209958.html 双引号(和单引号)是否抑制了通配符扩展?如果是这样,那么为什么以下呢? find -name " ...

  4. linux下通过脚本方式对中间件weblogic进行补丁升级

    转至:http://bbs.learnfuture.com/topic/48 linux下通过脚本方式对中间件weblogic进行补丁升级 刘五奎 [摘要] 在运维行业,系统或软件漏洞每每牵动着每一个 ...

  5. Linux下搭建iSCSI共享存储

    转至:https://www.linuxidc.com/Linux/2016-09/135655.htm Linux下搭建iSCSI共享存储 拓扑: 实验步骤: ------------------- ...

  6. 60天shell脚本计划-6/12-渐入佳境

    --作者:飞翔的小胖猪 --创建时间:2021年2月21日 --修改时间:2021年2月25日 说明 每日上传更新一个shell脚本,周期为60天.如有需求的读者可根据自己实际情况选用合适的脚本,也可 ...

  7. Pandas:to_excel时如何不覆盖之前的Excel表、ExcelWriter类

    如果只是想把一个DataFrame保存为单独的一个Excel文件,那么直接写: data.to_excel('xxx.excel','sheet1',index=False) 但是这样做,只会保存为单 ...

  8. 进程&线程(三):外部子进程subprocess、异步IO、协程、分布式进程

    1.外部子进程subprocess python之subprocess模块详解--小白博客 - 夜风2019 - 博客园 python subprocess模块 - lincappu - 博客园 之前 ...

  9. Python:爬取一个可下载的PDF链接并保存为本地pdf文件

    问题:网页http://gk.chengdu.gov.cn/govInfo/detail.action?id=2653973&tn=2中有一个PDF需要下载,开发者模式下该PDF的链接为htt ...

  10. petite-vue源码剖析-属性绑定`v-bind`的工作原理

    关于指令(directive) 属性绑定.事件绑定和v-modal底层都是通过指令(directive)实现的,那么什么是指令呢?我们一起看看Directive的定义吧. //文件 ./src/dir ...