使用websocket出现的坑
The remote endpoint was in state [TEXT_FULL_WRITING] which is an invalid state for called method
websocket推送数据的方法有:
session.getBasicRemote().sendText(message); //同步发送
session.getAsyncRemote().sendText(message); //异步发送
经过测试,在高并发的情况下,两种发送方法都会抛出上面的异常。查了很多博客,发现是多个线程同时使用同一session发送的原因。决定做如下修改。
synchronized(session){
session.getAsyncRemote().sendText(message);
}
经测试异步发送还是会抛出上述异常,同步不会出现。猜想:异步应该是new一个线程去发送,即使使用synchronized同样会出现两个session同时被不同的线程操作的时机。
决定使用:
synchronized(session){
session.getBasicRemote().sendText(message);
}
//在网络非常好的情况下。多线程同一个session发送数据
150ms/条的情况下没有出现上述异常
建议:上述发送方式虽然不会抛出上述异常,建议在并发量非常高的情况下,尽量单个sesson创建线程去发送。因为在循环session群发的时候,一个session回话网络不好,会出现超时异常,当前线程会因此中断。导致后面的session没有进行发送操作。使用单个线程,单个session情况下避免session之间的相互影响。具体超时异常如下:
java.util.concurrent.TimeoutException: null
at java.util.concurrent.FutureTask.get(FutureTask.java:205)
at com.icancontrol.meter.controller.ListMapWebSocket.execute(ListMapWebSocket.java:241)
at com.icancontrol.meter.controller.ListMapWebSocket.sendMessage(ListMapWebSocket.java:187)
at com.icancontrol.meter.common.thread.SendThread.run(SendThread.java:40)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
at java.lang.Thread.run(Thread.java:745)