http/sse/websocket 三大协议演化历史以及 sse协议下 node.js express 服务实现打字机案例 负载均衡下的广播实现机制

news2025/1/11 4:02:06

背景

自从2022年底chatgpt上线后,sse就进入了大众的视野,之前是谁知道这玩意是什么?但是打字机的效果看起来是真的很不错,一度吸引了很多人的趋之若鹜,当然了这个东西的确挺好用,而且实现很简单,之前我用python的demo讲了一下SSE的概念,看起来有很多人看,但是并没有说明白这个原理,这次再彻底把这个原理给说明白,而且我发现通过node.js 的Express框架来说明这个概念更加简洁,所以今天就用Express框架来说明SSE概念,这样对前端同学更加友好。

之前的Python SSE的文章:

https://blog.csdn.net/wangsenling/article/details/130911465

https://blog.csdn.net/wangsenling/article/details/130490769

回看协议演化历史

学过计算机网络的人都知道socket连接就是全双工的,怎么到http协议这里就变成了单向的了?且连一次就over了,这其实是浏览器编程者故意而为,另外服务端根本不保留会话信息,处理完成直接就把这次请求的相关信息从内存清理了,所以才出现这种单向沟通,后来网络资源越来越便宜,才逐渐的走向了SSE和Websocket

HTTP 协议设计背景

HTTP 是在 1990 年代早期互联网环境中设计出来的,当时网络资源(如带宽和连接数)非常有限。基于这种限制,HTTP 协议被设计为无状态、短连接的模型,以尽可能地节省服务器和客户端的资源。这意味着:

  1. 请求-响应模型:HTTP 是典型的请求-响应协议,客户端发起请求,服务器处理并返回响应,随后连接立即关闭。这样设计的目的是为了快速释放资源,特别是服务器的连接数。

  2. 短连接:早期的 HTTP/1.0 协议默认每个请求完成后都立即关闭连接,这种方式减少了保持大量长连接对服务器造成的负担,但也带来了效率上的问题,特别是在需要频繁通信的情况下。

  3. 无状态:HTTP 的无状态设计使得每个请求都是独立的,服务器不必保留任何会话信息,从而进一步降低了对服务器资源的需求。这在资源稀缺的互联网早期非常重要。

HTTP 的局限性

随着网络应用的复杂性增加,HTTP 的短连接和无状态特性逐渐暴露出一些问题,特别是在需要实时更新双向通信的应用场景中,例如:

  • 实时聊天在线协作工具股票行情更新通知系统等。

  • 在这些场景下,HTTP 的传统请求-响应模型显得过于笨重,因为每次更新都需要客户端主动发起新的请求。

为了弥补这些局限,出现了一些技术,例如:

  • 轮询Polling:客户端定期向服务器发送请求,以获取最新的数据。这种方法虽然可以在一定程度上实现实时更新,但效率较低,因为大量请求可能只会获得很少或没有新数据。

  • 轮询(Long Polling:客户端发送请求后,服务器保持连接打开直到有新数据,然后返回响应。这减少了一些不必要的请求,但仍然需要不断建立新连接,并且是非标准化的。

SSE 的诞生

为了解决 HTTP 请求-响应模型中的不足,SSE 被引入。SSE 是基于 HTTP 协议之上的一种扩展,它允许服务器在一个长时间保持的连接中,不断地向客户端推送事件流。与 WebSocket 相比,SSE 更加轻量,并且完全基于 HTTP 协议,这使它具有很好的兼容性。

  • 长连接保持:SSE 通过建立一个 HTTP 长连接,使得服务器可以在连接保持的状态下,推送多个事件。这样一来,服务器可以在有新数据时立即推送给客户端,而客户端不必频繁发起新的请求。

  • 单向通信:SSE 仅允许服务器向客户端发送数据,这与 WebSocket 的全双工通信形成对比。虽然通信方式有限,但它简化了很多实时更新场景中的开发工作,并且避免了 WebSocket 的复杂性。

  • 兼容 HTTP:由于 SSE 基于 HTTP 协议,因此它能够很好地与现有的 HTTP 基础设施(如代理、防火墙等)配合工作,不容易遇到兼容性问题。

为什么 HTTP 自动关闭连接?

你提到的 HTTP 自动关闭连接是基于早期互联网的设计初衷——节省资源。在那个时代,保持长连接对于资源有限的服务器来说是一个很大的负担:

  • 资源节约:每个连接都占用系统的文件描述符、内存和 CPU 资源。如果每个客户端保持长时间连接,服务器很容易耗尽这些资源。因此,HTTP 通过每次请求结束后立即关闭连接来减少服务器负担。

  • 提高并发能力:通过让连接快速关闭,服务器可以同时处理更多的客户端请求,提高并发能力。

WebSocket 与 HTTP 的不同设计哲学

正如你提到的,WebSocket 是长连接全双工的,它在设计上更类似于低层的 TCP 套接字通信。这使得 WebSocket 更适合需要实时双向通信的应用,比如聊天、在线游戏等。

  • WebSocket 提供的长连接全双工通信:适用于需要双向持续交互的场景,但它不是基于 HTTP 的标准模型,因此需要在协议层上进行升级。

相比之下,HTTP 的设计初衷是基于短连接的请求-响应模型,更适合传统的静态内容传输场景。在现代互联网中,虽然有了更高效的 WebSocket,但 HTTP 的请求-响应模型仍然有其合理性,尤其在静态资源加载、API 请求等方面。

总结

SSE 的出现正是为了解决 HTTP 的局限性,提供一种简单、基于 HTTP 协议的长连接机制,适用于实时更新但无需复杂双向通信的场景。HTTP 早期设计为短连接的原因在于资源有限,自动关闭连接是为了节省资源。而像 WebSocket 这样的长连接协议则适用于实时性和双向通信要求较高的场景,因此两者都有各自的应用场景。

SSE就是HTTP协议下的一个协议补充

Demo效果

 

Gitee 源码地址:

https://gitee.com/sen2020/express-test

我们来看request和response的格式,请求核心就两个,返回核心是三个,不缓存,保持存活,事件流类型,只要发出去的http请求带着这些header参数,那么SSE协议连接就建立了,非常的简单。

 

结合之前的文档,你就能理解了,fetch只要拼凑出来这样header就可以建立SSE连接,不需要什么特殊的处理,几乎任何一个Web框架都支持,接下来用node.js的Express实现一个打字机的小demo

服务端代码

import express from 'express';
import fs from 'fs';
import readline from 'readline';
import { EventEmitter } from 'events';
import path from 'path';
import { fileURLToPath } from 'url';

const app = express();
const port = 3000;

// 获取当前文件的路径
const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(__filename);

// 创建事件管理器
const eventEmitter = new EventEmitter();

// 设置静态文件目录来提供前端页面和资源
app.use(express.static(path.join(__dirname, 'public')));

// SSE 路由,根据 ISBN 请求电子书
app.get('/events', (req, res) => {
    const { isbn } = req.query;

    if (!isbn) {
        res.status(400).send('Missing ISBN');
        return;
    }

    // 设置 headers 确保保持长连接
    res.setHeader('Content-Type', 'text/event-stream');
    res.setHeader('Cache-Control', 'no-cache');
    res.setHeader('Connection', 'keep-alive');

    // 确保每个连接只会被注册一次,避免重复订阅同一事件
    const onNewData = (data) => {
        res.write(`data: ${JSON.stringify(data)}\n\n`);
    };

    eventEmitter.on(isbn, onNewData);

    // 当客户端断开时,移除事件监听
    req.on('close', () => {
        eventEmitter.removeListener(isbn, onNewData);
        res.end();
    });

    // 开始读取并发送字符流
    readFileCharacterByCharacter(isbn);
});

// 从文件中逐字符读取数据并触发事件
const readFileCharacterByCharacter = async (isbn) => {
    const filePath = path.join(__dirname, 'books', `${isbn}.txt`);

    // 如果文件不存在,则返回错误
    if (!fs.existsSync(filePath)) {
        eventEmitter.emit(isbn, { message: `Error: Book with ISBN ${isbn} not found.` });
        return;
    }

    const fileStream = fs.createReadStream(filePath);
    const rl = readline.createInterface({
        input: fileStream,
        crlfDelay: Infinity,
    });

    for await (const line of rl) {
        for (const char of line) {
            await new Promise(resolve => setTimeout(resolve, 50)); // 模拟打字机效果,每50ms发送一个字符
            eventEmitter.emit(isbn, { message: char });
        }
        eventEmitter.emit(isbn, { message: '\n' });
    }

    // 告诉前端书本内容已经结束
    eventEmitter.emit(isbn, { message: 'End of book.' });
};

app.listen(port, () => {
    console.log(`SSE server and static files serving at http://localhost:${port}`);
});

html代码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Book Reader</title>
    <link rel="stylesheet" href="styles.css">
</head>
<body>
<div class="container">
    <h1>Book Reader</h1>
    <div class="form-container">
        <input type="text" id="isbn" value="1234567890" placeholder="Enter ISBN">
        <button id="start">Start Reading</button>
    </div>
    <div id="content"></div>
</div>
<script src="scripts.js"></script>
</body>
</html>

前端JS代码

document.getElementById('start').addEventListener('click', () => {
    const isbn = document.getElementById('isbn').value.trim();

    if (!isbn) {
        alert('Please enter a valid ISBN');
        return;
    }

    const contentDiv = document.getElementById('content');
    contentDiv.innerHTML = ''; // 清空之前的内容

    // 如果已经有一个 EventSource,先关闭它
    if (window.eventSource) {
        window.eventSource.close();
    }

    // 创建一个新的 EventSource 连接
    window.eventSource = new EventSource(`/events?isbn=${isbn}`);

    let currentLine = ''; // 用于累积当前行的字符
    let paragraph = document.createElement('p'); // 创建一个段落元素
    contentDiv.appendChild(paragraph); // 初始添加一个段落

    window.eventSource.onmessage = (event) => {
        const data = JSON.parse(event.data);
        const char = data.message;

        // 检查是否接收到 "End of book." 消息
        if (char === 'End of book.') {
            paragraph = document.createElement('p');
            paragraph.textContent = 'End of book.';
            contentDiv.appendChild(paragraph);

            // 关闭 EventSource 连接
            window.eventSource.close();
            return; // 停止进一步处理
        }

        // 如果是换行符,渲染当前行并创建新的段落
        if (char === '\n') {
            paragraph = document.createElement('p'); // 创建新的段落
            contentDiv.appendChild(paragraph); // 添加新段落到内容区
            currentLine = ''; // 清空当前行
        } else {
            // 累积字符到当前行并更新当前段落内容
            currentLine += char;
            paragraph.textContent = currentLine; // 逐字符更新当前段落内容
        }

        // 自动滚动到底部
        contentDiv.scrollTop = contentDiv.scrollHeight;
    };

    window.eventSource.onerror = () => {
        console.log('EventSource connection closed.');
        window.eventSource.close();
    };
});

CSS样式表

body {
    font-family: Arial, sans-serif;
    background-color: #f4f4f4;
    margin: 0;
    padding: 0;
    display: flex;
    justify-content: center;
    align-items: center;
    height: 100vh;
}

.container {
    text-align: center;
    background-color: #fff;
    padding: 20px;
    border-radius: 8px;
    box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
}

.form-container {
    margin-bottom: 20px;
}

#isbn {
    padding: 10px;
    font-size: 16px;
    width: 200px;
    margin-right: 10px;
}

button {
    padding: 10px 15px;
    font-size: 16px;
    cursor: pointer;
}

#content {
    text-align: left;
    margin-top: 20px;
    max-height: 300px;
    overflow-y: auto;
    background-color: #f9f9f9;
    padding: 15px;
    border: 1px solid #ddd;
}

目录结构

 package.json配置

 

SSE的应用场景注意事项

  1. php这种语言是无法搞这个事情了,因为一个连接需要一个进程,一台主机4核4G,估计也只能建立200个进程就是极限了,再长期保持,就是扯淡,所以你必须用异步IO的Web框架来实现SSE的连接,才可以大大提高连接数。

用SSE做推送-负载均衡场景怎么破?

  1. 首先我们要知道的是,一个Web服务一旦启动,就是一个独立的进程,你想往这个进程中塞东西,要么你能拿到这个进程对外的变量app或者server,这个大概率不可能,因为启动时是不对外提供这种变量的,那怎么办?

    1. 这肯定就是进程间通信的事情了,这里不做深入探讨,无非就是管道,共享内存,队列等等,还有个套接字?套接字是啥?说实话我既懂又不懂

    2. 但是这个进程可以注册到一个端口上,这样外界就可以发消息进来,这也合理,就像人有个耳朵一样,如果一个人聋了,你当然是没办法跟他说话的。

    3. 可以让这个进程自己提供一个http服务,这样大家调用时就通用了,通过这个http接口,我们就可以将信息发送给这个服务里去

  2. 拿来做服务端推送机制,那就是要在服务器级别收集client,如果是负载均衡方式启动了多台服务器怎么办?

    1. 刚好1列表就讲到这个问题,每个独立服务都将自己的IP:端口号存放在redis中,同时也罢自己的收集到的client创建个编码塞到redis里去,例如user_id:client-ip-port 这种对应,这样如果有一个广播需要广播一批人,那么就能从redis中找到这批人,然后再提取他们所在的ip:port,然后把消息发给这些服务,这些服务接收到user_id后,再在自己Map表中找到client,然后一个个发送数据过去,就实现了负载均衡下的消息推送,你学废了吗?

    2. 类似hyperf框架,这种开了多个进程来承接不同的client,玩法是一样,找到user_id:ip:port:process_id,就可以实现同样的广播效果,不过这玩意自己实现起来有点麻烦,我也没玩过,看了下调度就是这样实现的。

Express实现单服务的广播机制代码

import express from 'express';
import fs from 'fs';
import readline from 'readline';
import { EventEmitter } from 'events';
import path from 'path';
import { fileURLToPath } from 'url';

const app = express();
const port = 3000;

// 获取当前文件的路径
const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(__filename);

// 创建事件管理器
const eventEmitter = new EventEmitter();

// 全局连接管理器
let clients = [];

// 设置静态文件目录来提供前端页面和资源
app.use(express.static(path.join(__dirname, 'public')));

// SSE 路由,客户端连接到服务器
app.get('/events', (req, res) => {
  const headers = {
    'Content-Type': 'text/event-stream',
    'Cache-Control': 'no-cache',
    'Connection': 'keep-alive',
  };

  res.writeHead(200, headers);

  // 新的客户端连接,加入全局连接管理器
  clients.push(res);

  // 清理客户端断开连接时的处理
  req.on('close', () => {
    clients = clients.filter(client => client !== res);
    res.end();
  });
});

// 广播新书上架通知给所有客户端
const broadcastNewBook = (bookTitle) => {
  clients.forEach(client => {
    client.write(`data: ${JSON.stringify({ message: `New book available: ${bookTitle}` })}\n\n`);
  });
};

// 上架新书时调用此函数
app.post('/new-book', express.json(), (req, res) => {
  const { title } = req.body;
  
  if (!title) {
    res.status(400).send('Book title is required');
    return;
  }

  // 广播新书消息给所有客户端
  broadcastNewBook(title);

  res.status(200).send(`New book "${title}" broadcasted to all clients.`);
});

// 启动服务器
app.listen(port, () => {
  console.log(`SSE server running at http://localhost:${port}`);
});

Express中如何一个启动一个服务同时支持http/websocket两种协议?

  1. 这个要想明白一个事情,Express本身分为两个模块,其他Web框架也是这样的,一个是协议处理模块app,另外,也即当http发送过来请求时,浏览器会根据你输入的协议http://ws:// 在request的header中追加一个upgrade:websocket,有了这个标识,被Express的http模块识别到之后,它调用的app,就是ws的模块的app,所有的逻辑都走这边,刚才我们已经了解到http的关闭是由app模块发的close来控制的,只要这边不发close,这个连接就会一直保持着,因为连接就是socket,socket就是全双工的,所以,只要ws模块不主动发close过来,那么这个连接就可以保持长连接,同时ws模块会把这个连接的信息记录下来,以方便后续不断地再接收数据,再发回数据,这就是两个协议可以共用一个端口的机制。

  2. 将两种协议整合在一个Express下有什么好处?他们连接对象都是在一个进程中,因为启动时就启动了一个进程,这个进程启动了http模块和ws模块,所以两者共用一个一套上下文,因此如果你在Express启动前创建一个clients = [],将连接过来的所有ws client都塞进去,那就意味着?

    1. 意味着你可以通过发送http请求给Express服务,来实现广播效果,是不是爽歪歪?

 

效果截图

服务端代码:

import express from 'express';
import { WebSocketServer } from 'ws';
import path from 'path';
import { fileURLToPath } from 'url';

const app = express();
const port = 3000;

// 使用 JSON 中间件解析 POST 请求体
app.use(express.json());

// 获取当前文件的路径
const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(__filename);

// 配置静态文件目录
app.use(express.static(path.join(__dirname, 'public/ws')));

app.use((req, res, next) => {
    res.setHeader("Content-Security-Policy", "connect-src 'self' http://localhost:3000");
    next();
});

// 创建 WebSocket 服务器并与 Express 集成
const wss = new WebSocketServer({ noServer: true });

// 保存所有 WebSocket 客户端
let clients = [];

// WebSocket 连接处理
wss.on('connection', (ws) => {
    console.log('Client connected');
    clients.push(ws);

    // 处理消息
    ws.on('message', (message) => {
        console.log(`Received message: ${message}`);
    });

    // 当客户端断开连接时,移除它
    ws.on('close', () => {
        clients = clients.filter(client => client !== ws);
        console.log('Client disconnected');
    });
});

// 广播消息给所有 WebSocket 客户端
const broadcastMessage = (message) => {
    clients.forEach(client => {
        if (client.readyState === client.OPEN) {
            client.send(message);
        }
    });
};

// 上架新书的 HTTP 接口
app.post('/new-book', (req, res) => {
    const { title } = req.body;

    if (!title) {
        return res.status(400).send('Book title is required');
    }

    const message = `New book available: ${title}`;
    broadcastMessage(message);

    res.status(200).send(`New book "${title}" has been broadcasted to all WebSocket clients.`);
});

// 处理升级请求,WebSocket 连接时需要的处理逻辑
app.server = app.listen(port, () => {
    console.log(`Server is running on port ${port}`);
});

app.server.on('upgrade', (request, socket, head) => {
    wss.handleUpgrade(request, socket, head, (ws) => {
        wss.emit('connection', ws, request);
    });
});

Html 代码

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>WebSocket Client</title>
</head>
<body>
<h1>WebSocket Book Notification</h1>
<div id="content"></div>

<script>
  const contentDiv = document.getElementById('content');

  // 创建 WebSocket 连接
  const ws = new WebSocket('ws://localhost:3000');

  // 处理接收到的消息
  ws.onmessage = (event) => {
    const paragraph = document.createElement('p');
    paragraph.textContent = `Received: ${event.data}`;
    contentDiv.appendChild(paragraph);
  };

  // 处理 WebSocket 连接打开
  ws.onopen = () => {
    console.log('WebSocket connected');
  };

  // 处理 WebSocket 连接关闭
  ws.onclose = () => {
    console.log('WebSocket disconnected');
  };
</script>
</body>
</html>

js发送广播的请求fetch

fetch('http://localhost:3000/new-book', {
    method: 'POST',
    headers: {
        'Content-Type': 'application/json'
    },
    body: JSON.stringify({
        title: 'The Great Adventure WSSSSSSSS'
    })
})
.then(response => response.text())
.then(data => console.log(data))
.catch(error => console.error('Error:', error));

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux环境本地搭建开发工具箱It-Tools并实现公网环境远程使用

文章目录 前言1. 安装Docker2.本地安装部署it-tools3. it-tools工具箱功能—生成docker-compose文件4. 安装cpolar内网穿透5. 固定it-tools公网地址 前言 本篇文章&#xff0c;我们将以Docker方式将IT-Tools部署至本地Linux系统个人服务器&#xff0c;并且结合cpolar内网穿透工…

【无标题】mysql读写分离架构+MyCAT实现读写分离

1、读写分离的目的 数据库负载均衡&#xff1a; 当数据库请求增多时&#xff0c;单例数据库不能够满足业务 需求。需要进行数据库实例的扩容。多台数据库同时相 应请求。也就是说需要对数据库的请求&#xff0c;进行负载均衡 但是由于数据库服务特殊原因&#xff0c;数据库…

安卓用户专属福利:OfficeSuite中文高级版,让你的工作更轻松!

OfficeSuite – 世界顶级移动办公软件&#xff01;Google Play商店下载最多的办公软件应用&#xff0c;迄今为止&#xff0c;智能手机平台上&#xff0c;功能最强大、兼容性最好的移动Office办公套件。创建&#xff0c;查看和编辑Word&#xff0c;Excel和PowerPoint文档&#x…

mysql主从数据库(5.7版本)与python的交互及mycat

mysql数据库基本操作&#xff1a; [rootm ~]# tar -xf mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz 解压压缩包 [rootm ~]# ls anaconda-ks.cfg mysql-5.7.44-linux-glibc2.12-x86_64 mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz [rootm ~]# cp -r mysql-5.7.44-linu…

Ubuntu 批量杀死进程

ps -ef|grep python|grep server|grep -v grep|cut -c 9-16|xargs kill -9这个命令序列是一个在Linux或类Unix系统中使用的脚本片段&#xff0c;用于批量终止&#xff08;强制杀死&#xff09;所有与特定条件&#xff08;这里是包含"python"和"wanghao"的&…

推荐浏览器爬虫插件:Instant Data Scraper 无需写一行代码

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

云计算29-------mysql主从数据库(5.7版本)与python的交互及mycat

mysql数据库基本操作&#xff1a; [rootm ~]# tar -xf mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz 解压压缩包 [rootm ~]# ls anaconda-ks.cfg mysql-5.7.44-linux-glibc2.12-x86_64 mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz [rootm ~]# cp -r mysql-5.7.44-lin…

如何判断树上一个点是否在直径上

# 旅游规划 ## 题目描述 W市的交通规划出现了重大问题&#xff0c;市政府下定决心在全市各大交通路口安排疏导员来疏导密集的车流。但由于人员不足&#xff0c;W市市长决定只在最需要安排人员的路口安排人员。 具体来说&#xff0c;W市的交通网络十分简单&#xff0c;由n个…

【Android Git】Mac配置支持 Gitlab、Gitee和阿里云效多平台

前言 在开发过程中&#xff0c;会遇到多平台项目管理问题&#xff0c;需要进行配置支持&#xff0c;常用的平台有Gitlab、Gitee、阿里云效等&#xff0c;本篇文章记录下使用同一邮箱配置支持的过程。 说明 首先认识下id_ras,一个用于SSH&#xff08;安全外壳协议&#xff09;…

java判断字符串某字符是否为大写/小写/数字?

Character类提供了很多静态方法&#xff0c;用于处理Unicode字符&#xff0c;如下&#xff1a; 也可以将字符转化成小写字母或大写字母。运用如下&#xff1a; 1是数字返回true a不是大写返回false a是小写返回true a转化成大写字母后为A A转化成小写字母后为a

备战金三银四、金九银十、软件测试面试问答

1、问&#xff1a;你在测试中发现了一个bug&#xff0c;但是开发经理认为这不是一个bug&#xff0c;你应该怎样解决&#xff1f; 首先&#xff0c;将问题提交到缺陷管理库里面进行备案。 然后&#xff0c;要获取判断的依据和标准&#xff1a; 根据需求说明书、产品说明、设计…

压缩软件里的文件名编码

由于默认编码环境不同&#xff0c;打包时正常的文件和目录&#xff0c;在解包时就是乱码了。就拿winrar来说&#xff0c;windows中文版下&#xff0c;默认的编码是GBK 你将一堆文件打包给mac用户或者linux用户&#xff0c;那边的默认编码是UTF8&#xff0c;解压出来文件内容没有…

前端进阶——浏览器篇

浏览器如何工作&#xff08;一&#xff09;进程架构 浏览器的工作过程复杂而高效&#xff0c;其核心在于其进程架构的设计。以下是对浏览器进程架构的详细解析&#xff1a; 一、浏览器的主要进程 现代浏览器大多采用多进程多线程的架构&#xff0c;以Chrome浏览器为例&…

你会读财务报表吗?快来看看如何正确解读

在现代商业的复杂网络中&#xff0c;每一家公司都像是一个精密运行的钟表&#xff0c;其运转的顺畅程度取决于各个齿轮的完美契合与精准配合。而财务报表&#xff0c;就像是是这钟表的指针&#xff0c;实时展现着公司运转的状态和效率&#xff0c;帮助管理者把握全局&#xff0…

用python实现视频中插入各种形式的文本,包括普通文本、数学公式、项目符号列表和标题

tex_mobject 模块提供了一系列可以使用 LaTeX 渲染文本的类。通过这个模块&#xff0c;你能够在视频中插入各种形式的文本&#xff0c;包括普通文本、数学公式、项目符号列表和标题等。具体类的功能如下&#xff1a; BulletedList&#xff1a;用于创建带项目符号的列表。MathT…

关于utf-8编码规范练习题

目录 一、代码内容 二、MySQL内容 三、代码遇到的问题 遭遇&#xff1a; 解决思路&#xff1a; 四、出现问题1 断点调试&#xff0c;分析问题 发现问题点&#xff1a; 问题解决 五、出现问题2 原因&#xff1a; 举例&#xff1a; 举例总结&#xff1a; 一、代码内容…

数学建模--浅谈多波束测线问题

目录 1.问题说明 2.问题分析 3.代码分析 1.问题说明 这个是国赛的真题&#xff0c;我们这个里面只是浅谈&#xff0c;就是对于这个里面运用的过程仿真的思路进行说明&#xff0c;这个探测的波束问题实际上也是一个简单的过程仿真问题&#xff0c;也是需要去进行作图的&#…

游戏管理系统

目录 Java程序设计课程设计 游戏管理系统 1系统简介 1.1需求分析 1.2 编程环境与工具 2系统总体设计 2.1 系统的功能模块图。 2.2 各功能模块简介。 3主要业务流程 &#xff08;1&#xff09;用户及管理员登录流程图 &#xff08;2&#xff09;信息添加流程 &#x…

Java语言程序设计——篇十三(3)

&#x1f33f;&#x1f33f;&#x1f33f;跟随博主脚步&#xff0c;从这里开始→博主主页&#x1f33f;&#x1f33f;&#x1f33f; 欢迎大家&#xff1a;这里是我的学习笔记、总结知识的地方&#xff0c;喜欢的话请三连&#xff0c;有问题可以私信&#x1f333;&#x1f333;&…

virtual_host.hpp模块

目录 一.VirtualHost虚拟机模块介绍 二.VirtualHost的实现 1. 类概述 2. 交换机操作 3. 队列操作 4. 绑定操作 5. 消息操作 6. 清理操作 总结 三.全部代码 一.VirtualHost虚拟机模块介绍 虚拟机是对之前几个数据管理模块的整合&#xff0c;并封装了之前的一些操作。…