工作七年,对消息推送使用的一些经验和总结

前言:不管是APP还是WEB端都离不开消息推送,尤其是APP端,push消息,小信箱消息;WEB端的代办消息等。因在项目中多次使用消息推送且也是很多项目必不可少的组成部分,故此总结下供自己参考。

一、什么是消息推送

消息推送(Push)指运营人员通过自己的产品或第三方工具对用户当前网页或移动设备进行的主动消息推送。用户可以在网页上或移动设备锁定屏幕和通知栏看到push消息通知

二、消息推送的种类

从数据模型分:推和拉

从终端分:APP端和WEB端

从实现层面分:短论询、Comet(长轮询)、Flash XMLSocket、SSE、Web-Socket

类型概念优点缺点备注
短轮询客户端通过定期向服务器发送请求来获取最新的消息。服务器在接收到请求后立即响应,无论是否有新消息。如果服务器没有新消息可用,客户端将再次发送请求后端编写简单

高延迟:因客户端定期发起请求,导致消息延迟,尤其是定期时间设置过长时

高网络负载:无新消息时也会频繁发起请求,消耗服务器资源和网络

时效性差:服务器产生了新消息,客户端不能立马感知到,需等到轮询时间到

Comet(长轮询)客户端发起请求,服务器接到请求后hold住连接,直到有新消息(或超时)才返回响应信息并关闭连接,客户端处理完响应信息后再向服务器发送新的请求

减少请求次数:相对于短轮训而言

减少网络负载:没有消息时会保持连接,减少了频繁请求

时效性稍提高:相对于短轮询而言

没有新消息时会保持请求挂起,直到有新消息到达或超时。相比于短轮询,长轮询可以更快地获取新消息,减少了不必要的请求。
Flash XMLSocket在 HTML 页面中内嵌入一个使用了 XMLSocket 类的 Flash 程序。JavaScript 通过调用此 Flash 程序提供的socket接口与服务器端的socket进行通信网络聊天室,网络互动游戏使用较多

SSE(Server-send Events)

服务器主动推送时效性好:SSE使用了持久连接,可以实现比短轮询和长轮询更好的实时性

单向通道:SSE是单向的,只允许服务器向客户端推送消息,客户端无法向服务器发送消息

不适用低版本浏览器:SSE是HTML5的一部分,不支持低版本的浏览器。在使用SSE时,需要确保客户端浏览器的兼容性

Web-SocketWebSocket是一种双向通信协议,允许在单个持久连接上进行全双工通信

时效性最佳:WebSocket 提供了真正的双向通信,可以实现实时的双向数据传输,具有最佳的实时性

低延迟:与轮询和长轮询相比,WebSocket 使用单个持久连接,减少了连接建立和断开的开销,从而降低了延迟

双向通信:WebSocket 允许服务器与客户端之间进行双向通信,服务器可以主动向客户端发送消息,同时客户端也可以向服务器发送消息

较高的网络负载:WebSocket 使用长连接,会占用一定的网络资源。在大规模并发场景下,需要注意服务器的负载情况

浏览器支持:大多数现代浏览器都支持 WebSocket,但需要注意在开发过程中考虑不同浏览器的兼容性

短轮询:客户端定时轮询发起请求

长轮询:客户端发起请求,等待后端响应并再次发起请求

Flash XMLSocket:

原理示意图:

利用Flash XML Socket实现”服务器推”技术前提:
(1)Flash提供了XMLSocket类,服务器利用Socket向Flash发送数据;
(2)JavaScript和Flash的紧密结合JavaScript和Flash可以相互调用。
优点是实现了socket通信,不再利用无状态的http进行伪推送。但是缺点更明显:
1.客户端必须安装 Flash 播放器;
2.因为 XMLSocket 没有 HTTP 隧道功能,XMLSocket 类不能自动穿过防火墙;
3.因为是使用套接口,需要设置一个通信端口,防火墙、代理服务器也可能对非 HTTP 通道端口进行限制。

SSE:当使用Server-Sent Events(SSE)时,客户端(通常是浏览器)与服务器之间建立一种持久的连接,使服务器能够主动向客户端发送数据。这种单向的、服务器主动推送数据的通信模式使得实时更新的数据能够被实时地传送到客户端,而无需客户端进行轮询请求

SSE的工作原理如下: 

  1. 建立连接:客户端通过使用EventSource对象在浏览器中创建一个与服务器的连接。客户端向服务器发送一个HTTP请求,请求的头部包含Accept: text/event-stream,以表明客户端希望接收SSE数据。服务器响应这个请求,并建立一个持久的HTTP连接。

  2. 保持连接:服务器保持与客户端的连接打开状态,不断发送数据。这个连接是单向的,只允许服务器向客户端发送数据,客户端不能向服务器发送数据。

  3. 服务器发送事件:服务器使用Content-Type: text/event-stream标头来指示响应是SSE数据流。服务器将数据封装在特定的SSE格式中,每个事件都以data:开头,后面是实际的数据内容,以及可选的其他字段,如event:id:。服务器发送的数据可以是任何文本格式,通常是JSON。

  4. 客户端接收事件:客户端通过EventSource对象监听服务器发送的事件。当服务器发送事件时,EventSource对象会触发相应的事件处理程序,开发人员可以在处理程序中获取到事件数据并进行相应的操作。常见的事件是message事件,表示接收到新的消息。

  5. 断开连接:当客户端不再需要接收服务器的事件时,可以关闭连接。客户端可以调用EventSource对象的close()方法来显式关闭连接,或者浏览器在页面卸载时会自动关闭连接。

在Spring Boot中,可以使用SseEmitter类来实现SSE:

@RestController
public class SSEController {
    private SseEmitter sseEmitter;
    @GetMapping("/subscribe")
    public SseEmitter subscribe() {
        sseEmitter = new SseEmitter();
        return sseEmitter;
    }
    @PostMapping("/send-message")
    public void sendMessage(@RequestBody String message) {
        try {
            if (sseEmitter != null) {
                sseEmitter.send(SseEmitter.event().data(message));
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
<script>
    // 创建一个EventSource对象,指定SSE的服务端端点
    var eventSource = new EventSource('/subscribe');
    console.log("eventSource=", eventSource)
    // 监听message事件,接收从服务端发送的消息
    eventSource.addEventListener('message', function(event) {
        var message = event.data;
        console.log("message=", message)
        var messageContainer = document.getElementById('message-container');
        messageContainer.innerHTML += '<p>' + message + '</p>';
    });
</script>

上述过程:客户端可以通过访问/subscribe接口来订阅SSE事件,服务器会返回一个SseEmitter对象。当有新消息到达时,调用SseEmitter对象的send()方法发送消息。

Web-Socket:

HTML代码: 

<script>
    // 创建WebSocket对象,并指定服务器的URL
    var socket = new WebSocket('ws://localhost:8080/上下文路径/channel/message/');
    // 监听WebSocket的连接事件
    socket.onopen = function(event) {
        console.log('WebSocket connected');
    };
    // 监听WebSocket的消息事件
    socket.onmessage = function(event) {
        var message = event.data;
        var messageContainer = document.getElementById('message-container');
        messageContainer.innerHTML += '<p>' + message + '</p>';
    };
    // 监听WebSocket的关闭事件
    socket.onclose = function(event) {
        console.log('WebSocket closed');
    };
    // 发送消息到服务器
    function sendMessage() {
        var messageInput = document.getElementById('message-input');
        var message = messageInput.value;
        socket.send(message);
        messageInput.value = '';
    }
</script>

三、项目中使用的消息推送

例子1:Web-Socket
1.引入websocket依赖
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-websocket</artifactId>
 </dependency>
2.websocket配置
/**
 * @描述 开启WebSocket支持的配置类
 * 自动注册使用@ServerEndpoint
 */
@Configuration
public class WebSocketConfig {
    @Bean
    public ServerEndpointExporter serverEndpointExporter() {
        return new ServerEndpointExporter();
    }
}
3.websocket服务器端代码

说明:@ ServerEndpoint 注解是一个类层次的注解,主要是将当前类定义成一个websocket服务器端, 注解的值将被用于监听用户连接的终端访问URL地址,客户端可以通过这个URL来连接到WebSocket服务器端

/**
 * 消息推送
 **/
@ServerEndpoint("/channel/message/{user-id}")
@Slf4j
@Component
@RequiredArgsConstructor
public class TodoChannel implements ApplicationListener<FlowMessageEvent> {
    private static final Map<String, Set<Session>> SESSION_MAP = new ConcurrentHashMap<>();
    private Session session;
    private String userId;

    @OnMessage
    public void onMessage(String message) {
        log.info("websocket消息(id={}): {}", this.session.getId(), message);
    }

    @OnOpen
    public void onOpen(Session session, @PathParam("user-id") String userId) {
        this.session = session;
        this.userId = userId;
        val sessionSet = SESSION_MAP.getOrDefault(this.userId, new CopyOnWriteArraySet<>());
        sessionSet.add(session);
        SESSION_MAP.put(this.userId, sessionSet);
        log.info("websocket连接: id={}", this.session.getId());
        val message = new MessageModel();
        //往todoModel放业务数据
        session.getAsyncRemote().sendText(JSON.toJSONString(message));
    }

    @OnClose
    public void onClose(CloseReason closeReason) {
        val sessionSet = SESSION_MAP.get(this.userId);
        if (sessionSet != null) {
            sessionSet.remove(session);
        }
        log.info("websocket断开: id={} {}", this.session.getId(), closeReason);
    }

    @OnError
    public void onError(Throwable throwable) {
        log.warn("websocket异常: id={} throwable:", this.session.getId(), throwable);
        val sessionSet = SESSION_MAP.get(this.userId);
        if (sessionSet != null) {
            sessionSet.remove(this.session);
        }
        try {
            this.session.close(new CloseReason(CloseReason.CloseCodes.UNEXPECTED_CONDITION, throwable.getMessage()));
        } catch (IOException e) {
            log.error("websocket关闭失败", e);
        }
    }

    @Override
    public void onApplicationEvent(@NotNull FlowMessageEvent event) {
        Set<String> userIds = CastUtils.cast(event.getSource());
        userIds.forEach(id -> {
            val sessionSet = SESSION_MAP.get(id);
            if (sessionSet == null) {
                return;
            }
           //业务处理todo
            sessionSet.forEach(s -> s.getAsyncRemote().sendObject(JSON.toJSON(message)));
        });
    }

    @Scheduled(fixedRate = 24 * 60 * 60 * 1000L)
    public void sessionCleaner() {
        log.info("websocket message channel session清理");
        val keyToClean = new HashSet<String>();
        SESSION_MAP.forEach((k, v) -> {
            val sessionToClean = new HashSet<Session>();
            v.forEach(s -> {
                if (!s.isOpen()) {
                    sessionToClean.add(s);
                }
            });
            v.removeAll(sessionToClean);
            if (v.isEmpty()) {
                keyToClean.add(k);
            }
        });
        keyToClean.forEach(SESSION_MAP::remove);
    }


    @Data
    private static class MessageModel implements Serializable {
        private static final long serialVersionUID = 1L;
        private List<Info> list;
        private Integer size;

        @AllArgsConstructor
        @Value
        private static class Info implements Serializable {
            private static final long serialVersionUID = 1L;
            String type;
            String name;
        }
    }
}
 4.事件类代码
/**
 * message事件
 **/
public class FlowMessageEvent extends ApplicationEvent {
    public FlowMessageEvent(Object source) {
        super(source);
    }
}
5.使用事件推送消息

applicationEventPublisher.publishEvent(new FlowMessageEvent(user));

例子2:RabbitMq:
1.引入rabbitmq依赖
2.编写rabbitmq配置类
/**
 * @author wux
 * @version 1.0.0
 */
@SpringBootConfiguration
@Slf4j
public class RabbitMqConfig {
    @Value("${spring.rabbitmq.host:10.128.30.xxx}")
    private String host;

    @Value("${spring.rabbitmq.port:5672}")
    private int port;

    @Value("${spring.rabbitmq.username:guest}")
    private String username;

    @Value("${spring.rabbitmq.password:guest}")
    private String password;

    @Bean
    public ConnectionFactory connectionFactory() {
        CachingConnectionFactory factory = new CachingConnectionFactory(host, port);
        factory.setUsername(username);
        factory.setPassword(password);
        //连接工厂开启消息确认和消息返回机制
//        factory.setPublisherConfirmType(CachingConnectionFactory.ConfirmType.CORRELATED);
//        factory.setPublisherReturns(true);
        return factory;
    }

    @Bean
    public RabbitListenerContainerFactory<?> rabbitListenerContainerFactory(ConnectionFactory connectionFactory) {
        SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
        factory.setConnectionFactory(connectionFactory);
        //使用json 序列化和反序列化
        factory.setMessageConverter(new Jackson2JsonMessageConverter());
        //factory.setAcknowledgeMode(AcknowledgeMode.AUTO);
        return factory;
    }

    @Bean
    @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    public RabbitTemplate rabbitTemplate() {
        RabbitTemplate template = new RabbitTemplate();
        template.setConnectionFactory(connectionFactory());
        template.setMessageConverter(new Jackson2JsonMessageConverter());
        return template;
    }

    @Bean
    public Queue testDirectQueue() {
        return new Queue(RabbitMqConsts.TEST_QUE_PHM_WARN_INFO, true, false, false);
    }

    @Bean
    public DirectExchange testDirectExchange() {
        return new DirectExchange(RabbitMqConsts.TEST_EXC_PHM_WARN_INFO, true,false);
    }

    @Bean
    public Binding TestBinding() {
        return BindingBuilder.bind(testDirectQueue()).to(testDirectExchange()).with(RabbitMqConsts.TEST_KEY_PHM_WARN_INFO);
    }
}
3.编写rabbitmq工具类
/**
 * @author wux
 * @version 1.0.0
 * @description rabbit工具类
 */
@Slf4j
public class RabbitMqUtil {
    private static AmqpAdmin getAmqpAdmin() {
        return SpringContextUtils.getBean("amqpAdmin",AmqpAdmin.class);
    }
    /**
     通过amqpAdmin动态创建队列、交换机和绑定关系
     ttlFlag :设置消息过期时间
     */
    public static void createQueueAndExchangeIfNeed(String businessName, ExchangeEnum typeEnum, Integer ttl) {
        String exchangeName = "exc_" + businessName;
        String queueName = "que_" + businessName;
        String routingKey = "key_" + businessName;
        if (CheckUtil.isNotEmpty(getQueueInfo(queueName))) {
            return;
        }
        //创建队列
        Queue queue = createAndBindQueue(queueName, ttl);
        //创建交换机
        Exchange exchange = createAndBindExchange(exchangeName, typeEnum);
        //绑定队列和交换机
        switch (typeEnum){
            case DIRECT:
                binding(queueName, exchangeName, routingKey, typeEnum);
                break;
            case FANOUT:
                fanoutBinding(queue, exchange);
                break;
            default:
                binding(queueName, exchangeName, routingKey, typeEnum);
        }
    }

    private static Queue createAndBindQueue(String queueName, Integer ttl) {
        Map<String, Object> arguments = new HashMap<>();
        //设置过期时间,单位是毫秒
        if (CheckUtil.isNotEmpty(ttl)) {
            arguments.put("x-message-ttl", ttl);
        }
        if (CheckUtil.isEmpty(queueName)) {
            log.error("队列名称为空!queueName=" + queueName);
            throw new BusinessException("队列名称为空!");
        }
        Queue queue = new Queue(queueName, true, false, false, arguments);
        getAmqpAdmin().declareQueue(queue);
        return queue;
    }

    public static QueueInformation getQueueInfo (String queueName) {
        if (CheckUtil.isEmpty(queueName)) {
            return null;
        }
        return getAmqpAdmin().getQueueInfo(queueName);
    }

    private static Exchange createAndBindExchange(String exchangeName, ExchangeEnum typeEnum){
        AbstractExchange exchange = null;
        switch (typeEnum){
            case DIRECT:
                exchange = new DirectExchange(exchangeName, true, false);
                break;
            case TOPIC:
                exchange = new TopicExchange(exchangeName, true, false);
                break;
            case FANOUT:
                exchange = new FanoutExchange(exchangeName, true, false);
                break;
            case HEADERS:
                exchange = new HeadersExchange(exchangeName, true, false);
                break;
            default:
                exchange = new DirectExchange(exchangeName, true, false);
        }
        getAmqpAdmin().declareExchange(exchange);
        return exchange;
    }

    private static void binding(String queueName, String exchangeName, String routingKey, ExchangeEnum typeEnum) {
        //绑定队列和交换机
        Binding binding = new Binding(queueName, Binding.DestinationType.QUEUE, exchangeName, routingKey, null);
        getAmqpAdmin().declareBinding(binding);
    }

    private static void fanoutBinding(Queue queue, Exchange exchange) {
        BindingBuilder.bind(queue).to(exchange);
    }
}
4.监听和推送消息
/**
 * @author wux
 * @version 1.0.0
 * @description 监听phm设备状态消息
 */
@Component
@Slf4j
public class MotePhmDeviceStatesListener {
    private static final String CLASS_NAME = "MotePhmDeviceStatesListener";

    @Autowired
    private Map<String, AssembleDeviceStatesStrategy> map = new ConcurrentHashMap<String, AssembleDeviceStatesStrategy>();

    @Resource
    private MoteMessageService moteMessageService;

    @Autowired
    protected BeanMapper beanMapper;

    @RabbitHandler
    @RabbitListener(bindings = @QueueBinding(value=@Queue("que_phm_device_states"),exchange = @Exchange("exc_phm_device_states")
            ,key = "key_phm_device_states"))
    public void process(@Payload BaseMessage req, Message msg, Channel channel) {
        final String METHOD_NAME ="process";
        log.info(CLASS_NAME + "-" + METHOD_NAME + "-start,req={},msg={}", req, msg);
        long deliverTag = msg.getMessageProperties().getDeliveryTag();
        if (!MessageTypeEnum.DEVICE_STATES.getKey().equals(req.getType())) {
            log.warn(CLASS_NAME + "-" + METHOD_NAME + "-message=消息类型不匹配!");
            //拒绝,重新回到队列
            //channel.clearReturnListeners();
            //channel.basicReject(deliverTag, true);
            return;
        }

        try {
            String service = MessageSubTypeEnum.getService(req.getSubType());
            AssembleDeviceStatesStrategy strategy = map.get(service);
            BaseMessage reqMessage = beanMapper.map(req, BaseMessage.class);
            BaseMessage retMessage = strategy.assembleDeviceStates(reqMessage);
            strategy.sendMessage(retMessage, reqMessage, channel);
        } catch (Exception e) {
            log.error(CLASS_NAME + "-" + METHOD_NAME + "-异常, e={}", e);
            return;
            //拒绝,重新回到队列
            //channel.basicNack(deliverTag, false,true);
        }
    }
}
 public void sendMessage(BaseMessage retMessage, BaseMessage req, Channel channel) {
        retMessage.setDate(new Date());
        retMessage.setDateStr(DateUtils.format(retMessage.getDate(), DateUtils.DATE_TIME_SECOND));
        if (CheckUtil.isEmpty(req.getQueueName())) {
            log.warn("AssembleDeviceStatesStrategy" + "队列名称为空,req={}", req);
            return;
        }
        QueueInformation queueInfo = RabbitMqUtil.getQueueInfo(req.getQueueName());
        if (CheckUtil.isEmpty(queueInfo) || CheckUtil.isEmpty(queueInfo.getName())) {
            log.warn("AssembleDeviceStatesStrategy" + "-" + "队列不存在!" + ",req={}", req);
            return;
        }
//        try {
//            long count = channel.messageCount(req.getQueueName());
//            if (count >= 5000) {
//                channel.queueDelete(req.getQueueName());
//            }
//
//        } catch (IOException e) {
//            log.error("AssembleDeviceStatesStrategy" + "-" + "清除队列消息失败" + ",retMessage={}", retMessage, e);
//        }
        rabbitTemplate.convertAndSend(req.getQueueName(), retMessage);
        log.info("AssembleDeviceStatesStrategy" + "-" + "sendMessage推给前端信息" + ",retMessage={},req={}", retMessage, req);
    }
例子3:Kafka:

使用@KafkaListene(topics="xxx", groupId="xxx")  接受消息

四、消息中间件:RabbitMQ、RocketMQ、Kafka

高并发情况下,或者规模较大,推荐使用消息中间件,搭建一个公共平台,统一管理消息推送,项目层面进行隔离即可。

RabbitMQ可看:https://blog.csdn.net/baidu_35160588/article/details/89027810

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/362453.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Vue.js 中子组件向父组件传值的方法

Vue.js 是一款流行的 JavaScript 前端框架&#xff0c;它提供了一套完整的工具和 API&#xff0c;使得开发者可以更加高效地构建交互式的 Web 应用程序。其中&#xff0c;组件化是 Vue.js 的一个核心概念&#xff0c;通过组件化可以将一个复杂的应用程序拆分成多个独立的部分&a…

3D应用开发平台HOOPS Platforms优化制造流程和数字化转型

Tech Soft 3D公司的HOOPS Platform &#xff08;包括HOOPS Native Platform 和HOOPS Web Platform&#xff09;&#xff0c;是一种用于开发顶级3D软件的集成技术。具有高性能3D图形&#xff0c;准确&#xff0c;快速的CAD数据转换&#xff0c;3D数据发布以及与流行的建模内核的…

iOS_Xcode_LLDB调试常用命令

文章目录 结构常用命令&#xff1a;1、流程控制&#xff1a;2、常用命令3、进程信息&#xff1a;4、寄存器&#xff1a;register5、镜像&#xff1a;image6、内存&#xff1a;memory7、符号断点&#xff1a;breakpoint8、内存断点&#xff1a;watchpoint9、Tips&#xff1a; 结…

音视频数字化(音频数字化)

在音视频领域,人们始终追求无限还原现场效果,因此音频越逼真越好,视频越清晰越好。之所以我们需要将音视频信号由模拟转为数字,目的是在录制、存储、编辑、复制、回放等环节的不失真,尽量保持原有细节,不因以上操作,导致音画的质量下降。 为此,视频系统分辨率越来越高,…

【iOS ARKit】手动配置环境探头

在上节中我们已经了解了环境探头以及如何使用自动环境探头&#xff0c;这节一起了解如何使用手动配置环境探头。 在使用自动环境反射时&#xff0c;开发人员无须进行有关环境反射的任何操作&#xff0c;只需要设置自动环境反射即可&#xff0c;其余工作完全由 RealityKit 自动完…

ArcGIS Pro字段编号相关代码

字段属于SHP文件的重要组成部分&#xff0c;在某些时候需要对字段进行编号&#xff0c;这里为大家介绍一下字段编号相关的代码&#xff0c;希望能对你有所帮助。 数据来源 教程所使用的数据是从水经微图中下载的POI数据&#xff0c;除了POI数据&#xff0c;常见的GIS数据都可…

全面掌握Django的web框架Django Rest_Framework(一)

文章目录 Django Rest_Framework1. DRF介绍2.DRF特点3.环境安装与配置&#xff08;1&#xff09;DRF需要以下依赖&#xff08;2&#xff09;创建django项目 4.序列化器的使用&#xff08;1&#xff09;创建序列化器 5. 反序列化器使用 Django Rest_Framework 1. DRF介绍 Djan…

springboot141夕阳红公寓管理系统的设计与实现

基于Spring Boot的夕阳红公寓管理系统的设计与实现 摘 要 如今社会上各行各业&#xff0c;都在用属于自己专用的软件来进行工作&#xff0c;互联网发展到这个时候&#xff0c;人们已经发现离不开了互联网。互联网的发展&#xff0c;离不开一些新的技术&#xff0c;而新技术的…

【爬虫专区】批量下载PDF (无反爬)

天命&#xff1a;只要没反爬&#xff0c;一切都简单 这次爬取的是绿盟的威胁情报的PDF 先看一下结构&#xff0c;很明显就是一个for循环渲染 burp抓包会发现第二次接口请求 接口请求一次就能获取到了所有的数据 然后一个循环批量下载数据即可&#xff0c;其实没啥难度的 imp…

使用Postman做API自动化测试

Postman最基本的功能用来重放请求&#xff0c;并且配合良好的response格式化工具。 高级点的用法可以使用Postman生成各个语言的脚本&#xff0c;还可以抓包&#xff0c;认证&#xff0c;传输文件。 仅仅做到这些还不能够满足一个系统的开发&#xff0c;或者说过于琐碎&#…

【鸿蒙】大模型对话应用(三):跨Ability跳转页面

Demo介绍 本demo对接阿里云和百度的大模型API&#xff0c;实现一个简单的对话应用。 DecEco Studio版本&#xff1a;DevEco Studio 3.1.1 Release HarmonyOS SDK版本&#xff1a;API9 关键点&#xff1a;ArkTS、ArkUI、UIAbility、网络http请求、列表布局、层叠布局 页面跳…

谷歌seo如何发布外链?

在谷歌SEO中发布外链就像是在网络世界中搭建桥梁&#xff0c;你需要在别人的网站里上精心放置通往你网站的路径&#xff0c;这种路径一般是单向的&#xff0c;可能只使用一次&#xff0c;但这依然是个需要花心思的工作 而对于谷歌seo的外链&#xff0c;很多人都会有一个误解&am…

pnpm : 无法加载文件 D:\tool\nvm\nvm\node_global\pnpm.ps1,因为在此系统上禁止运行脚本

你们好&#xff0c;我是金金金。 场景 新创建的项目&#xff0c;在vscode编辑器终端输入 pnpm i&#xff0c;显示报错如上 解决 在终端输入get-ExecutionPolicy(查看执行策略/权限) 输出Restricted(受限的) 终端再次输入Set-ExecutionPolicy -Scope CurrentUser命令给用户赋予…

Oracle篇—普通表迁移到分区表(第五篇,总共五篇)

☘️博主介绍☘️&#xff1a; ✨又是一天没白过&#xff0c;我是奈斯&#xff0c;DBA一名✨ ✌✌️擅长Oracle、MySQL、SQLserver、Linux&#xff0c;也在积极的扩展IT方向的其他知识面✌✌️ ❣️❣️❣️大佬们都喜欢静静的看文章&#xff0c;并且也会默默的点赞收藏加关注❣…

【乳腺肿瘤诊断分类及预测】基于PNN概率神经网络

课题名称&#xff1a;基于PNN的乳腺肿瘤诊断分类及预测 版本日期&#xff1a;2023-06-15 运行方式: 直接运行PNN0501.m 文件即可 代码获取方式&#xff1a;私信博主或QQ&#xff1a;491052175 模型描述&#xff1a; 威斯康辛大学医学院经过多年的收集和整理&#xff0c;建…

JAVASE进阶:内存原理剖析(1)——数组、方法、对象、this关键字的内存原理

&#x1f468;‍&#x1f393;作者简介&#xff1a;一位大四、研0学生&#xff0c;正在努力准备大四暑假的实习 &#x1f30c;上期文章&#xff1a;首期文章 &#x1f4da;订阅专栏&#xff1a;JAVASE进阶 希望文章对你们有所帮助 技术栈我已经基本上是学完了的&#xff0c;这段…

LeetCode 828. 统计子串中的唯一字符

一开始想的是两次前缀和&#xff0c;发现自己蠢了 看了灵神的题解&#xff0c;类似于DP的思想 我们维护以每个字符串结尾的子字符串对答案的贡献&#xff0c;s[i]的贡献是多少&#xff1f;首先我们知道他需要自己单独一个串或者接在以s[i-1]结尾的那些字符串的后面&#xff0c…

【C++】类和对象(1)

上节我们学习了C入门的一些语法知识&#xff0c;这篇博客来学习类和this指针。 目录 面向过程和面向对象的初步认识 类的引入 类的定义 类的访问限定符及封装 访问限定符 封装 类的作用域 类的实例化 类对象大小 this指针 this指针特性 面向过程和面向对象的初步认识…

Transformer 自然语言处理(二)

原文&#xff1a;Natural Language Processing with Transformers 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 第五章&#xff1a;文本生成 基于 Transformer 的语言模型最令人不安的特点之一是它们生成的文本几乎无法与人类写的文本区分开。一个著名的例子是 Ope…

Redis 管道技术——Pipeline

背景 面试官&#xff1a;Redis 管道技术pipeline用过吗&#xff1f;为什么要用&#xff1f;解决什么问题&#xff1f;使用过程中应该注意什么&#xff1f; 我&#xff1a;一键三连&#xff0c;内心gg了&#xff0c;没听说过&#xff0c;也不知道用来干什么的&#xff0c;我只能…