当前位置: 首页 > news >正文

网站建设一条龙包括哪些服务西安建筑科技大学就业信息网

网站建设一条龙包括哪些服务,西安建筑科技大学就业信息网,山东省建设工程执业资格中心网站,宁波快速建站公司最近刚好有个场景,业务处理一份报告需要关注实时处理的进度。 本来打算使用前端轮训方式,但是考虑到这样效率比较低,也无法精确知道处理进度,就想到用websocket和前端实时交互,进度有更新就通知前端,避免了…

最近刚好有个场景,业务处理一份报告需要关注实时处理的进度。

本来打算使用前端轮训方式,但是考虑到这样效率比较低,也无法精确知道处理进度,就想到用websocket和前端实时交互,进度有更新就通知前端,避免了无用的空轮训请求。

websocket通过session链接,和前端保持链接,将客户的信息存储在内存中,用户每个请求都有一个唯一的uuid,后续通过uuid查找对应客户的session发送websocket。

系统架构图

流程图

import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.socket.config.annotation.EnableWebSocket;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;/*** @author * @since 2024/8/26*/
@Configuration
@EnableAutoConfiguration
@EnableWebSocket
public class WebSocketConfig {@Beanpublic ServerEndpointExporter serverEndpointExporter() {return new ServerEndpointExporter();}}

import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;import javax.websocket.OnClose;
import javax.websocket.OnError;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.util.concurrent.CopyOnWriteArraySet;/*** @author * @since 2024/8/26*/
@ServerEndpoint(value = "/websocket/progress")
@Component
@Slf4j
public class WebSocketServer {//静态变量,记录当前在线连接数private static int onlineCount = 0;//存储客户端对应的WebSocket对象private static CopyOnWriteArraySet<WebSocketServer> webSocketSet = new CopyOnWriteArraySet();//与某个客户端的连接会话private Session session;//客户端sidprivate String sid = "";//建立连接成功后调用的方法@OnOpenpublic void onOpen(Session session) {//获取当前会话this.session = session;//存储WebSocket对象String sid = getUuid();//客户端标识this.sid = sid;webSocketSet.add(this);//在线连接计数addOnlineCount();try {//调用服务端发送信息方法sendMessage("连接成功" + this.sid);} catch (Exception e) {log.error("IO异常:{} ", e);}}private String getUuid() {IdWorker idWorker = new IdWorker();long l = idWorker.nextId();return String.valueOf(l);}@OnClosepublic void onClose() {//移除WebSocket对象webSocketSet.remove(this);subOnlineCount();}@OnMessagepublic void onMessage(String message) {log.info("WebSocketServer 收到message{}", this.sid);}@OnErrorpublic void onError(Throwable error) {log.error("onError,{}", error);}public void sendMessage(String message) throws IOException {this.session.getBasicRemote().sendText(message);}public static void sendInfo(String message, String sid) {if (sid == null) {return;}for (WebSocketServer item : webSocketSet) {try {if (item.sid.equals(sid)) {//发给单一用户item.sendMessage(message);}} catch (IOException e) {log.error("发送websocket失败{}", e);}}}public static synchronized int getOnlineCount() {return onlineCount;}public static synchronized void addOnlineCount() {WebSocketServer.onlineCount += 1;}public static synchronized void subOnlineCount() {WebSocketServer.onlineCount -= 1;}
}

和前端本地测试的好好的,一点毛病没有,以为万事大吉了。

结果一发布到测试环境,gg了,进度条时好时坏,一开始还没查出来问题,没有往集群的方面想,还以为是哪里不小心自动关闭了websocket,查了n久之后,突然顿悟了,有集群啊,多台服务器,前端链接之后,session是保持了,但是一开始链接的可能是服务器A,后续的请求可能就到服务器B了,那就拿不到客户的session,就无法发送消息了。

找到问题,就得想解决方法了

方法一 固定ip请求,用nginx的ip_hash负载均衡请求,可惜行不通,问了运维那边,ip是动态了,搞不定。

方法二 改网关的负载均衡策略,因为原来的负载均衡策略是均衡分发,改到源地址策略,就会固定ip请求到同一服务器了,和nginx ip_hash类似,但是也行不通,不能改策略。

方法三 最后只剩这个方法,通过rabbitmq广播各个服务器,根据服务器本身存储的session信息(uuid),判断这个消息时候需要处理。

customer自己发送消息,自己监听,发送mq的时候会自动在头尾带上",以及会对字符串中的“引号进行转义,加上\,所以都要进行替换


import org.apache.commons.lang3.StringUtils;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.stereotype.Component;import java.io.IOException;/*** @author * @since 2024/9/19*/
@Component
public class WebsocketMqCustomer {@RabbitListener(queues = "#{psQueue.name}")public void pubsubMqMsg(String message) throws IOException {String[] xxxx = message.replace("\\", "").split(";;;");WebSocketServer.sendInfo(xxxx[1].substring(0, xxxx[1].length() - 1).replace("\\", ""), xxxx[0].substring(1));}public static void sendInfo(String message, String sid) {if (StringUtils.isNotEmpty(sid)) {RabbitTemplate rabbitTemplate = SpringUtil.getBean(RabbitTemplate.class);rabbitTemplate.convertAndSend(MainRabbitMqCreateConfig.EXCHANGE_WEBSOCKET, sid, sid + ";;;" + message);}}
}

还遇见了一个问题,@ServerEndpoint注解添加的类无法通过@Autowired直接注入,因为websocket处理是基于Servlet的,而Servlet容器并不像Spring mvc那样自动管理bean的依赖注入,可以通过Application来获取其他bean。

http://www.dtcms.com/a/425392.html

相关文章:

  • 网站首页的模块布局网站运营与建设
  • 哪些网站可以做驾考试题cms做网站
  • 电子商务网站建设合同标准范文安卓手机app开发工具软件
  • 小程序联盟商家版福州网seo
  • 软件园做网站杨凌企业网站开发
  • 做网站赚钱的点在哪里成都建站优化公司
  • 贵州软件开发 网站开发营销型网站制作流程
  • 做网站 收费网页类网站
  • wordpress子站点打不开商丘网站建设费用
  • 创业网站建设规划书宣传片拍摄思路怎么写
  • 第一次做网站程序开发公司名大全
  • 镇江建设银行网站成都 建设网站
  • 京东网站怎么做凡科建站源码
  • 杭州网站运营中国室内设计
  • 手机参数对比的网站中小型网站建设平台
  • 行业网站建设详解网站开发 流程图
  • 企业公示信息查询系统吉林小红书seo排名
  • seo建站推广机构组织网站建设
  • 网站建站要求建筑招聘最好的网站
  • 苏州网站网络推广潍坊外贸网站优化
  • 网站设计制作视频做网站需要先买域名吗
  • 景区网站怎么做郑州市招投标信息网
  • 网站开发毕业设计书广州平台公司
  • 东营专业网站建设辽宁招标网
  • 网站建设的公文格式龙岗专业做网站公司
  • 做网站游戏推广赚钱可喜安cms系统
  • 电子商务网站的建设论文台州建设网站制作
  • 邯郸市建设局查中级职称网站通过云主机建设网站
  • 大姚县建设工程招标网站房源网站建设
  • 网站开发常见毕业设计题目网站建设石家庄