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

施磊老师基于muduo网络库的集群聊天服务器(七)

文章目录

  • 数据表字符集问题
    • 支持中文和英文
      • **为什么使用 `utf8mb4`?**
    • 推荐 查看整个表, 再单独修改
  • 客户端群组功能
    • 创建群组
    • 添加群组
    • 群组聊天
    • 接收在线群组消息
    • 接收离线群组消息
    • 补充服务器事件处理器
    • 补充服务器查询群组列表
    • 问题解决
    • 测试
  • 目前报错总结
  • 目前为止最恶心的错误
  • 客户端用户注销功能
    • 1. 用户注销功能的设计与实现
    • 2. 客户端状态管理与循环控制
    • 3. 数据重复加载问题与解决方案
    • 4. 接收线程的单例控制-**重点**
    • 巧用静态局部变量
    • 老师这节课进行了一个小总结
    • 5. 架构设计思想
    • 6. 集群化扩展的铺垫
    • 错误
    • 代码

数据表字符集问题

支持中文和英文

为什么使用 utf8mb4

  1. 支持完整的 Unicode 字符
    • utf8mb4 可以存储几乎所有语言的字符,支持 中文日文韩文,以及 Emoji 和符号。
  2. utf8 更加可靠
    • utf8 只支持最多三个字节的字符(不支持一些 4 字节字符,如 Emoji),而 utf8mb4 支持最多四个字节的字符。

查看数据库 所有表的 字符集

SELECT TABLE_NAME,TABLE_COLLATION
FROM information_schema.TABLES
WHERE TABLE_SCHEMA = '你的数据库名';

单独修改某个表

ALTER TABLE 表名 CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;

批量修改字符集–修改某个字符集的所有表

SELECT CONCAT('ALTER TABLE ', TABLE_NAME, ' CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;') 
FROM information_schema.TABLES
WHERE TABLE_SCHEMA = '你的数据库名'AND TABLE_COLLATION LIKE 'latin1%';

推荐 查看整个表, 再单独修改

客户端群组功能

创建群组

// creategroup函数
void creategroup(int clientfd, string msg)
{int idx = msg.find(":"); // 查找第一个:的位置if (idx == string::npos) // 没有找到  ==-1->不建议用{cout << "creategroup command: group name is invalid!" << endl;return;}string groupname = msg.substr(0, idx);                        // 截取群组名称string groupdesc = msg.substr(idx + 1, msg.size() - idx - 1); // 截取群组描述json js;js["msgid"] = CREATE_GROUP_MSG;   // 创建群组消息js["id"] = g_currentUser.getId(); // 当前登录用户idjs["groupname"] = groupname;      // 群组名称js["groupdesc"] = groupdesc;      // 群组描述string request = js.dump();                                                // json转字符串  序列化int len = send(clientfd, request.c_str(), strlen(request.c_str()) + 1, 0); // 发送数据if (len < 0){cerr << "send creategroup msg error: " << request << endl;}
}

添加群组

// addgroup函数
void addgroup(int clientfd, string msg)
{int groupid = atoi(msg.c_str()); // 转成整型json js;js["msgid"] = ADD_GROUP_MSG;      // 添加群组消息js["id"] = g_currentUser.getId(); // 当前登录用户idjs["groupid"] = groupid;          // 群组idstring request = js.dump();                                                // json转字符串  序列化int len = send(clientfd, request.c_str(), strlen(request.c_str()) + 1, 0); // 发送数据if (len < 0){cerr << "send addgroup msg error: " << request << endl;}
}

群组聊天

// groupchat函数
void groupchat(int clientfd, string msg)
{int idx = msg.find(":"); // 查找第一个:的位置if (idx == string::npos) // 没有找到  ==-1->不建议用{cout << "groupchat command: group id is invalid!" << endl;return;}int groupid = atoi(msg.substr(0, idx).c_str());             // 截取群组idstring message = msg.substr(idx + 1, msg.size() - idx - 1); // 截取聊天信息json js;js["msgid"] = GROUP_CHAT_MSG;         // 群组聊天消息js["id"] = g_currentUser.getId();     // 当前登录用户idjs["name"] = g_currentUser.getName(); // 当前登录用户姓名js["groupid"] = groupid;              // 群组id  -- 字段要对应服务器那边的js["msg"] = message;                  // 聊天信息js["time"] = getCurrentTime();        // 时间// 发送聊天请求string request = js.dump();                                                // json转字符串  序列化int len = send(clientfd, request.c_str(), strlen(request.c_str()) + 1, 0); // 发送数据if (len < 0){cerr << "send groupchat msg error: " << request << endl;}
}

接收在线群组消息

区分群组消息和个人消息

readTaskHandler函数

// 解析json数据
json response = json::parse(buffer);   // 反序列化 字符串转json
if (response["msgid"] == ONE_CHAT_MSG) // 一对一聊天消息
{cout << response["time"].get<string>() << "[" << response["id"] << "] " << response["name"].get<string>() << " said: " << response["msg"].get<string>() << endl;continue;
}if (response["msgid"] == GROUP_CHAT_MSG) // 群组聊天消息
{cout << "群消息-->[" << response["groupid"] << "] " << response["time"].get<string>() << "[" << response["id"] << "] " << response["name"].get<string>() << " said: " << response["msg"].get<string>() << endl;
}

接收离线群组消息

同样进行区分

main

// 处理离线消息
if (response.contains("offlinemsg")) // 判断是否包含字段, 跟好点,  而不是看 是不是空
{vector<string> offlinemsg = response["offlinemsg"]; // 类型是vector<string>, 不是vector<User>,  根据服务器业务,存的是js.dump() 字符串for (auto &msg : offlinemsg){json js = json::parse(msg); // 反序列化// 时间+fromid+fromname+msg-----详看笔记 一对一聊天发送的格式// 分一下 个人离线和群组离线if (js["msgid"] == ONE_CHAT_MSG){cout << js["time"].get<string>() << "[" << js["id"].get<int>() << "] " << js["name"].get<string>() << " said: " << js["msg"].get<string>() << endl;}if (js["msgid"] == GROUP_CHAT_MSG) // 群组聊天消息{cout << "群消息-->[" << js["groupid"] << "] " << js["time"].get<string>() << "[" << js["id"] << "] " << js["name"].get<string>() << " said: " << js["msg"].get<string>() << endl;}}
}

补充服务器事件处理器

服务器业务—> 没有加 群组相关的 事件处理器

补充服务器查询群组列表

登录成功–查询群组列表

// 查询群组列表
vector<Group> groupVec = _groupModel.queryGroups(id);
if (!groupVec.empty())
{vector<string> vec;for (auto &groupl : groupVec){json js ;js["id"] = groupl.getId();js["groupname"] = groupl.getName();js["groupdesc"] = groupl.getDesc();vector<string> usersvec;for (auto &user : groupl.getUsers()){json js;js["id"]=user.getId();js["name"]=user.getName();js["state"]=user.getState();js["role"] = user.getRole();usersvec.push_back(js.dump());}js["users"] = usersvec;vec.push_back(js.dump());}response["groups"] = vec; // 群组列表
}

问题解决

服务器部分 groupmodel.cpp 有一些逻辑问题, 导致读不到用户, 提前返回了, 已进行修改

测试

自行测试— 至此,功能都正常

目前报错总结

目前了解的:

数据库不支持 中文-----英文正常

数据库本身问题----把语句 先在mysql命令行试一下, 看情况处理

json在gdb不能直接看----在代码上添加 临时量 存储 js.dump(), 进行查看

由于客户端 和 服务器 和 mysql分离, 所以 有时候 可以分步测试, 看mysql 有没有, 再去排除问题

目前为止最恶心的错误

terminate called after throwing an instance of 'nlohmann::json_abi_v3_12_0::detail::type_error'
what():  [json.exception.type_error.302] type must be string, but is null
Aborted (core dumped)

这个错误, 基本就是 json 变量名写错了

type must be string, but is null: 你试图像字符串一样使用一个字段(如 j["name"].get<std::string>()),但 j["name"] 实际是 null,无法转换成字符串。

下面这段, cout里面是 js. 不是response.

// 处理离线消息// 分一下 个人离线和群组离线
if (js["msgid"] == ONE_CHAT_MSG)
{cout << js["time"].get<string>() << "[" << js["id"].get<int>() << "] " << js["name"].get<string>() << " said: " << js["msg"].get<string>() << endl;
}
if (js["msgid"] == GROUP_CHAT_MSG) // 群组聊天消息
{cout << "群消息-->[" << js["groupid"] << "] " << js["time"].get<string>() << "[" << js["id"] << "] " << js["name"].get<string>() << " said: " << js["msg"].get<string>() << endl;
}

客户端用户注销功能

用户登出:

1. 结束mainMenu

2. 结束对应的 接受线程

1. 用户注销功能的设计与实现

区分了 quit–>正常注销 和 客户端异常–>ctrl+c

  • 正常注销 vs. 异常退出
    • 正常注销:客户端主动发送 LOGOUT 消息,服务端收到后:
      • connectionMap 移除该用户的连接(需加锁保证线程安全)。
      • 更新数据库,将该用户状态设为 offline
    • 异常退出(如客户端强制关闭):
      • 服务端检测到连接断开,触发 closeException 处理。
      • 由于无法直接获取用户ID,需遍历 connectionMap 比对连接来删除。
  • 关键点
    • 正常注销比异常退出更高效,因为能直接通过 userID 定位用户连接。
    • 服务端需要保证对共享资源(如 connectionMap)的线程安全操作。

2. 客户端状态管理与循环控制

我们希望, 在正常退出后, 会回到主菜单页面!

之前的代码, 在登录后会进入 菜单, 进入死循环, 可以一直 发送消息

  • 全局变量控制页面跳转
    • 引入 isMainMenuRunning(布尔值)控制主菜单循环:
      • 登录成功 → true(进入主菜单循环)。
      • 注销 → false(退出循环,返回登录页面)。
    • 避免因循环阻塞导致无法返回登录界面。
  • 关键点
    • 通过状态变量而非死循环控制流程,使代码更清晰、可维护。
    • 注销后需重置用户数据(如好友列表、群组列表),防止重复加载。

3. 数据重复加载问题与解决方案

  • 问题
    • 用户注销后重新登录时,未清空旧数据,导致好友/群组列表重复累积。
  • 解决方案
    • 每次登录时先清空容器(如 friendList.clear()),再加载新数据。
    • 确保数据唯一性,避免重复存储。

4. 接收线程的单例控制-重点

  • 问题
    • 多次登录会重复启动接收线程,导致多个线程同时监听消息,引发混乱。
  • 解决方案
    • 使用静态变量确保线程只启动一次:
      • 首次登录时启动线程,后续登录直接复用。
      • 避免资源浪费和线程竞争。

巧用静态局部变量

静态局部变量(static 局部变量)是 C/C++ 中的一种局部变量,其特点是:

  1. 作用域:与普通的局部变量一样,静态局部变量的作用域仅限于函数内部。它只能在声明它的函数内使用,外部无法访问。
  2. 生命周期:与普通的局部变量不同,静态局部变量的生命周期是整个程序运行期间。它在程序开始时就被分配内存,并在程序结束时销毁,而普通局部变量则是在函数调用时创建,在函数调用结束时销毁。
  3. 初始化:静态局部变量只会在第一次进入该函数时进行初始化,之后不会再次初始化。如果没有显式初始化,静态局部变量会被默认初始化为零(对于基本数据类型)。
#include <iostream>void counterFunction() {static int counter = 0; // 静态局部变量counter++;std::cout << "Counter: " << counter << std::endl;
}int main() {counterFunction(); // 输出 Counter: 1counterFunction(); // 输出 Counter: 2counterFunction(); // 输出 Counter: 3return 0;
}

老师这节课进行了一个小总结

5. 架构设计思想

  • 分层解耦
    • 网络层:仅负责数据传输(如收发JSON消息)。
    • 业务层:处理具体逻辑(如注销、更新状态),不直接操作数据库。
    • 数据层:通过ORM(如UserGroup类)封装数据库操作,业务层无需关心SQL细节。
  • 关键优势
    • 代码可维护性高,各层职责清晰。
    • 扩展性强(如未来支持集群化,只需修改服务端,客户端无感知)。

6. 集群化扩展的铺垫

  • 客户端无感知
    • 无论服务端是单机还是集群,客户端只需发送请求,不关心后端如何负载均衡。
  • 服务端集群化
    • 后续可通过Nginx实现负载均衡,水平扩展服务器性能。在很短的时间内 提升服务器并发能力
    • 需解决共享状态问题(如用户连接信息需集中管理,如用Redis)。

错误

  1. main 的 for循环 最后 不要return, 不然 注销后, 主线程就结束了

  2. 老师的 课中, 出现正常注销后, 再次登录, 会打印两边信息, 是因为 那几个变量是 全局的, 每次重新登录后变量没有清空, 之前的信息还在

    g_currentUserGroupsList.clear();
    // vector.clear()清空一下
    

代码

include/public.hpp

LOGINOUT_MSG, // 登录成功

include/server/chatservice.hpp

// 处理注销业务void loginout(const TcpConnectionPtr &conn, json &js, Timestamp time);

src/server/chatservice.cpp

_msghandlermap.insert({LOGINOUT_MSG, std::bind(&ChatService::loginout, this, _1, _2, _3)});// 处理注销业务
void ChatService::loginout(const TcpConnectionPtr &conn, json &js, Timestamp time)
{int userid = js["id"].get<int>();{lock_guard<mutex> lock(_connMutex);auto it = _userConnMap.find(userid);if(it != _userConnMap.end()){_userConnMap.erase(it);}}// 更新用户状态User user(userid, "","", "offline");// user.setId(userid);// user.setState("offline");_usermodel.updateState(user); // 仅需要id和状态, 剩下的具体 由函数在数据库完成 
}

src/client/main.cpp

// 控制聊天页面--注销需要退出聊天页面
bool isMainMenuRunning = false;
// 登录成功, 启动接收线程----只要客户端 不完全退出, 就只启动一次!
static int threadnum = 0;
if (threadnum == 0)
{std::thread readTask(readTaskHandler, clientfd); // thread 支持跨平台readTask.detach();
}// 分离线程, 让其独立运行, 不阻塞主线程// 主线程继续执行, 进入聊天菜单页面
isMainMenuRunning = true;
mainMenu(clientfd);
// 删除 return 0;
// 主页面聊天程序
void mainMenu(int clientfd)
{help();// for (;;)while (isMainMenuRunning){...}
}
// quit函数
void quit(int clientfd, string msg)
{json js;js["msgid"] = LOGINOUT_MSG;       // 注销消息js["id"] = g_currentUser.getId(); // 当前登录用户idstring request = js.dump();                                                // json转字符串  序列化int len = send(clientfd, request.c_str(), strlen(request.c_str()) + 1, 0); // 发送数据if (len < 0){cerr << "send quit msg error: " << request << endl;}isMainMenuRunning = false; // 退出聊天页面// close(clientfd);  放在服务端处理// exit(0);
}

相关文章:

施磊老师基于muduo网络库的集群聊天服务器(七)

文章目录 数据表字符集问题支持中文和英文**为什么使用 utf8mb4&#xff1f;** 推荐 查看整个表, 再单独修改 客户端群组功能创建群组添加群组群组聊天接收在线群组消息接收离线群组消息补充服务器事件处理器补充服务器查询群组列表问题解决测试 目前报错总结目前为止最恶心的错…...

多模态(3):实战 GPT-4o 视频理解

最近&#xff0c;OpenAI 团队的 GPT-4o 模型&#xff0c;在多模态方面的能力有了大幅提升&#xff0c;这次我们就使用 GPT-4o 完成一个视频理解的实战。 1. 环境搭建 1.1 安装 FFmpeg 做视频处理&#xff0c;我们需要用到 FFmpeg 这款功能强大的开源多媒体处理工具。FFmpeg…...

基于python实现一个二维图片的路径规划问题

一、场景 基于如下的一个楼层平面图&#xff0c;假设有几个预置的点&#xff08;实际项目中可能是动态的点&#xff0c;比如找车位&#xff0c;找工位&#xff09;&#xff0c;做路径规划&#xff0c;并画在平面图上 二、方案 1.准备平面室内图 可以自己用QGIS/cad等其他方式…...

云服务器centos 安装hadoop集群

百度 搜索 云服务器centos 安装hadoop 创建Hadoop用户 sudo useradd hadoop -m -s /bin/bash sudo passwd hadoop 123456 下载Hadoop wget https://mirrors.tuna.tsinghua.edu.cn/apache/hadoop/common/hadoop-3.2.4/hadoop-3.2.4.tar.gz 解压并移动Hadoop到指定目录 tar …...

【k8s】sidecar边车容器

一、Sidecar 模式简介 Sidecar 模式是一种常见的微服务架构设计模式。它通过将附加功能或服务与主应用程序部署在同一容器或主机上&#xff0c;从而实现对主应用程序的增强和扩展。Sidecar 的名称来源于摩托车的边车&#xff0c;它与摩托车紧密相连&#xff0c;为主车提供额外…...

Web漏洞--XSS之订单系统和Shell箱子

本文主要内容 手法 XSS平台使用 XSS工具使用 XSS结合其他漏洞 XSS具体使用场景 某订单系统XSS盲打_平台 某Shell箱子系统XSS盲打_工具 [1]订单系统经典案例 第一个简易攻击流程&#xff08;订单系统&#xff09;&#xff1a;通过平台完成XSS跨站之后&a…...

# 构建词汇表:自然语言处理中的关键步骤

构建词汇表&#xff1a;自然语言处理中的关键步骤 在自然语言处理&#xff08;NLP&#xff09;任务中&#xff0c;词汇表&#xff08;Vocabulary&#xff09;是文本数据预处理的核心组件之一。它将文本中的单词或字符映射为数值索引&#xff0c;从而让计算机能够理解和处理语言…...

新!在 podman-machine-default 中安装 CUDA、cuDNN、Anaconda、PyTorch 等并验证安装

#工作记录 一、前言 在 Windows 系统开发环境中&#xff0c;Podman Desktop 凭借强大的容器管理与 WSL-Linux 子系统集成能力备受开发者关注。 其中&#xff0c;podman-machine-default 是 Podman Desktop 安装后自带的默认 WSL-Fedora 子系统&#xff0c;支持与显卡通信&am…...

python_BeautifulSoup提取html中的信息

目录 描述&#xff1a; 过程&#xff1a; step one 下载html网页到本地 step two 提取html信息 list_con soup.select(.list-con) [0] li_list list_con.find_all(li) a li.find(span).find(a) title a.get(title) url a.get(href) span li.find(span).find(spa…...

pcd2pgm的launch文件实现

1.新建工作空间和克隆代码 mkdir -p pcd2pgm_launch/src && cd pcd2pgm_launch/src git clone https://github.com/Hinson-A/pcd2pgm_package 2. 编译 cd .. catkin_make -j4 3.修改launch 在launch文件目录下&#xff0c;可以用gedit 打开launch文件&#xff0c…...

Vue里面elementUi-aside 和el-main不垂直排列

先说解决方法 main.js少导包 import element-ui/lib/theme-chalk/index.css; //加入此行即可 问题复现 排查了一个小时终于找出来问题了&#xff0c;建议导包去看官方的文档&#xff0c;作者就是因为看了别人的导包流程导致的问题 导包官网地址Element UI导包快速入门...

论文阅读:2024 ACL ArtPrompt: ASCII Art-based Jailbreak Attacks against Aligned LLMs

总目录 大模型安全相关研究&#xff1a;https://blog.csdn.net/WhiffeYF/article/details/142132328 Artprompt: Ascii art-based jailbreak attacks against aligned llms https://www.doubao.com/chat/3846685176618754 https://arxiv.org/pdf/2402.11753 https://github…...

项目maven版本不一致 导致无法下载

路程&#xff1a;打开一个新项目发现&#xff0c;maven加载不了 报错&#xff1a; Error running ‘dataManage [clean]’ No valid Maven installation found. Either set the home directory in the configuration dialog or set the M2_HOME environment variable on your s…...

论文阅读:2024 NeurIPS Group Robust Preference Optimization in Reward-free RLHF

Group Robust Preference Optimization in Reward-free RLHF https://www.doubao.com/chat/3870738843518978 https://arxiv.org/pdf/2405.20304 速览 研究动机 传统RLHF忽视群体偏好差异&#xff0c;导致模型对少数群体表现不佳&#xff0c;需提升群体鲁棒性。研究问题 如…...

数据可视化平台产品介绍及功能特色

数据可视化平台是一款适用于高校教学和各领域企业的零门槛可视化工具&#xff0c;能够解决高校数据分析与可视化类课程教学、实训问题。平台通过浏览器即可访问&#xff0c;无需安装客户端。平台内置公式编辑器与指标构建器&#xff0c;学生可通过四则运算、分组聚合等方式衍生…...

MySQL索引优化、SQL分析与运行原理 - Java架构师面试实战

MySQL索引优化、SQL分析与运行原理 - Java架构师面试实战 第一轮提问 面试官&#xff1a;马架构&#xff0c;请问您对MySQL的B树索引有什么理解&#xff1f; 马架构&#xff1a;B树是一种平衡多路查找树&#xff0c;所有的数据节点都存储在叶子节点上。相比于B树&#xff0c…...

C++学习:六个月从基础到就业——STL:函数对象与适配器

C学习&#xff1a;六个月从基础到就业——STL&#xff1a;函数对象与适配器 本文是我C学习之旅系列的第二十九篇技术文章&#xff0c;也是第二阶段"C进阶特性"的第八篇&#xff0c;主要介绍C STL中的函数对象与适配器。查看完整系列目录了解更多内容。 引言 在前面的…...

Linux基础篇、第四章_02磁盘及分区管理fdisk 和 gdisk

题目&#xff1a;Linux 磁盘及分区管理 版本号: 1.0,0 作者: 老王要学习 日期: 2025.04.25 适用环境: Centos7 文档说明 本教程适用于 Centos7 环境&#xff0c;详细介绍 Linux 磁盘及分区管理操作。包含虚拟机添加磁盘的关机与开机添加方法、MBR 和 GPT 两种分区方式特点、…...

火山云的市场竞争

火山云是字节跳动旗下的云计算服务&#xff0c;对吧&#xff1f;那它的竞争对手应该包括国内外的大型云服务提供商。首先&#xff0c;国际市场上&#xff0c;像AWS、Azure、Google Cloud这些巨头肯定是大头。国内的话&#xff0c;阿里云、腾讯云、华为云这些应该都是主要的竞争…...

创建型设计模式之:简单工厂模式、工厂方法模式、抽象工厂模式、建造者模式和原型模式

创建型设计模式之&#xff1a;简单工厂模式、工厂方法模式、抽象工厂模式、建造者模式和原型模式 &#xff08;一&#xff09;简单工厂模式 简单工厂模式将对象的实例化过程封装到一个工厂类中&#xff0c;根据输入的条件创建不同类型的对象。 角色划分&#xff1a; 抽象产品…...

【Linux内核设计与实现】第三章——进程管理01

文章目录 1. 引言2. 进程&线程——概念3. 进程控制块/进程描述符(PCB)4. 进程内核栈&#xff08;Kernel Stack&#xff09;4.1. 进程内核栈的定义4.2. thread_info 体系结构相关进程描述4.3. 定位进程描述符(task_struct)和内核栈以及内核栈指针的问题 5. 进程 ID&#xff…...

正大模型视角下的市场结构判断逻辑

正大模型视角下的市场结构判断逻辑 在多数交易策略中&#xff0c;结构识别往往先于方向判断。以正大的数据研判风格为例&#xff0c;其核心逻辑是&#xff1a;价格行为不能孤立解读&#xff0c;必须结合时间与成交效率来判断当前结构的有效性。 例如&#xff0c;一个上涨过程&…...

4.25学习——文件上传之00截断

继昨天学习的基础文件上传内容&#xff0c;进一步学习文件上传的绕过方式 00截断绕过 原理&#xff1a;00截断是操作系统层的漏洞&#xff0c;由于操作系统是C语言或汇编语言编写的&#xff0c;这两种语言在定义字符串时&#xff0c;都是以\0&#xff08;即0x00&#xff09;作…...

黑马Redis(三)黑马点评项目

优惠卷秒杀 一、全局唯一ID 基于Redis实现全局唯一ID的策略&#xff1a; Component RequiredArgsConstructor public class RedisIdWorker {private static final Long BEGIN_TIMESTAMP1713916800L;private static final int COUNT_BITS 32;Resourceprivate final StringRed…...

dedecms织梦arclist标签noflag属性过滤多个参数

织梦dedecms系统arclist标签noflag属性默认是只能过滤一个参数&#xff0c;比如过滤推荐是noflagc&#xff0c;过滤有图片的文章是noflagc&#xff0c;在模板制作过程中&#xff0c;有时候我们为了seo和避免重复&#xff0c;需要过滤多个参数。今天小编就来跟大家讲讲织梦dedec…...

Jira、PingCode、Redmine等18款缺陷管理工具对比评测

本文主要介绍了以下&#xff1a;1. PingCode; 2. Worktile; 3. Jira; 4. Bugzilla; 5. TAPD; 6. 码云; 7. Redmine; 8. Trac; 9. 蓝鲸智云; 10. 阿里云效等等18款缺陷管理工具。 在现代软件开发和项目管理中&#xff0c;缺陷管理工具扮演着至关重要的角色。随着企业对软件质量的…...

京东以图搜图(拍立淘)API接口返回参数详解

京东以图搜图&#xff08;拍立淘&#xff09;API接口的返回参数通常以结构化JSON格式呈现&#xff0c;涵盖商品基础信息、相似度评分、库存状态及扩展字段&#xff0c;以下为关键参数详解及使用建议&#xff1a; 一、核心返回参数解析 状态标识类 status&#xff1a;请求状态…...

LSTM+KNN - 多元数据异常检测 !

大家好!我是我不是小 upper~ 今天想和大家分享一个超实用的案例:如何通过 LSTM 与 KNN 实现多元数据异常检测。 想象一下,在工厂的智能化监控场景中,各类传感器实时采集着温度、湿度、压力等海量数据。我们的目标,就是从中精准识别出设备潜在故障等异常情况。 LSTM 作为时…...

OpenHarmony之电源管理子系统公共事件定义

OpenHarmony之电源管理子系统公共事件定义 电源管理子系统面向应用发布如下系统公共事件&#xff0c;应用如需订阅系统公共事件&#xff0c;请参考公共事件接口文档。 COMMON_EVENT_BATTERY_CHANGED 表示电池充电状态、电平和其他信息发生变化的公共事件的动作。 值&#x…...

angular 实现可编辑可选择复制的表格

这个实现的核心就是ag-grid 当然有类似的库就不必多说&#xff0c;React, Vue和纯h5类似。简单贴一下代码 1.首先是h5部分&#xff0c;就一个id为supply-chain-material-grid-table的div&#xff0c;记住要设置高度 <div class"dki-supply-chain-page-body">…...

组织用户数统计实现

# 完整的组织用户数统计实现 完整的组织用户数统计实现&#xff0c;包括模拟SQL查询、完整的Java代码实现以及详细解释。 ## 1. 模拟SQL查询 假设我们有一个组织表(organization)和用户表(user)&#xff0c;以下是模拟查询SQL&#xff1a; sql -- 获取各组织及其用户数量&a…...

天机学堂day10作业,完善兑换优惠券功能

UserCouponServiceImpl /*** 兑换码兑换优惠券* param code*/TransactionalOverridepublic void exchangeCoupon(String code) {//1、校验code是否为空if (StringUtils.isBlank(code)) {throw new BadRequestException("非法参数&#xff01;");}//2、解析兑换码&…...

Python编程的真谛:超越语法,理解编程本质

你是否也曾陷入这样的误区&#xff1a;学了无数的 Python 语法、刷了几十套题&#xff0c;写起代码却仍然卡顿、举步维艰&#xff1f;这时候你才发现&#xff0c;真正阻碍进步的&#xff0c;从不是语法&#xff0c;而是你对“编程本质”的理解。 如果你只是死记硬背Python的语…...

C语言 函数补充

目录 static和extern函数 1.static和extern函数 static和extern都是C语言中的关键字 static 是 静态的 的意思&#xff0c;可以用来: - 修饰局部变量- 修饰全局变量- 修饰函数 extern 是用来声明外部符号的。 在讲解 static 和 extern 之前再讲一下: 作用域和生命周期。 …...

【AI图像创作变现】04实操路径—插图/绘本/创意图集

引言 如果说头像是“一个角色的起点”&#xff0c;那么插图、绘本和图集就是“这个角色能走多远”。相比于头像这种单图任务&#xff0c;插图类创作更强调批量性、叙事性与风格统一性&#xff0c;它既可以承载故事&#xff0c;也可以构成一套完整的内容产品结构。 这类任务特…...

Lesar: 面向 Lustre/Scade 语言的形式化模型检查工具

在《同步反应式系统》的第一课中&#xff0c;介绍了同步数据流语言 Lustre 生态中的形式化模型检查器 Lesar 的用法。Lesar 可对 lustre v4 语言以及 Scade 语言中部分数据流核心特性进行模型检查。 Lesar 介绍 Lesar 是 Verimag 研发维护的形式化方法模型检查工具。该工具的理…...

告别 “幻觉” 回答:RAG 中知识库与生成模型的 7 种对齐策略

一、引言 大语言模型&#xff08;LLM&#xff09;在文本生成领域展现出惊人能力&#xff0c;但 “幻觉” 问题&#xff08;生成虚构或偏离事实的内容&#xff09;始终是落地应用的核心挑战。检索增强生成&#xff08;RAG&#xff09;通过将外部知识库与 LLM 结合&#xff0c;形…...

【Web应用服务器_Tomcat】一、Tomcat基础与核心功能详解

在 Java Web 应用开发领域&#xff0c;Apache Tomcat 是一座不可或缺的基石。作为一款开源、轻量级的 Servlet 容器和 Web 服务器&#xff0c;Tomcat 以其稳定可靠、易于部署和高度可定制性&#xff0c;被广泛应用于各类 Web 应用的部署与运行。 一、Tomcat 简介​ Tomcat 是…...

Cesium实现地形可视域分析

Cesium实现可视化分析 一、地形可视域主要实现技术(Ray + 地形碰撞检测) Cesium 本身的 Ray 类可以用来执行非常精确的射线检测,我们可以结合地形高度(sample)来逐点检测光线是否与 terrain 相交,从而判断是否可见。 1.1 优势 实时判断每条射线是否被 terrain 遮挡地形…...

Java—— 常见API介绍 第五期

JDK8以后新增的时间相关类 Date类ZoneId&#xff1a;时区Instant&#xff1a;时间戳ZoneDateTime&#xff1a;带时区的时间 日期格式化类 SimpleDateFormat DateTimeFormatter&#xff1a;用于时间的格式化和解析 日历类 Calendar LocalDate&#xff1a;年、月、日LocalTime…...

ViewPager FragmentPagerAdapter在系统杀死应用后重建时UI不刷新的问题

解决方案 通过重写getItemId方法&#xff0c;返回Fragment的hashCode&#xff1a; Override public long getItemId(int position) {/*** 恢复状态重建时&#xff0c;新的 Fragment 不刷新UI。* 原因&#xff1a;instantiateItem 中通过 mFragmentManager.findFragmentByTag(…...

第3讲、大模型如何理解和表示单词:词嵌入向量原理详解

1. 引言 大型语言模型&#xff08;Large Language Models&#xff0c;简称LLM&#xff09;如GPT-4、Claude和LLaMA等近年来取得了突破性进展&#xff0c;能够生成流畅自然的文本、回答复杂问题、甚至编写代码。但这些模型究竟是如何理解人类语言的&#xff1f;它们如何表示和处…...

关于STM32f1新建工程

创建文件夹 首先创建一个存放工程的文件夹&#xff0c;建议建立在D&#xff0c;E盘 新建工程 在kiel5里面 找到刚刚建立的文件夹&#xff0c;然后在此文件夹里面新建一个文件夹用来存放本次工程&#xff0c;文件夹可以根据工程内容所编写&#xff0c;然后给自己工程也就是…...

Linux:进程间通信---匿名管道

文章目录 1. 进程间通信1.1 什么是进程间通信&#xff1f;1.2 为什么进程要进行进程间通信&#xff1f;1.3 怎么实现进程间通信&#xff1f; 2. 匿名管道2.1 匿名管道的原理2.2 匿名管道的系统接口2.3 匿名管道的使用2.4 匿名管道的运用场景 序&#xff1a;在上一篇文章中我们知…...

python代做推荐系统深度学习知识图谱c#代码代编神经网络算法创新

以下是针对推荐系统、深度学习、知识图谱和神经网络算法创新的代码框架及开发建议&#xff0c;适用于C#和Python的跨语言协作项目。以下内容分为几个部分&#xff0c;涵盖技术选型、代码示例和创新方向。 1. 推荐系统&#xff08;Python C#&#xff09; Python部分&#xff0…...

【动手学大模型开发】VSCode 连接远程服务器

Visual Studio Code&#xff08;VSCode&#xff09;是一款由微软开发的免费、开源的现代化代码编辑器。它以其轻量级、高性能和广泛的编程语言支持而受到开发者的青睐。VSCode 的核心特点包括&#xff1a; 跨平台&#xff1a;支持 Windows、macOS 和 Linux 操作系统。扩展市场…...

PostgreSQL 漏洞信息详解

PostgreSQL 漏洞信息详解 PostgreSQL 作为一款开源关系型数据库&#xff0c;其安全漏洞会被社区及时发现和修复。以下是 PostgreSQL 漏洞相关的重要信息和资源。 一、主要漏洞信息来源 1. 官方安全公告 PostgreSQL 安全信息页面&#xff1a;https://www.postgresql.org/sup…...

华为L410上制作内网镜像模板:在客户端配置模板内容

华为L410上制作内网镜像模板&#xff1a;在客户端配置模板内容 在本教程中&#xff0c;我们将继续在华为L410上配置内网镜像模板&#xff0c;具体介绍如何在客户端设置以便于在首次开机时自动安装软件。我们将主要使用WeChat作为示例。 1. 制作镜像模板&#xff0c;开启 rc.l…...

分布式队列对消息语义的处理

在分布式系统中&#xff0c;消息的处理语义&#xff08;Message Processing Semantics&#xff09;是确保系统可靠性和一致性的关键。有三种语义&#xff1a; 在分布式系统中&#xff0c;消息的处理语义&#xff08;Message Processing Semantics&#xff09;是确保系统可靠性和…...

《免费开放”双刃剑:字节跳动Coze如何撬动AI生态霸权与暗涌危机?》

战略动机分析 降低技术门槛为数据采集接口 Coze平台宣称**“30秒无代码生成AI Bot”&#xff0c;大幅降低了企业开发AI应用的技术门槛。任何不懂编程的业务人员都可以通过可视化流程和提示词&#xff0c;在半分钟内搭建聊天机器人或智能代理。这种极低门槛意味着更多企业和个人…...