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

网络编程(一):服务器模型、Java I/O模型、Reactor事件处理模型、I/O复用

文章目录

  • 一、Socket和TCP/IP协议族的关系
  • 二、服务器模型
    • 1.C/S模型(Client/Server Model)
    • 2.P2P模型(Peer-to-Peer Model)
  • 三、Java的I/O演进
    • 1.BIO(阻塞)
      • (1)工作流程
      • (2)代码实现
    • 2.NIO(多路复用/轮询)
      • (1)工作流程
      • (2)代码实现
    • 3.NIO2.0——AIO(异步/事件驱动)
  • 四、并发事件处理模式
    • 1.Reactor模式(NIO)
      • (1)单Reactor单线程模型
      • (2)单Reactor多线程模型(Redis)
      • (3)主从Reactor多线程模型(Netty)
    • 2.Proactor模式(AIO)
  • 五、I/O复用

一、Socket和TCP/IP协议族的关系

Socket和TCP/IP协议族是网络编程中的两个重要概念,它们之间存在密切的关系。

首先,TCP/IP协议族是一组用于互联网通信的网络协议的集合。它由多个协议组成,其中最核心的协议是TCP(Transmission Control Protocol)和IP(Internet Protocol)。TCP协议提供可靠的数据传输和连接管理,而IP协议则负责将数据包从源地址传输到目的地址。

Socket是一种抽象概念,它提供了应用程序与网络之间的接口。通过Socket,应用程序可以通过网络与其他应用程序进行通信。在编程语言中,Socket通常被封装成库或API,以提供方便的网络编程接口。

TCP/IP协议族中的TCP协议使用Socket来实现端到端的数据传输。在TCP/IP网络中,每个主机都有一个唯一的IP地址,而每个运行TCP协议的应用程序都使用一个Socket来标识自己。一个Socket由IP地址和端口号组成,用于唯一标识网络中的一个应用程序。

当应用程序使用Socket进行网络通信时,它可以创建一个Socket对象,并指定目标主机的IP地址和端口号。通过Socket对象,应用程序可以使用TCP协议建立与目标主机的连接,并进行数据的发送和接收。TCP协议负责将数据分割成小的数据包,并通过IP协议将这些数据包从源主机传输到目的主机。

总结来说,Socket是应用程序与网络之间的接口,而TCP/IP协议族则是网络通信的基础协议。通过Socket和TCP协议,应用程序可以在TCP/IP网络中进行可靠的数据传输和通信。

二、服务器模型

服务器模型是指在网络中进行通信和资源共享时所采用的不同架构模式。常见的服务器模型包括C/S模型(Client/Server Model)和P2P模型(Peer-to-Peer Model)。

1.C/S模型(Client/Server Model)

C/S模型是一种常见的服务器模型,其中客户端(Client)和服务器(Server)之间存在明确的角色和功能分工。

  • 客户端:客户端是发起请求的一方,它向服务器发送请求并接收服务器的响应。客户端通常是一台终端设备,如个人计算机、智能手机等。
  • 服务器:服务器是提供服务的一方,它接收客户端的请求并提供相应的服务或资源。服务器通常是一台高性能的计算机或设备,具备处理请求和提供服务的能力。

在C/S模型中,客户端和服务器之间通过网络进行通信。客户端发起请求,服务器接收请求并处理,然后将响应发送回客户端。这种模型可以实现中心化的控制和管理,服务器负责处理和存储数据,客户端主要负责用户界面的展示和交互。

2.P2P模型(Peer-to-Peer Model)

P2P模型是一种去中心化的服务器模型,其中参与通信的设备之间平等地协作,没有明确的客户端和服务器的区别。

  • 对等节点:在P2P模型中,所有参与通信的设备都是对等节点,它们既是服务的提供者,也是服务的请求者。每个节点都可以与其他节点直接通信,共享资源或提供服务。

在P2P模型中,设备之间通过直接连接进行通信,而不依赖于中央服务器。每个设备既可以发起请求,也可以响应其他设备的请求,实现了资源和服务的共享。P2P模型常用于文件共享、实时通信等场景,例如BitTorrent协议就是一种典型的P2P协议。

总结来说,C/S模型是一种中心化的服务器模型,客户端和服务器之间存在明确的角色和功能分工;而P2P模型是一种去中心化的服务器模型,参与通信的设备平等地协作,共享资源和服务。选择适合的服务器模型取决于具体的应用需求和网络架构。

三、Java的I/O演进

1.BIO(阻塞)

网络编程的基本模型是Client/Server模型,也就是两个进程之间进行相互通信,其中服务端提供位置信息(绑定的IP地址和监听端口),客户端通过连接操作向服务端监听的地址发起连接请求,通过三次握手建立连接,如果连接建立成功,双方就可以通过网络套接字(Socket)进行通信。

在基于传统同步阻塞模型开发中,ServerSocket负责绑定IP地址,启动监听端口;Socket负责发起连接操作。连接成功之后,双方通过输入和输出流进行同步阻塞式通信。

BIO,即Blocking IO,阻塞型I/O。阻塞体现在两个地方,连接线程的阻塞和读写的阻塞。

(1)工作流程

服务端启动ServerSocket;
客户端启动 Socket 对服务器进行通信,服务端对每个客户端建立一个线程与之通讯(可以使用线程池进行优化);
客户端发出请求后,先咨询服务器是否有线程响应,如果没有则会等待(即阻塞);
如果有响应,客户端线程会等待请求结束后,再继续执行。

在这里插入图片描述

(2)代码实现

  • 服务端:
package bio;import java.io.IOException;
import java.io.PrintStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;public class BIOServer {//创建一个线程池,用于处理客户端连接后的工作public static ThreadPoolExecutor pool=new ThreadPoolExecutor(10, 10, 60, TimeUnit.SECONDS, new LinkedBlockingDeque<>());public static void main(String[] args) throws IOException{ServerSocket serverSocket=new ServerSocket(8888);while(true){//1 等待客户端连接是阻塞的Socket socket=serverSocket.accept();System.out.println("客户端连接上了");//2 连接上以后向线程池提交一个任务用于处理连接pool.execute(new Runnable() {@Overridepublic void run() {while(true){try{//读写也是阻塞的//创建输出流,server向client输出PrintStream printStream = new PrintStream(socket.getOutputStream());printStream.println("message from server 8888");printStream.close();socket.close();}catch(IOException e){e.printStackTrace();}}}});}}
}
  • 客户端:
package bio;import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.Socket;public class BIOClient {public static void main(String[] args) throws IOException{Socket socket = new Socket("127.0.0.1", 8888);BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));System.out.println("This message comes from server:"+bufferedReader.readLine());bufferedReader.close();socket.close();}
}

在这里插入图片描述

  • 缺点:
    • accept()等待客户端连接是阻塞的,有时候需要进行无谓的等待,效率低下,浪费资源。
    • 引入线程池进行优化提升了高并发能力,即能够同时处理多个客户端请求了,但是却带来了一个问题,随着开启的线程数目增多,将会消耗过多的内存资源,导致服务器变慢甚至崩溃。
    • 读写操作仍然是阻塞的,如果客户端半天没有操作,也会浪费资源,因此效率不高。

2.NIO(多路复用/轮询)

NIO,即non-blocking lO,非阻塞型IO。

  • 非阻塞——减少线程资源的浪费:
    BIO提供非阻塞读写模式,使一个线程从某通道发送请求或者读取数据,但是它仅能得到目前可用的数据,如果目前没有数据可用时,就什么都不会获取,而不是保持线程阻塞,所以直至数据变的可以读取之前,该线程可以继续做其他的事情。非阻塞写也是如此,一个线程请求写入一些数据到某通道,但不需要等待它完全写入,这个线程同时可以去做别的事情。可以做到 用一个线程来处理多个操作,体现了一种多路复用的思想。 而不是像BIO那样,一个连接过来就得分配一个线程,造成资源的浪费。

  • 处理数据的方式:
    BIO 以流的方式处理数据,而 NIO 以缓冲区(也被叫做块)的方式处理数据,块 IO 效率比流 IO 效率高很多。BIO 基于字符流或者字节流进行操作,而 NIO 基于 Channel 和 Buffer 进行操作,数据总是从通道读取到缓冲区或者从缓冲区写入到通道。

  • 复用:
    Selector(选择器)用于监听多个通道的事件(比如连接请求,数据到达等),因此使用单个线程就可以监听多个客户端通道。

(1)工作流程

Channel(通道),Buffer(缓冲区), Selector(选择器)为NIO的三大核心组件。

  • Channel(通道):
    相比于BIO流的读写,Channel的读写是双向的,既可以从通道中读取数据,又可以写数据到通道。通道可以非阻塞读取和写入通道/缓冲区,也支持异步地读写。

  • Buffer(缓冲区):
    在客户端和Channel之间,增加Buffer缓冲区的支持,更加容易操作和管理。

  • Selector(选择器):
    用来 轮询 检查一个或多个NIO通道,并确定哪些通道已经准备好进行读取或写入。这样,一个单独的线程可以管理多个channel,从而管理多个网络连接,提高效率。

在这里插入图片描述

(2)代码实现

代码来自:here

  • 服务端:
package nio;import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Iterator;
import java.util.Set;public class NIOServer {public static void main(String[] args) throws IOException {// 创建ServerSocketChannelServerSocketChannel serverSocketChannel = ServerSocketChannel.open();// 创建一个Selector对象,Selector selector = Selector.open();// 绑定端口6666, 在服务器端监听serverSocketChannel.socket().bind(new InetSocketAddress(6666));// 设置为非阻塞serverSocketChannel.configureBlocking(false);// 把serverSocketChannel注册到selectorserverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);// 循环等待用户连接while (true){if (selector.select(1000) == 0){ //等待(阻塞)一秒, 没有事件发生
//            if (selector.selectNow() == 0){ // 也可以设置成非阻塞的System.out.println("服务器等待了一秒,无连接");continue;}// 如果返回的>0 , 说明客户端有了动作,就获取相关的selectionKey集合Set<SelectionKey> selectionKeys = selector.selectedKeys(); // 返回关注事件的集合// 遍历selectionKeysIterator<SelectionKey> keyIterator = selectionKeys.iterator();while (keyIterator.hasNext()){// 获取到selectionKeySelectionKey key = keyIterator.next();//根据key对应的通道获取事件并做相应处理if (key.isAcceptable()){//如果是OP_ACCEPT, 表示有新的客户端产生//给该客户端生成SocketChannelSocketChannel socketChannel = serverSocketChannel.accept();//将socketChannnel设置为非阻塞socketChannel.configureBlocking(false);//将socketChannel注册到selector上, 设置事件为OP_READ,同时给socketChannel关联一个buffersocketChannel.register(selector,SelectionKey.OP_READ, ByteBuffer.allocate(1024));}if (key.isReadable()){// 发生了OP_READSocketChannel channel=(SocketChannel)key.channel();ByteBuffer buffer = (ByteBuffer)key.attachment();channel.read(buffer);System.out.println("from 客户端"+new String(buffer.array()));}// 手动从集合中移除当前的selectionKey, 防止多线程情况下的重复操作keyIterator.remove();}}}
}
  • 客户端:
package nio;import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;public class NIOClient {public static void main(String[] args) throws IOException {// 获取一个网络通道SocketChannel socketChannel = SocketChannel.open();// 设置为非阻塞socketChannel.configureBlocking(false);//设置服务器端ip和端口InetSocketAddress inetSocketAddress = new InetSocketAddress("127.0.0.1", 6666);if (!socketChannel.connect(inetSocketAddress)){while (!socketChannel.finishConnect()){//如果没有连接成功,客户端是非阻塞的,可以做其它工作System.out.println("等待连接...");}}// 如果连接成功,就发送数据String str = "hello world";ByteBuffer buffer = ByteBuffer.wrap(str.getBytes());// 发送数据 , 将buffer中的数据写入到channel中socketChannel.write(buffer);System.in.read();}}

在这里插入图片描述

3.NIO2.0——AIO(异步/事件驱动)

AIO,即Asynchronous I/O,异步非阻塞IO。AIO提供的最大的特点是具备异步功能,采用“订阅-通知”模式,即应用程序向操作系统注册IO监听,然后继续做自己的事情。当操作系统发生IO事件,并且准备好数据后,在主动通知应用程序,触发相应的函数。

在这里插入图片描述

下面是一段简单的代码示例:

import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousFileChannel;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;
import java.util.concurrent.Future;public class NIO2AsyncFileIOExample {public static void main(String[] args) {try {// 通过路径获取文件通道Path path = Paths.get("test.txt");AsynchronousFileChannel fileChannel = AsynchronousFileChannel.open(path, StandardOpenOption.READ, StandardOpenOption.WRITE);// 分配缓冲区ByteBuffer buffer = ByteBuffer.allocate(1024);// 异步读取文件Future<Integer> readResult = fileChannel.read(buffer, 0);while (!readResult.isDone()) {// 在等待异步读取完成时可以进行其他操作System.out.println("Waiting for read operation to complete...");}// 打印读取结果buffer.flip();System.out.println("Read data: ");while (buffer.hasRemaining()) {System.out.print((char) buffer.get());}System.out.println();// 异步写入数据String newData = "Hello, NIO 2.0!";buffer.clear();buffer.put(newData.getBytes());buffer.flip();Future<Integer> writeResult = fileChannel.write(buffer, 0);while (!writeResult.isDone()) {// 在等待异步写入完成时可以进行其他操作System.out.println("Waiting for write operation to complete...");}System.out.println("Data written to file.");// 关闭文件通道fileChannel.close();} catch (IOException e) {e.printStackTrace();}}
}

四、并发事件处理模式

1.Reactor模式(NIO)

使用 Java NIO 构建的 IO 程序,它的工作模式是:主动轮训 IO 事件,IO 事件发生后程序的线程主动处理 IO 工作,这种模式也叫做 Reactor 模式。

(1)单Reactor单线程模型

只有一个线程来执行所有的任务,效率低下,并且也有可靠性问题。

在这里插入图片描述

(2)单Reactor多线程模型(Redis)

相比于上一个模型,增加了线程池的支持,从一定程度上提升了并发效率,但是引入线程池可能会涉及到数据同步问题。Redis底层就是基于这种模型。

在这里插入图片描述

(3)主从Reactor多线程模型(Netty)

在上一个模型的基础上,一个Reactor变成了两个,主Reactor创建连接,从Reactor分发读写任务,能支持更高的并发量。Netty是基于这种模型。
在这里插入图片描述

2.Proactor模式(AIO)

使用 Java AIO 构建的 IO 程序,它的工作模式是:将 IO 事件的处理托管给操作系统,操作系统完成 IO 工作之后会通知程序的线程去处理后面的工作,这种模式也叫做 Proactor 模式。

现在AIO和Proactor使用还不怎么广泛。

五、I/O复用

select、poll、epoll

相关文章:

网络编程(一):服务器模型、Java I/O模型、Reactor事件处理模型、I/O复用

文章目录 一、Socket和TCP/IP协议族的关系二、服务器模型1.C/S模型&#xff08;Client/Server Model&#xff09;2.P2P模型&#xff08;Peer-to-Peer Model&#xff09; 三、Java的I/O演进1.BIO&#xff08;阻塞&#xff09;&#xff08;1&#xff09;工作流程&#xff08;2&am…...

flyway适配高斯数据库

文章目录 flyway适配高斯数据库 flyway适配高斯数据库 flyway-core 源码版本&#xff1a;6.2.2 tag 由于高斯和postgresql使用的驱动都是一样的&#xff0c;所以基于flyway支持已有的postgresql数据库来改造 修改点如下&#xff1a; 1、PostgreSQLConnection 类中的 doRest…...

LVS keepalived实现高可用负载群集

目录 1 Keepalived及其工作原理 1.1 Keepalived体系主要模块及其作用&#xff1a; 2 LVSKeepalived 高可用群集部署 2.1 配置负载调度器&#xff08;主、备相同&#xff09; 2.1.1 配置keeplived&#xff08;主、备DR 服务器上都要设置&#xff09; 2.1.2 启动 ipvsadm 服…...

HTTP RESTFul RPC

一、简介 &#xff08;1&#xff09;HTTP&#xff08;Hypertext Transfer Protocol&#xff09;是一种应用层协议。它经常用于在Web和服务器之间通讯&#xff0c;或服务与服务之间通讯。 &#xff08;2&#xff09;RESTFul 约束HTTP协议实现上的规范设计。 &#xff08;3&am…...

短视频seo矩阵系统源码开发搭建--代用户发布视频能力

短视频SEO矩阵系统源码开发搭建的代用户发布视频能力&#xff0c;主要是指在系统平台上&#xff0c;允许用户将其创作的内容发布到指定的账号或平台&#xff0c;并设置好相关的标题、话题、锚点等信息。 一、搭建步骤及注意事项 确定使用场景。根据业务需求&#xff0c;确定该…...

真实的产品开发中,后端的设计规约可以写哪些

真实的产品开发中&#xff0c;后端的设计规约可以写哪些 产品开发的后端设计规约通常包括以下内容&#xff1a; 数据模型设计&#xff1a;详细描述数据库的结构&#xff0c;包括数据表的设计、字段的定义和关系的设置等。 业务逻辑设计&#xff1a;详细描述后端的业务逻辑&a…...

Pytorch 多卡并行(2)—— 使用 torchrun 进行容错处理

前文 Pytorch 多卡并行&#xff08;1&#xff09;—— 原理简介和 DDP 并行实践 介绍了使用 Pytorch 的 DDP 库进行单机多卡训练的方法&#xff0c;本文进一步说明如何用 torchrun 改写前文代码&#xff0c;以提高模型训练的效率和容错性torchrun 是从 Pytorch 1.9.0 开始引入的…...

Java异常处理(详解)

Java异常处理 前言一、异常与异常类1.异常的概念2.异常类Error类Exception类&#xff08;1&#xff09;非检查异常&#xff08;2&#xff09;检查异常 二、异常处理1.异常的抛出与捕获2.try-catch-finally语句3.声明方法抛出异常3.用throw 语句抛出异常 三、自定义异常类 前言 …...

嵌入式-数据进制之间的转换

目录 一.简介 1.1十进制 1.2二进制 1.3八进制 1.4十六进制 二.进制转换 2.1二进制-十进制转换 2.2八进制-十进制转换 2.3十六进制-十进制转换 2.4十进制-二进制转换 2.5十进制-八进制转换 2.6十进制-十六进制转换 2.7小数部分转换 一.简介 被传入到计算机的数据要…...

腾讯mini项目-【指标监控服务重构】2023-08-20

今日已办 PPT制作 答辩流程 概述&#xff1a;对项目背景、架构进行介绍&#xff08;体现我们分组的区别和需求&#xff09;人员&#xff1a;小组成员进行简短的自我介绍和在项目中的定位&#xff0c;分工进展&#xff1a;对项目进展介绍&#xff0c;其中a、b两组的区别和工作…...

智能文本纠错API的应用与工作原理解析

引言 在数字时代&#xff0c;文本撰写和传播变得日益重要&#xff0c;无论是在学校里写论文、在职场中发送邮件&#xff0c;还是在社交媒体上发表观点。然而&#xff0c;文字错误、标点符号错误、语法问题和不当的表达常常会削弱文本的质量&#xff0c;降低信息传达的效果。为…...

在springboot下将mybatis升级为mybatis-plus

在springboot下将mybatis升级为mybatis-plus 1. 整体描述2. 具体步骤2.1 更新pom引用2.2 更新yml配置2.3 更新config配置2.4 BaseEntity修改 3. 程序启动4. 总结 1. 整体描述 之前项目工程用的是mybatis&#xff0c;现在需要将其替换为mybatis-plus&#xff0c;mybatis-plus的…...

Vuex详解:Vue.js的状态管理方案

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…...

栈与队列经典题目——用队列实现栈

本篇文章讲解栈和队列这一部分知识点的经典题目&#xff1a;用栈实现队列、用队列实现栈。对应的题号分别为&#xff1a;Leetcode.225——用队列实现栈&#xff0c;。 在对两个题目进行解释之前&#xff0c;先回顾以下栈和队列的特点与不同&#xff1a; 栈是一种特殊的线性表…...

Python stomp 发送消息无法显示文本

我们向消息服务器通过 stomp 发送的是文本消息。 当消息服务器发送成功后&#xff0c;消息服务器上的文本没有显示&#xff0c;显示的是 2 进制的数据。 如上图&#xff0c;消息没有作为文本来显示。 问题和解决 消息服务器是如何判断发送的小时是文本还是二进制的。 根据官…...

postgresql-视图

postgresql-视图 视图概述使用视图的好处 创建视图修改视图删除视图递归视图可更新视图WITH CHECK OPTION 视图概述 视图&#xff08;View&#xff09;本质上是一个存储在数据库中的查询语句。视图本身不包含数据&#xff0c;也被称为 虚拟表。我们在创建视图时给它指定了一个…...

科技资讯|Vision Pro头显无损音频仅限USB-C AirPods Pro 2耳机

彭博社的马克・古尔曼在最新发布的推文中表示&#xff0c;苹果 Vision Pro 头显的无损音频仅限于 USB-C AirPods Pro 2 耳机。 新款采用 USB-C 的 AirPods Pro 2 升级到了 IP54 级别&#xff08;原版不防尘&#xff0c;仅 IPX4 级抗水&#xff09;&#xff0c;可陪伴用户在恶劣…...

Postman应用——初步了解postman

Postman 是一个用于构建和使用 API 的 API 平台&#xff0c;Postman 简化了 API 生命周期的每个步骤并简化了协作&#xff0c;可以更快地创建更好的 API。 Postman 包含一个基于Node.js的强大的运行时&#xff0c;允许您向请求&#xff08;request&#xff09;和分组&#xff…...

分析报告显示,PHP是编程语言主力军,且在电商领域占据“统治地位”

日前有有业内专家透露了PHP语言的使用数据&#xff0c;并强调了PHP语言对于互联网的作用。 而根据W3 Techs发布的《全球前1000万个网站使用的编程语言分析(截至 2023.8)》中&#xff0c;有这样一组数据引起广泛的关注。PHP占比 77.2%、ASP占比 6.9%、Ruby 占比5.4%。 此外&am…...

关于Greenplum Platform Extension Framework(PXF)

本文翻译自 https://docs.vmware.com/en/VMware-Greenplum-Platform-Extension-Framework/6.6/greenplum-platform-extension-framework/overview_pxf.html 随着数据存储和云服务的爆炸式增长&#xff0c;数据现在以各种格式驻留在许多不同的系统中。通常&#xff0c;数据根据…...

Python|GIF 解析与构建(5):手搓截屏和帧率控制

目录 Python&#xff5c;GIF 解析与构建&#xff08;5&#xff09;&#xff1a;手搓截屏和帧率控制 一、引言 二、技术实现&#xff1a;手搓截屏模块 2.1 核心原理 2.2 代码解析&#xff1a;ScreenshotData类 2.2.1 截图函数&#xff1a;capture_screen 三、技术实现&…...

springboot 百货中心供应链管理系统小程序

一、前言 随着我国经济迅速发展&#xff0c;人们对手机的需求越来越大&#xff0c;各种手机软件也都在被广泛应用&#xff0c;但是对于手机进行数据信息管理&#xff0c;对于手机的各种软件也是备受用户的喜爱&#xff0c;百货中心供应链管理系统被用户普遍使用&#xff0c;为方…...

【Oracle APEX开发小技巧12】

有如下需求&#xff1a; 有一个问题反馈页面&#xff0c;要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据&#xff0c;方便管理员及时处理反馈。 我的方法&#xff1a;直接将逻辑写在SQL中&#xff0c;这样可以直接在页面展示 完整代码&#xff1a; SELECTSF.FE…...

React Native在HarmonyOS 5.0阅读类应用开发中的实践

一、技术选型背景 随着HarmonyOS 5.0对Web兼容层的增强&#xff0c;React Native作为跨平台框架可通过重新编译ArkTS组件实现85%以上的代码复用率。阅读类应用具有UI复杂度低、数据流清晰的特点。 二、核心实现方案 1. 环境配置 &#xff08;1&#xff09;使用React Native…...

leetcodeSQL解题:3564. 季节性销售分析

leetcodeSQL解题&#xff1a;3564. 季节性销售分析 题目&#xff1a; 表&#xff1a;sales ---------------------- | Column Name | Type | ---------------------- | sale_id | int | | product_id | int | | sale_date | date | | quantity | int | | price | decimal | -…...

在web-view 加载的本地及远程HTML中调用uniapp的API及网页和vue页面是如何通讯的?

uni-app 中 Web-view 与 Vue 页面的通讯机制详解 一、Web-view 简介 Web-view 是 uni-app 提供的一个重要组件&#xff0c;用于在原生应用中加载 HTML 页面&#xff1a; 支持加载本地 HTML 文件支持加载远程 HTML 页面实现 Web 与原生的双向通讯可用于嵌入第三方网页或 H5 应…...

【生成模型】视频生成论文调研

工作清单 上游应用方向&#xff1a;控制、速度、时长、高动态、多主体驱动 类型工作基础模型WAN / WAN-VACE / HunyuanVideo控制条件轨迹控制ATI~镜头控制ReCamMaster~多主体驱动Phantom~音频驱动Let Them Talk: Audio-Driven Multi-Person Conversational Video Generation速…...

短视频矩阵系统文案创作功能开发实践,定制化开发

在短视频行业迅猛发展的当下&#xff0c;企业和个人创作者为了扩大影响力、提升传播效果&#xff0c;纷纷采用短视频矩阵运营策略&#xff0c;同时管理多个平台、多个账号的内容发布。然而&#xff0c;频繁的文案创作需求让运营者疲于应对&#xff0c;如何高效产出高质量文案成…...

JVM 内存结构 详解

内存结构 运行时数据区&#xff1a; Java虚拟机在运行Java程序过程中管理的内存区域。 程序计数器&#xff1a; ​ 线程私有&#xff0c;程序控制流的指示器&#xff0c;分支、循环、跳转、异常处理、线程恢复等基础功能都依赖这个计数器完成。 ​ 每个线程都有一个程序计数…...

安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲

文章目录 前言第一部分&#xff1a;体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分&#xff1a;体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...