Java 控制台程序实现类似广播功能

服务器端代码

添加 maven 依赖

<dependency>
<groupId>javax.websocket</groupId>
<artifactId>javax.websocket-api</artifactId>
<version>1.1</version>
<scope>provided</scope>
</dependency>

服务器端代码

package com.seliote.web.http;

import javax.websocket.*;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List; /**
* 每次有 WebSocket 连接请求都会创建一个该类的实例
*/
@ServerEndpoint(value = "/broadcast")
public class SocketServer {
private static final List<Session> onlinePeople = new ArrayList<>(); @OnOpen
public void onOpen(Session aSession) {
System.out.println(System.currentTimeMillis() + ": OnOpen:::" + onlinePeople.size() + 1);
if (!onlinePeople.contains(aSession)) {
onlinePeople.add(aSession);
}
} // 该方法是用于被动接收信息的
@OnMessage
public void onMessage(Session aSession, String aS) throws IOException {
System.out.println(System.currentTimeMillis() + ": OnMessage:::" + aS);
for (Session session : onlinePeople) {
session.getBasicRemote().sendText(aS);
}
} // OnMessage 可以有多个不同签名的
@OnMessage
public void onMessage(Session aSession, InputStream aInputStream) {
System.out.println(System.currentTimeMillis() + ": OnMessage");
// TODO
} /**
* 每次有客户端异常关闭该方法也会调用
* @param aSession
* @param aCloseReason
*/
@OnClose
public void onClose(Session aSession, CloseReason aCloseReason) {
System.out.println(System.currentTimeMillis() + ": OnClose:::" + aCloseReason.getReasonPhrase());
if (onlinePeople.contains(aSession)) {
onlinePeople.remove(aSession);
}
} @OnError
public void onError(Session aSession, Throwable aThrowable) {
System.out.println(System.currentTimeMillis() + ": OnError");
aThrowable.printStackTrace();
}
}

如果连接时需要携带客户端信息,那么可以在路径中加入参数,如客户端路径加入用户 Token 变为 127.0.0.1/broadcast/123456,服务器端的标注就可改为 @ServerEndpoint(value = "/broadcast/{token}") ,之后的 @OnOpen 方法中就可以有一个 @PathParam("token") String aToken 代表客户端传入的 Token

客户端代码

添加 maven 依赖,注意这里使用的是 tyrus-standalone-client 而非 javax.websocket-client-api 后者会报错

<dependency>
<groupId>org.glassfish.tyrus.bundles</groupId>
<artifactId>tyrus-standalone-client</artifactId>
<version>1.3.3</version>
<scope>compile</scope>
</dependency>

客户端代码

package com.seliote;

import javax.websocket.*;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Scanner; @ClientEndpoint
public class Demo {
private static Session sSession; public static void main(String... args) throws URISyntaxException, DeploymentException, IOException {
// https 协议对应使用 wss
URI uri = new URI("ws", "127.0.0.1:8080", "/broadcast", null, null);
// 通过 ContainerProvider 的 static 方法 getWebSocketContainer() 获得 WebSocketContainer
sSession = ContainerProvider.getWebSocketContainer().connectToServer(Demo.class, uri);
try (Scanner scanner = new Scanner(System.in)) {
String broadcastMsg = "";
while (true) {
broadcastMsg = scanner.nextLine();
// 通过 Session 对象主动发送信息
sSession.getBasicRemote().sendText(broadcastMsg);
//sSession.getBasicRemote().getSendStream().write(....);
}
}
} @OnOpen
public void onOpen() {
System.out.println(System.currentTimeMillis() + ": OnOpen ");
} // 该方法是用于被动接收信息的
@OnMessage
public void onMessage(String aS) {
System.out.println(System.currentTimeMillis() + ": OnMessage::: " + aS);
}
}

在一个客户端输入信息后服务器会及时收到信息并广播给所有在线的客户端

------------------------------------------2019.01.09 更新

如果需要支持相应的实体类型,WebSocket 服务器端大概长 这样,而客户端配置如下

Maven 依赖(这里用了 JSONObject 而不是服务器端的 Jackson)

<dependency>
<groupId>org.glassfish.tyrus.bundles</groupId>
<artifactId>tyrus-standalone-client</artifactId>
<version>1.3.3</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>org.json</groupId>
<artifactId>json</artifactId>
<version>20180813</version>
<scope>compile</scope>
</dependency>
package com.seliote.demo;

/**
* @author seliote
* @date 2019-01-09
* @description WebSocket 信息实体
*/
@SuppressWarnings({"unused", "WeakerAccess"})
public class BroadcastMsg {
private String mSessionId;
private String mTimestamp;
private String mMsg; public BroadcastMsg() {} public BroadcastMsg(String aSessionId, String aTimestamp, String aMsg) {
mSessionId = aSessionId;
mTimestamp = aTimestamp;
mMsg = aMsg;
} public String getSessionId() {
return mSessionId;
} public void setSessionId(String aSessionId) {
mSessionId = aSessionId;
} public String getTimestamp() {
return mTimestamp;
} public void setTimestamp(String aTimestamp) {
mTimestamp = aTimestamp;
} public String getMsg() {
return mMsg;
} public void setMsg(String aMsg) {
mMsg = aMsg;
} @Override
public String toString() {
return mSessionId + " - " + mTimestamp + " - " + mMsg;
}
}
package com.seliote.demo;

import org.json.JSONObject;

import javax.websocket.Decoder;
import javax.websocket.Encoder;
import javax.websocket.EndpointConfig;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets; /**
* @author seliote
* @date 2019-01-09
* @description BroadcastMsg 用于 WebSocket 的编码与解码器
*/
public class BroadcastMsgCoder implements Encoder.BinaryStream<BroadcastMsg>, Decoder.BinaryStream<BroadcastMsg> { @Override
public void init(EndpointConfig aEndpointConfig) { } @Override
public void destroy() { } @Override
public void encode(BroadcastMsg aBroadcastMsg, OutputStream aOutputStream) throws IOException {
aOutputStream.write(new JSONObject(aBroadcastMsg).toString().getBytes(StandardCharsets.UTF_8));
} @Override
public BroadcastMsg decode(InputStream aInputStream) throws IOException {
byte[] buffer = new byte[1024];
int length;
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
while ((length = aInputStream.read(buffer)) != -1) {
byteArrayOutputStream.write(buffer, 0, length);
}
String json = new String(byteArrayOutputStream.toByteArray(), StandardCharsets.UTF_8);
JSONObject jsonObject = new JSONObject(json);
return new BroadcastMsg(
jsonObject.getString("sessionId"),
jsonObject.getString("timestamp"),
jsonObject.getString("msg")
);
}
}
package com.seliote.demo;

import javax.websocket.ClientEndpoint;
import javax.websocket.ContainerProvider;
import javax.websocket.DeploymentException;
import javax.websocket.EncodeException;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Scanner; @ClientEndpoint(
encoders = BroadcastMsgCoder.class,
decoders = BroadcastMsgCoder.class
)
public class Demo { public static void main(String... args) throws URISyntaxException, DeploymentException, IOException {
// https 协议对应使用 wss
URI uri = new URI("ws", "127.0.0.1:8080", "/time/1", null, null);
// 通过 ContainerProvider 的 static 方法 getWebSocketContainer() 获得 WebSocketContainer
Session session = ContainerProvider.getWebSocketContainer().connectToServer(Demo.class, uri);
try (Scanner scanner = new Scanner(System.in)) {
//noinspection InfiniteLoopStatement
while (true) {
// 通过 Session 对象主动发送信息
try {
String msg = scanner.nextLine();
BroadcastMsg broadcastMsg = new BroadcastMsg(
session.getId(),
System.currentTimeMillis() + "",
msg
);
session.getBasicRemote().sendObject(broadcastMsg);
} catch (EncodeException exp) {
exp.printStackTrace();
}
//sSession.getBasicRemote().getSendStream().write(....);
}
}
} @OnOpen
public void onOpen() {
System.out.println(System.currentTimeMillis() + ": OnOpen ");
} @OnMessage
public void onMessage(String aS) {
System.out.println(System.currentTimeMillis() + ": OnMessage::: " + aS);
} @OnMessage
public void onMessage(BroadcastMsg aBroadcastMsg) {
System.out.println(aBroadcastMsg);
}
}

WebSocket 的使用的更多相关文章

  1. 漫扯:从polling到Websocket

    Http被设计成了一个单向的通信的协议,即客户端发起一个request,然后服务器回应一个response.这让服务器很为恼火:我特么才是老大,我居然不能给小弟发消息... 轮询 老大发火了,小弟们自 ...

  2. 细说WebSocket - Node篇

    在上一篇提高到了 web 通信的各种方式,包括 轮询.长连接 以及各种 HTML5 中提到的手段.本文将详细描述 WebSocket协议 在 web通讯 中的实现. 一.WebSocket 协议 1. ...

  3. java使用websocket,并且获取HttpSession,源码分析

    转载请在页首注明作者与出处 http://www.cnblogs.com/zhuxiaojie/p/6238826.html 一:本文使用范围 此文不仅仅局限于spring boot,普通的sprin ...

  4. WebSocket - ( 一.概述 )

    说到 WebSocket,不得不提 HTML5,作为近年来Web技术领域最大的改进与变化,包含CSS3.离线与存储.多媒体.连接性( Connectivity )等一系列领域,而即将介绍的 WebSo ...

  5. php+websocket搭建简易聊天室实践

    1.前言 公司游戏里面有个简单的聊天室,了解了之后才知道是node+websocket做的,想想php也来做个简单的聊天室.于是搜集各种资料看文档.找实例自己也写了个简单的聊天室. http连接分为短 ...

  6. Demo源码放送:打通B/S与C/S !让HTML5 WebSocket与.NET Socket公用同一个服务端!

    随着HTML5 WebSocket技术的日益成熟与普及,我们可以借助WebSocket来更加方便地打通BS与CS -- 因为B/S中的WebSocket可以直接连接到C/S的服务端,并进行双向通信.如 ...

  7. Cowboy 开源 WebSocket 网络库

    Cowboy.WebSockets 是一个托管在 GitHub 上的基于 .NET/C# 实现的开源 WebSocket 网络库,其完整的实现了 RFC 6455 (The WebSocket Pro ...

  8. 借助Nodejs探究WebSocket

    文章导读: 一.概述-what's WebSocket? 二.运行在浏览器中的WebSocket客户端+使用ws模块搭建的简单服务器 三.Node中的WebSocket 四.socket.io 五.扩 ...

  9. 细说websocket - php篇

    下面我画了一个图演示 client 和 server 之间建立 websocket 连接时握手部分,这个部分在 node 中可以十分轻松的完成,因为 node 提供的 net 模块已经对 socket ...

  10. webSocket and LKDBHelper的使用说明

    socketket与lkdbhelper来处理数据 客户需求: 当我们有需要从自己的后台推送消息给我们的用户时,用户需要实时的接收到来自我们的推送消息.前提是没有使用第三方的推送框架,那么这个使用we ...

随机推荐

  1. 【Spring实战】—— 2 构造注入

    本文讲解了构造注入以及spring的基本使用方式,通过一个杂技演员的例子,讲述了依赖注入属性或者对象的使用方法. 如果想要使用spring来实现依赖注入,需要几个重要的步骤: 1 定义主要的类和需要分 ...

  2. 简单的PHP算法题

    简单的PHP算法题 目录 1.只根据n值打印n个0 2.根据n值打印一行 0101010101010101010101…… 3.根据n值实现1 00 111 0000 11111…… 4.根据n值实现 ...

  3. linux shell中 if else以及大于、小于、等于逻辑表达式介绍

    在linux shell编程中,大多数情况下,可以使用测试命令来对条件进行测试,这里简单的介绍下, 比如比较字符串.判断文件是否存在及是否可读等,通常用"[]"来表示条件测试. 注 ...

  4. 从零开始Vue项目实战(一)-准备篇

    从前参与过一个react项目的代码编写,大神搭建的框架,我主要负责业务逻辑代码编写,现在回想起来似乎又什么都不会,现在为了巩固前端知识,决定用Vue来做这个项目的移动端网站,我本人Vue是从零开始的, ...

  5. TSP 遗传算法

    GA——遗传算法 同模拟退火算法一样,都是现代优化算法之一.模拟退火是在一定接受程度的情况下仍然接受一个比较差的解. 遗传算法,是真真正正的和大自然的遗传进化有着非常紧密的联系的,当然遗传进化的只是在 ...

  6. pocsuite 实现一个verify检测功能

    今天在测试中发现一个命令执行漏洞,尝试用创宇的pocsuite框架实现.说实话,这玩意儿确实没有自己写POC顺手,非得就着他的标准来,就很难受,以至于耽误了很多时间在规范上.. 影响参数后直接用||连 ...

  7. [18/12/01]super 关键字和final 关键字

    一.super 关键字 1.super是直接父类对象的引用.可以通过super来访问父类中被子类覆盖的方法或属性. 使用super调用普通方法,语句没有位置限制,可以在子类中随便调用. 代码示例: c ...

  8. listBox获取项的方法

    获取所有项 ; i < LB.Items.Count;i++ )2 {3 str_arr.Add(LB.Items[i].ToString()); 4 } 获取指定项 string str=LB ...

  9. spring(三)-事务管理

    1. Spring事务管理 事务管理:本质是使用spring管理事务,完成数据库对于事务的支持. 事务:一系列对数据库操作的集合,一个操作错误,所有都必须回滚,其特点是acid. (1)事务并发存在问 ...

  10. 如何用Redlock实现分布式锁

    转载请标明出处: http://blog.csdn.net/forezp/article/details/70305336 本文出自方志朋的博客 之前写过一篇文章<如何在springcloud分 ...