本文章完全参考了JSR 356, Java API for WebSocket。
JSR 356
JSR 356是Java的WebSocket实现API,是Java EE7的一部分。JSR 356包括WebSocket客户端API和服务端API,一般包括下面组件:
介于我们一般使用Java做服务端开发,这里我们只实现服务端代码。
编程模型
JSR 356支持两种编程模型:
- 基于注解。在POJO上使用注解。
- 基于接口。实现类必须implements接口
Endpoint
。
为了方便开发,我在springboot环境中编写代码。
基于注解的方式
- 定义业务控制器。
在业务控制器类上使用@ServerEndpoint
修饰,标识这是一个websocket服务,@ServerEndpoint
的value
属性表示请求的path。
@Component
@ServerEndpoint("/app/hello")
public class AnnotationGreetingEndpoint {}
- 接收消息
在业务控制器类方法上使用@OnMessage
修饰,用来接收客户端发送的消息。
@OnMessage
public void greeting(String text) throws IOException {
System.out.println("AnnotationGreetingEndpoint<=" + text);
}
- 发送消息
发送消息需要使用WebSocket的Session
对象。我们在业务控制器属性中定义Session session
成员变量,在websocket链接建立的时候设置session值,使用@OnOpen
注解修饰处理websocket链接建立逻辑的方法。
private Session session;
@OnOpen
public void init(Session session) {
this.session = session;
}
发送一个同步消息可以使用session.getBasicRemote().sendText
方法。
@OnMessage
public void greeting(String text) throws IOException {
System.out.println("AnnotationGreetingEndpoint<=" + text);
this.session.getBasicRemote().sendText("AnnotationGreetingEndpoint=>hello," + text);
}
AnnotationGreetingEndpoint
类全部代码:
@Component
@ServerEndpoint("/app/hello")
public class AnnotationGreetingEndpoint {
private Session session;
@OnOpen
public void init(Session session) {
this.session = session;
}
@OnMessage
public void greeting(String text) throws IOException {
System.out.println("AnnotationGreetingEndpoint<=" + text);
this.session.getBasicRemote().sendText("AnnotationGreetingEndpoint=>hello," + text);
}
@OnClose
public void destroy() {
this.session = null;
}
}
- 把业务类部署到websocket的web容器中。
使用Spring的ServerEndpointExporter
自动把带有@ServerEndpoint
注解的类添加的WebSocket容器中。
@Configuration
public class WebSocketConfig {
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
}
ServerEndpointExporter
作为Spring的bean,自动注入ServerContainer
属性。
public class ServerEndpointExporter extends WebApplicationObjectSupport
implements InitializingBean, SmartInitializingSingleton {
@Nullable
private ServerContainer serverContainer;
@Override
protected void initServletContext(ServletContext servletContext) {
if (this.serverContainer == null) {
this.serverContainer =
(ServerContainer) servletContext.getAttribute("javax.websocket.server.ServerContainer");
}
}
@Override
public void afterPropertiesSet() {
Assert.state(getServerContainer() != null, "javax.websocket.server.ServerContainer not available");
}
}
并自动扫描Spring容器带有ServerEndpoint
注解的bean,添加到ServerContainer
中。
public class ServerEndpointExporter extends WebApplicationObjectSupport
implements InitializingBean, SmartInitializingSingleton {
@Override
public void afterSingletonsInstantiated() {
registerEndpoints();
}
protected void registerEndpoints() {
Set<Class<?>> endpointClasses = new LinkedHashSet<>();
ApplicationContext context = getApplicationContext();
if (context != null) {
String[] endpointBeanNames = context.getBeanNamesForAnnotation(ServerEndpoint.class);
for (String beanName : endpointBeanNames) {
endpointClasses.add(context.getType(beanName));
}
}
for (Class<?> endpointClass : endpointClasses) {
registerEndpoint(endpointClass);
}
}
private void registerEndpoint(Class<?> endpointClass) {
ServerContainer serverContainer = getServerContainer();
try {
serverContainer.addEndpoint(endpointClass);
}
catch (DeploymentException ex) {
throw new IllegalStateException("Failed to register @ServerEndpoint class: " + endpointClass, ex);
}
}
}
- 客户端调用
为了方便,我们直接打开浏览器窗口的控制台执行js客户端代码。
- 创建WebScoket对象。
ws =new WebSocket("ws://localhost:8090/app/hello");
```
控制台返回:
```shell
·> WebSocket {url: 'ws://localhost:8090/app/hello___', readyState: 0, bufferedAmount: 0, onopen: null, onerror: null, …}
2.添加消息回调方法。
// 直接把收到的消息打印在控制台
ws.onmessage=console.log
3.发送消息
ws.send("aodi");
控制台会打印onmessage回调的消息参数:
> MessageEvent {isTrusted: true, data: 'AnnotationGreetingEndpoint=>hello,aodi', origin: 'ws://localhost:8090', lastEventId: '', source: null, …}
基于接口的方式
- 定义业务控制。
业务控制器类需要实现Endpoint
接口。与基于注解的实现方式不同,在于Endpoint
接口没有OnMessage
方法,需要向Session
中添加消息处理器来处理消息。这里直接给出实现AnnotationGreetingEndpoint
相同功能的代码。
public class InterfaceGreetingEndpoint extends Endpoint {
@Override
public void onOpen(Session session, EndpointConfig config) {
session.addMessageHandler(new EchoMessageHandler(session));
}
public static class EchoMessageHandler implements MessageHandler.Whole<String> {
private final Session session;
public EchoMessageHandler(Session session) {
this.session = session;
}
@Override
public void onMessage(String message) {
System.out.println("InterfaceGreetingEndpoint<=" + message);
try {
session.getBasicRemote().sendText("InterfaceGreetingEndpoint=>hello," + message);
} catch (IOException e) {
//
}
}
}
}
- 把业务类部署到websocket的web容器中。
通过ServerEndpointConfig
把业务类控制器注入到Spring容器中。
@Configuration
public class WebSocketConfig {
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
@Bean
public ServerEndpointConfig serverEndpointConfig() {
return ServerEndpointConfig.Builder.create(InterfaceGreetingEndpoint.class, "/app/hello_").build();
}
}
搭配ServerEndpointExporter
添加到websocket容器中。
public class ServerEndpointExporter extends WebApplicationObjectSupport
implements InitializingBean, SmartInitializingSingleton {
@Override
public void afterSingletonsInstantiated() {
registerEndpoints();
}
/**
* Actually register the endpoints. Called by {@link #afterSingletonsInstantiated()}.
*/
protected void registerEndpoints() {
ApplicationContext context = getApplicationContext();
if (context != null) {
Map<String, ServerEndpointConfig> endpointConfigMap = context.getBeansOfType(ServerEndpointConfig.class);
for (ServerEndpointConfig endpointConfig : endpointConfigMap.values()) {
registerEndpoint(endpointConfig);
}
}
}
private void registerEndpoint(ServerEndpointConfig endpointConfig) {
ServerContainer serverContainer = getServerContainer();
try {
serverContainer.addEndpoint(endpointConfig);
}
catch (DeploymentException ex) {
throw new IllegalStateException("Failed to register ServerEndpointConfig: " + endpointConfig, ex);
}
}
}
- 客户端调用。