一个 Java 的问题,HTTP 与 Websocket

2019-12-25 16:14:22 +08:00
 v2410117

现在的情况是这样的,Java 接收 HTTP 请求,在这个 HTTP 请求中会使用到 Websocket 去请求其他数据,怎么在这个 HTTP 中同步等待 Websocket 的业务返回?代码如下,主要是 Websocket 收到消息是走 OnMessage,我没想出好办法做这个处理,目前的处理办法就是在接到 HTTP 请求的时候,往数据库里加一条记录,然后一直使用这个记录 ID 循环查询数据库,当 Websocket 返回数据了,就将数据写入数据库并改变那条记录的状态,HTTP 请求里就直接相当于在数据库里查询到 Websocket 的业务返回,虽然这样做我是可以暂时完成功能,但好像感觉不太靠谱,求大神指点一下

public class TestController {
    @Autowired
    private MyWebSocket myWebSocket;

    @RequestMapping("/test")
    public Object test(){
       myWebSocket.sendText("ceshi");
       return xxx;
    }

}



@ServerEndpoint(port = 9898)
@Component
public class MyWebSocket {
    

private static final Set<Session> sessions = Collections.synchronizedSet(new HashSet<Session>());

    @OnOpen
    public void onOpen(Session session, HttpHeaders headers, ParameterMap parameterMap) throws IOException {
        System.out.println("new connection");
        sessions.add(session);
    }

    @OnMessage
    public void onMessage(Session session, String message) {
        System.out.println(message);
        if ("heart".equals(message)) {
            session.sendText("heart");
        }else {
            session.sendText(message);
        }
    }

     public void sendText(String msg) {
             for (Session session: sessions) {
                 session.sendText(msg);
             }
      }
}
4501 次点击
所在节点    Java
19 条回复
v2410117
2019-12-25 16:38:06 +08:00
求大佬来指点下思路呀! V2 大神们,出来了,别潜水了!
muzuiget
2019-12-25 16:45:51 +08:00
你那个数据库用法,其实就是当锁用了,关键词“Java Lock”,搜索下。
v2410117
2019-12-25 16:48:16 +08:00
@muzuiget 好的,感谢,我先搜索一下
672795574
2019-12-25 16:48:53 +08:00
用 CompletableFuture 试试?
gaius
2019-12-25 16:55:12 +08:00
不用 websocket 用服务调用,或者把 http 的业务流程也改成异步的
gy123
2019-12-25 17:07:19 +08:00
你写这个应该只是演示吧,因为你 http 收到请求后,发送给的是所有 session,然后需求是等待 onmessage 返回结果后响应 http;实际中应该是通过 http 请求给相应 session 发送吧?目前想到的方案是这样:
一.你要把 http 跟 websocket 全用 netty 来写可能灵活性好一些;
二.通过传递响应对象和锁工具:
(1) http 接收到请求后,拿到当前请求 sessionid 或者你根据业务自己生成的随机数,然后分别存储 key 为随机数,value 为 HttpServletResponse 和类似 CountDownLatch 对象存储到两个全局 map 中;并且将该随机数发送给 websocket 接收者;
后面用 CountDownLatch 等工具 hold 住线程;
(2)onmessage 接受到消息后,通过随机数取出 map 中 HttpServletResponse 对象和 CountDownLatch,进行返回数据和释放锁;
buruoyanyang
2019-12-25 17:10:26 +08:00
做过差不多的,http 请求过来,通过 MQ 调用其他模块等返回。
用 Future,然后做线程等待,等 MQ 回消息后,在唤醒等待线程即可
gy123
2019-12-25 17:22:00 +08:00
@buruoyanyang 具体说说,请教下?目前看你描述是在整个 http 请求同步代码中,只是使用 future 做异步请求 mq 拿到结果,如何能应用到他这个案例?
buruoyanyang
2019-12-25 17:32:48 +08:00
@gy123 http 请求->注册消息通知器->添加等待线程->提交到线程池,得到 Future->同步发送消息->收到 MQ 返回->唤醒等待线程->future.get()
大概是这么个流程,太久了也不太记得了
http 请求会在 future.get()被阻塞
他这个就是把 MQ 换成 WebSocket,WebSocket 返回后唤醒等待线程
pws22
2019-12-25 17:39:53 +08:00
@buruoyanyang 你这块 Future 是获取 mq 的返回状态码,还是说获取消费端的结果,如果是获取结果,是 future 去拉结果?
跟题主的问题可能有点出路吧
gy123
2019-12-25 17:41:47 +08:00
@buruoyanyang future.get()这个会阻塞没错,但是 callback 在 onmessage 里怎么拿到并做返回?这里有点疑问
pws22
2019-12-25 17:51:59 +08:00
要不直接内存是做判断 , http.sendMsg(Callback callback,long timeout),callback 里面传入你当前的 id,然后在内存中(map)去循环判断有没有该 id,当 socket 返回信息的时候,在内存(map)里面存入这个 id,其实跟你存数据库道理一样,最好就是不用 socket 那块,直接改成 http,都做到同步,或者是你在提供一个接口,是获取状态数据的,前台去长轮询获取最终结果
buruoyanyang
2019-12-25 17:58:14 +08:00
@gy123 所有被 wait 的消息通知器都会被丢到一个 key 为 id,value 为消息通知器的 map 里面,onMessage 通过消息 id 从 map 里面得到消息通知器,然后 notify,可能这个消息通知器的名字有点误导人
mizzle
2019-12-25 17:59:39 +08:00
这个场景用,涉及多线程交互。直观方法是 exchanger,或者用通知队列。
当年看的《如何造火箭》,还是能派上用场的。
buruoyanyang
2019-12-25 18:05:43 +08:00
@pws22 Future 获取的是 mq 的返回状态码,不获取结果,结果直接丢到消息通知器里面
zsy979
2019-12-25 18:24:46 +08:00
是不是我没看懂题 Thread.join 不🉑️吗
```
@RequestMapping("/test")
public Object test(){
Thread t = new Thread(new sendText());
t.run();
t.join();
//t 执行完执行。。
return xxx;
}
```
0NF09LJPS51k57uH
2019-12-25 19:09:37 +08:00
用发布订阅去做
http 业务逻辑中创建并监听这个 key 的 delete event
在 websocket 处理逻辑中删除这个 key。
araaaa
2019-12-25 19:11:11 +08:00
reactor 的 mono.zip
Malthael
2019-12-25 19:41:47 +08:00
DeferredResult 用过吗?我现在的项目 http 接口和 tcp 异步转同步用的就是这个。http 接口请求后发送 tcp 请求,然后等待客户端回复之后接口返回,还有超时设置。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/632230

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX