很久以前的一个Demo,这里服务器只做转发功能,根据ID地址和端口号来标识身份,群聊和私聊只是简单实现,

服务器代码如下:

import java.util.*;
import java.io.*;
import java.net.*; public class Server extends Thread
{
private static Vector<Socket> vecClient = new Vector<Socket>();
private PrintWriter out;
private BufferedReader in;
private Socket clientsocket; public static void main(String[] args) throws IOException {
ServerSocket server = new ServerSocket(2014); // 创建
System.out.println("启动服务器! ");
Socket sock; while (true) {
sock = server.accept(); // 等待客户请求
Server ser = new Server(sock);
ser.start(); // 启动
}
} public Server(Socket socket) {
this.clientsocket = socket;
vecClient.addElement(socket); // 增加客户线程到数组
try {
// 获取服务端传来的字节流好,存到缓冲区
in = new BufferedReader(new InputStreamReader(
socket.getInputStream()));
// printwriter,文本输出流打印对象的格式化表示形式,只有调用println等,自动刷新即true,强行把缓冲区的数据输出。
out = new PrintWriter(new OutputStreamWriter(
socket.getOutputStream()), true);
// input、OutputStreamWriter 是字符流通向字节流的桥梁
} catch (IOException e) {
}
} public void run()
{
try
{
Newclient(); // 向新客户发出欢迎信息,通知所有在线客户有新客户连接了
while(true) // 处理与客户的交流
{
String msg=in.readLine(); //数据读取 if(msg.equals("bye")|| msg.equals("拜拜"))//当输入拜拜或bye时候退出
{
Xiaxian();
in.close(); // 关闭输入流
clientsocket.close(); // 关闭socket
break;
}
else if(msg.indexOf("@")==0&&msg.indexOf(" ")>0){
int end=msg.indexOf(" ");
String findAddr=msg.substring(1,end);
if(vecClient.isEmpty() == false) //判断为空
{
for(int i=0;i<vecClient.size();i++)
{
Socket socket = vecClient.get(i);
String addr = socket.getInetAddress().toString().replace("/", "") + ":"
+ socket.getPort();
if(findAddr.equals(addr)){
String addr2 = clientsocket.getInetAddress().toString().replace("/", "") + ":"
+ clientsocket.getPort();
//elementAt返回指定索引处的组件,而get返回列表中指定位置处的元素
PrintWriter pw = new PrintWriter(new OutputStreamWriter(vecClient.get(i).getOutputStream()),true);
pw.println("大神"+addr2+" 对你 说: "+msg.substring(end));
}
}
}
}
// 否则将接收到的信息向所有在线客户发出去
else
{
if(vecClient.isEmpty() == false)
for(int i=0;i<vecClient.size();i++)
{
//elementAt返回指定索引处的组件,而get返回列表中指定位置处的元素
PrintWriter pw = new PrintWriter(new OutputStreamWriter(vecClient.get(i).getOutputStream()),true);
pw.println("大神"+clientsocket.getInetAddress().toString()+":"+clientsocket.getPort()+" 说: "+msg);
}
}
}
}
catch(IOException e)
{ }
} public void Newclient() throws IOException // 新来的,用此函数
{
String addr = clientsocket.getInetAddress().toString() + ":"
+ clientsocket.getPort();
out.println("欢迎:" + addr+"加入! ");
try {
if (vecClient.isEmpty() == false)
for (int i = 0; i < vecClient.size(); i++)
{
Socket socket = vecClient.get(i);
PrintWriter pw = new PrintWriter(new OutputStreamWriter(
socket.getOutputStream()), true);
String addr2 = socket.getInetAddress().toString() + ":"
+ socket.getPort();
if (addr.equals(addr2)) {
pw.println("欢迎加入聊天,私聊请用@ip:port 内容");
} else {
pw.println("新成员:"
+ clientsocket.getInetAddress().toString()
+ ":" + clientsocket.getPort() + " 嗨起来吧!");
}
}
} catch (IOException e) {
}
} public void Xiaxian() throws IOException // 下线方法
{
out.println("再见,连接关闭!");
if (vecClient.isEmpty() == false)
for (int i = 0; i < vecClient.size(); i++) {
PrintWriter pw = new PrintWriter(new OutputStreamWriter(vecClient
.get(i).getOutputStream()), true);
pw.println("--" + clientsocket.getInetAddress().toString()
+ ":" + clientsocket.getPort() + " 已经下线");
}
vecClient.remove(clientsocket);
}
}

客服端,可在本地命令运行多个测试,私聊格式:@ip:port(这里必须有空格)内容,原理如下:

import java.net.*;
import java.io.*;
import java.util.*; class Client
{
public static void main(String[] args)
{
try
{
Socket socket=new Socket("localhost",2014);
ClientSend send=new ClientSend(socket); // 创建发送线程
ClientReceive receive=new ClientReceive(socket); // 创建接收线程
send.start(); // 启动发送线程
receive.start(); // 启动接收线程
}
catch(Exception e)
{
System.out.println("服务器没有开启呢!");
e.printStackTrace();
} }
} class ClientSend extends Thread
{
private Socket socket;
private PrintWriter out; public ClientSend(Socket socket)
{
this.socket=socket;
try
{
out=new PrintWriter(new OutputStreamWriter(socket.getOutputStream()),true);
//返回一个服务器与客户端的输出流,true强行把缓冲区的数据输出
}
catch(IOException e)
{
e.printStackTrace();
}
} public void run() // 发送信息到服务器
{
String msg;
Scanner input=new Scanner(System.in);//输入
msg = input.nextLine(); //nextLine方法返回的是回车之前的所有字符
while(true)
{
//用equalsIgnoreCase可忽略大小写
if(msg.equals("bye") || msg.equals("拜拜"))
{
out.println(msg);
break;
}
out.println(msg);
msg = input.nextLine();
}
}
} class ClientReceive extends Thread //接收
{
private Socket socket;
private BufferedReader in; public ClientReceive(Socket socket)
{
this.socket= socket;
try
{
in=new BufferedReader(new InputStreamReader(socket.getInputStream()));
}
catch(IOException e)
{
e.printStackTrace();
}
} public void run() // 接收服务器发来的信息
{
try
{
String msg=in.readLine();
while(msg.equals("bye") == false)
{
System.out.println(msg);
msg=in.readLine();
}
in.close();
socket.close();
}
catch(IOException e)
{
e.printStackTrace();
}
}
}

Socket通信 简单实现私聊、群聊(dos命令下)的更多相关文章

  1. 简单通讯聊天 群聊功能 Windows下的客户端 Linux下的epoll服务器

    1 服务器代码  Linux eclipse C++ //======================================================================= ...

  2. 基于ejabberd简单实现xmpp群聊离线消息

    首先,xmpp服务器是基于ejabberd.离线消息模块是mod_interact,原地址地址:https://github.com/adamvduke/mod_interact: 修改后实现群聊离线 ...

  3. Java在DOS命令下的运行及其API文档制作过程

    该文档主要描述java程序在DOS命令下的运行,以及一些常用的命令 常用DOS命令: d: 回车 盘符切换 dir(directory):列出当前目录下的文件以及文件夹 md (make direct ...

  4. DOS命令下输入:java Hello 出现以下几种结果可能的原因:

    DOS命令下输入:java Hello 出现以下结果:Bad command or the file name 没有这个命令或文件名 原因可能是没有成功安装jdk或者没有配置好jdk 的环境变量,或者 ...

  5. 安装Oracle之后在DOS命令下不能使用sqlplus命令启动Oracle?

    就像完成安装JDK后需要给系统配置环境变量JAVA_HOME和PATH,以便让系统认识在JDK里面的Javac命令.同样的,安装Oracle也需要这么一个过程.在正常情况下安装Oracle时系统安装程 ...

  6. Dos命令下目录操作

    Dos命令下目录操作 1.cd 操作 显示当前目录名或改变当前目录 cd [盘符][路径]                      进入指定盘符下的目录 cd [..]               ...

  7. dos命令下安装pip报错 不是内部命令

    在dos命令下: pip install requests 遇到这种情况一般是Python的环境变量没有设置好 解决方案一:设置环境变量 C:\Python\scripts   如图 是否有pytho ...

  8. 第五讲 smart qq poll包处理 以及 私聊 群聊消息收发

    发送 poll包 public static void Login_PostPoll() { try { string url = "http://d1.web2.qq.com/channe ...

  9. Java的Socket通信简单实例

    服务端 package testlxd; import java.io.BufferedReader; import java.io.IOException; import java.io.Input ...

随机推荐

  1. CSS2样式中选择器的介绍

    这里主要是对css2中的选择器进行了一下总结!

  2. django之分页、cookie装饰器

    一.分页代码如下 from django.utils.safestring import mark_safe class Page: def __init__(self, current_page, ...

  3. 读《编写可维护的JavaScript》第二三章总结

    第二章 注释 添加注释的一般原则是,在需要让代码变得清晰时添加注释. 2.1 ① 单行注释 独占一行的注释,用来解释下一行代码.这行注释之前总是有一个空行,且缩进层级和下一行代码保持一致. 在代码行的 ...

  4. JavaScript中的各种宽高属性

    转自慕课网:http://www.imooc.com/article/14516   在js中,存在着N多的关于高度和宽度的属性,比如:clientHeight.offsetHeight.scroll ...

  5. 译\Node.js应用的持续部署

    Node.js应用的持续部署 翻译前 翻译自:https://blog.risingstack.com/continuous-deployment-of-node-js-applications/ 正 ...

  6. img的空白符

    在div里面嵌套了一个img出来问题,div的高度超过了img的高度,大约3px ,查看了样式没什么问题,后来百度了一下,找到了2种的方法都有效: 第一种:img{display:block} 第二种 ...

  7. Android自定义View的三种实现方式

    在毕设项目中多处用到自定义控件,一直打算总结一下自定义控件的实现方式,今天就来总结一下吧.在此之前学习了郭霖大神博客上面关于自定义View的几篇博文,感觉受益良多,本文中就参考了其中的一些内容. 总结 ...

  8. [Java] SoapUI使用Java获取各时间日期方法

    import java.util.*; import java.text.SimpleDateFormat; // current date String dateNew = today() // t ...

  9. 转!!负载均衡器技术Nginx和F5的优缺点对比

    对于数据流量过大的网络中,往往单一设备无法承担,需要多台设备进行数据分流,而负载均衡器就是用来将数据分流到多台设备的一个转发器. 目前有许多不同的负载均衡技术用以满足不同的应用需求,如软/硬件负载均衡 ...

  10. PPTP协议

    PPTP协议 PPTP(Point-to-Point Tunneling Protocol)点对点隧道协议是PPP协议的一种扩展,它将PPP帧封装进IP包中,通过IP网络进行传输.它通过PPTP控制连 ...