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

计算机网络之TCP和UDP的底层机制

文章目录1. TCP和UDP区别2.TCP为什么可靠传输3. 怎么用UDP实现HTTP4. TCP粘包怎么解决5. 滑动窗口6. 拥塞控制1. TCP和UDP区别TCP报头TCP发送数据客户端#includeiostream#includestring#includecstring#includesys/socket.h#includearpa/inet.h#includeunistd.hintmain(){// 1. 创建 TCP Socket (SOCK_STREAM 代表面向字节流的 TCP)intclient_fdsocket(AF_INET,SOCK_STREAM,0);if(client_fd-1){std::cerrTCP Socket 创建失败std::endl;return-1;}// 2. 配置服务端的 IP 和端口structsockaddr_inserver_addr;memset(server_addr,0,sizeof(server_addr));server_addr.sin_familyAF_INET;server_addr.sin_porthtons(8080);// 目标端口inet_pton(AF_INET,127.0.0.1,server_addr.sin_addr);// 目标 IP// 3. 发起连接 (此处会阻塞底层自动完成 TCP 三次握手)if(connect(client_fd,(structsockaddr*)server_addr,sizeof(server_addr))-1){std::cerrTCP 连接失败std::endl;close(client_fd);return-1;}std::coutTCP 三次握手成功连接已建立std::endl;// 4. 发送业务数据 (连接建立后只需关注发送的数据本身)std::string msgHello, TCP Server!;send(client_fd,msg.c_str(),msg.length(),0);// 5. 业务结束关闭连接 (底层触发四次挥手的第一步发送 FIN)close(client_fd);return0;}服务端监听socket只负责迎客通信socket才负责收发数据。前置准备socket函数创建监听socket-bind绑定端口-listen开始监听。迎客阻塞等待调用accept函数当有客户端完成三次握手连上来时accept函数会返回一个全新的socket文件描述符通信fd。接收数据针对那个新的通信fd调用recv函数或read函数来读取字节流数据。#includeiostream#includestring#includecstring#includesys/socket.h#includearpa/inet.h#includeunistd.hintmain(){// 1. 创建 TCP Socket (SOCK_STREAM 代表面向字节流、可靠的 TCP)intlisten_fdsocket(AF_INET,SOCK_STREAM,0);if(listen_fd-1){std::cerrTCP Socket 创建失败std::endl;return-1;}// 2. 配置服务端要绑定的 IP 和端口structsockaddr_inserver_addr;memset(server_addr,0,sizeof(server_addr));server_addr.sin_familyAF_INET;server_addr.sin_porthtons(8888);// 监听 8888 端口server_addr.sin_addr.s_addrhtonl(INADDR_ANY);// 监听本机所有网卡// 3. 绑定 (Bind)把 Socket 固定在 8888 端口上if(bind(listen_fd,(structsockaddr*)server_addr,sizeof(server_addr))-1){std::cerr端口绑定失败std::endl;close(listen_fd);return-1;}// 4. 监听 (Listen)告诉内核这个 Socket 进入“被动等待”状态准备迎接客人// 第二个参数 128 是“积压队列”的长度表示允许多少个连接在排队等待 acceptif(listen(listen_fd,128)-1){std::cerr监听失败std::endl;close(listen_fd);return-1;}std::coutTCP 服务端启动成功正在 8888 端口等待连接...std::endl;// --- 开始处理连接和接收数据 ---while(true){structsockaddr_inclient_addr;socklen_t client_lensizeof(client_addr);// 5. 接受连接 (Accept)这是 TCP 特有的步骤// accept 会阻塞直到有客户端完成“三次握手”。// 它会返回一个全新的 Socket FD (conn_fd)专门用于和这一个客户端说话。intconn_fdaccept(listen_fd,(structsockaddr*)client_addr,client_len);if(conn_fd-1){std::cerr接收连接出错std::endl;continue;}std::cout新客户端已连上IP: inet_ntoa(client_addr.sin_addr), 端口: ntohs(client_addr.sin_port)std::endl;charbuffer[1024];while(true){// 针对这一个连接循环读取数据memset(buffer,0,sizeof(buffer));// 6. 接收数据 (Recv)使用专属的 conn_fd而不是监听的 listen_fdssize_t bytes_readrecv(conn_fd,buffer,sizeof(buffer)-1,0);if(bytes_read0){std::cout收到数据: bufferstd::endl;// 回复客户端send(conn_fd,OK,2,0);}elseif(bytes_read0){// recv 返回 0代表对方关闭了连接发了 FINstd::cout客户端已断开连接。std::endl;break;}else{std::cerr读取出错std::endl;break;}}// 7. 关闭与该客户端的连接close(conn_fd);}// 8. 关闭监听 Socketclose(listen_fd);return0;}UDPUDP报头UDP发送数据客户端#includeiostream#includestring#includecstring#includesys/socket.h#includearpa/inet.h#includeunistd.hintmain(){// 1. 创建 UDP Socket (SOCK_DGRAM 代表面向报文的 UDP)intclient_fdsocket(AF_INET,SOCK_DGRAM,0);if(client_fd-1){std::cerrUDP Socket 创建失败std::endl;return-1;}// 2. 配置服务端的 IP 和端口structsockaddr_inserver_addr;memset(server_addr,0,sizeof(server_addr));server_addr.sin_familyAF_INET;server_addr.sin_porthtons(8080);// 目标端口inet_pton(AF_INET,127.0.0.1,server_addr.sin_addr);// 目标 IP// 3. 直接发送数据 (没有握手过程拿着地址直接发)std::string msgHello, UDP Server!;// 注意每次发送都需要带上目标地址 server_addrssize_t sent_bytessendto(client_fd,msg.c_str(),msg.length(),0,(structsockaddr*)server_addr,sizeof(server_addr));if(sent_bytes-1){std::cerrUDP 数据发送失败std::endl;}else{std::coutUDP 数据包已投递进网络层不保证对方一定收到。std::endl;}// 4. 关闭 Socket (本地资源回收没有任何网络交互报文)close(client_fd);return0;}服务端UDP服务端的流程极其简单粗暴没有握手没有Listen函数也没有ACCEPT函数。前置准备socket函数创建socketbind函数绑定本机的端口。直接接收调用函数recvfrom函数。直谁给这个端口发送数据他就收谁的。由于没有连接状态recvfrom函数必须提供一个空的地址结构体用来让内核把发送的IP和端口填进去否则服务端连接该把回报发给谁。都不知道。#includeiostream#includestring#includecstring#includesys/socket.h#includearpa/inet.h#includeunistd.hintmain(){// 1. 创建 UDP Socket (SOCK_DGRAM 代表不可靠的、面向报文的 UDP)intudp_fdsocket(AF_INET,SOCK_DGRAM,0);if(udp_fd-1){std::cerrUDP Socket 创建失败std::endl;return-1;}// 2. 配置服务端要监听的 IP 和端口structsockaddr_inserver_addr;memset(server_addr,0,sizeof(server_addr));server_addr.sin_familyAF_INET;server_addr.sin_porthtons(8080);// 使用 htons 转换端口号为网络字节序// 注意这里INADDR_ANY 表示监听本机的所有网卡 IP。// 假设你的服务器有内网 IP 和外网 IP设为 INADDR_ANY 意味着无论数据包从哪个网卡进来只要端口是 8080 都能收到。server_addr.sin_addr.s_addrhtonl(INADDR_ANY);// 3. 绑定 (Bind)把这个 Socket 和刚才配置的 IP 端口“死死绑定”在一起if(bind(udp_fd,(structsockaddr*)server_addr,sizeof(server_addr))-1){std::cerr端口绑定失败可能端口 8080 已被占用std::endl;close(udp_fd);return-1;}std::coutUDP 服务端启动成功正在监听 8080 端口...std::endl;// --- 以下是你之前看到的接收逻辑 ---charbuffer[1024]{0};structsockaddr_insender_addr;// 用来存放发件人的地址信息socklen_t sender_lensizeof(sender_addr);while(true){// 服务端通常是一个死循环不断接收数据memset(buffer,0,sizeof(buffer));// 每次接收前清空缓冲区// 4. 接收数据 (阻塞等待)ssize_t bytes_receivedrecvfrom(udp_fd,buffer,sizeof(buffer)-1,0,(structsockaddr*)sender_addr,sender_len);if(bytes_received0){// 利用 inet_ntoa 和 ntohs 把对方的网络字节序 IP 和端口转回我们能看懂的格式std::cout收到来自 [inet_ntoa(sender_addr.sin_addr):ntohs(sender_addr.sin_port)] 的消息: bufferstd::endl;// 如果需要回包直接用 sendto// std::string reply 服务端已收到;// sendto(udp_fd, reply.c_str(), reply.length(), 0, (struct sockaddr*)sender_addr, sender_len);}else{std::cerr接收数据出错std::endl;}}// 5. 关闭 Socket (实际上死循环服务端很少走到这里通常通过捕获终止信号来关闭)close(udp_fd);return0;}区别连接TCP是面向连接的传输层协议传输数据前先要建立连接。 UDP是不需要连接即刻传输数据。服务对象TCP是一对一的两点服务即一条连接只有两个端点UDP支持一对一一对多多对多的交互通信。可靠性TCP是可靠交付数据的数据可以无差错不丢失、不重复按序到达。UDP是不可靠的传输协议不保证可靠交付数据发送数据丢了就丢了不会有任何措施。但是我们可以基于UDP传输协议实现一个可靠的传输协议比如QUIC协议。拥塞控制流量控制TCP有拥塞控制和流量控制机制保证数据传输的安全性。UDP则没有即使网络非常拥堵了也不会影响UDP发送速率。首部开销TCP首部长度较长会有一定的开销首部在没有使用选项字段时是20个字节如果使用了选项字段则会变得更长。UDP首部只有八个字节并且是固定不变的开销较小。传输方式TCP是流式传输没有边界但保证顺序和可靠。UDP是一个包一个包的发送是有边界的但可能会丢包和乱序。2.TCP为什么可靠传输TCP主要通过以下几点来保证传输可靠性连接管理、序列号、确认序列号、超时重传、流量控制、拥塞控制。连接管理即通过三次握手和四次挥手确保连接可靠性。这是保证可靠传输的前提。序列号TCP将每个字节数据都进行编号。序列号具体作用如下能保证数据可靠性既能防止数据丢失又能防止数据重复。避免乱序按照序列号将数据进行还原。能够提高效率基于序列号可以实现多次发送一次接收。确认序列号接收方接收到数据后会回传ACK报文。报文中带有此的次确认序列号用于告知发送方此次已经接收情况。在指定时间后发送端仍未收到确认应答就会启动超时重传。流量控制接收端处理数据速度是有限的。如果发送方发送数据过快这会导致接收端缓冲区溢出从而丢包。为了避免上述情况发生TCP支持根据接收端处理能力来决定发送端的发送速度这就是流量控制。流量控制是通过在TCP报文首部维护一个滑动窗口来实现。拥塞控制拥塞控制是当网络严重拥堵发送端减少发送数据。拥塞控制是通过发送端维护一个拥塞窗口来实现。可以得出发送端发送速度受限于滑动窗口和拥塞窗口的最小值。拥塞控制方法分为慢开始拥塞避免、快重传和快恢复。3. 怎么用UDP实现HTTP实现思路把TCP在操作系统内核中做的事情搬到用户态自己实现一遍。解决乱序与去重问题UDP发送的包可能错误顺序被接收端接收。或因为网络问题导致重传导致接收端收到重复包。实现在udp包内自定义一个应用层包头。加入序列号seq接收端需要在内存中维护一个接收缓冲区按照序列号把乱序的包重新拼装成连续的字节后再交给HTTP协议器处理。解决丢包问题问题udp发出后就不管了丢了也不知道。实现引入确认应答ACK机制。客户端发送seq100的udp包开启一个定时器。服务端收到后会回复一个带有ACK101的包。如果客户端的定时器超时还没有收到ACK就主动重传seq100的包。大文件传输的分片与重组问题HTTP经常用来传大图片或视频如果把1M的HTTP报文扔给udp这会导致底层IP层严重分片一旦丢一个IP分片整个1M数据全部作废。实现应用层必须主动将巨大HTTP报文切分成小于网络MTU通常是1500字节减去IP和udp头部后大约1400字节左右的小块。每一块独立封装成一个udp包发送。连接状态管理问题TCP是由[源IP源端口目的IP目的端口]四元组唯一标识的。如果用户拿着手机从5G切换到WiFi IP变了TCP连接也会断开。实现既然用udp我们可以彻底摒弃IP和端口的束缚在自定义的报头里加一个唯一的连接标识符。不管客户端的IP怎么变只要报头里的ID没变服务端就认为这是同一个HTTP会话。这就是QUIC协议连接迁移的特性。流量控制与拥塞控制滑动窗口)问题不能因为udp快就往死里发这会把对方接收缓冲区打满或者把中间路由器打挂。实现需要在应用层实现一套类似于TCP滑动窗口机制根据接收端处理能力流量控制和网络拥堵程序度拥塞控制动态调整发送窗口的大小。4. TCP粘包怎么解决TCP是面向字节流的协议底层根本没有包的概念。数据就像流水一样源源不断的从发送端流向接收端。因此在应用层人为地规定消息的边界。主要解决办法有以下3种。消息定长核心思路规定发送的每一个消息长度都是固定不变的比如固定1024个字节如果真实数据不够1024字节就用空格或/0补齐。接收端逻辑每次死循环recv只要缓冲区里凑够了1024字节就直接截断拿出来作为一个完整消息处理。优缺点极其简单但非常浪费网络带宽。特殊分隔符核心思路在每个消息的尾部加上一个约定的特殊字符比如/r或/r/n接收端逻辑不断读取字节流并在内存中扫描一旦扫到这个特殊字符就把前面数据当做一个完整的包切出来。经典应用HTTP协议的header部分通过/r/n/r/n分割Header和Body。优缺点实现较为直观但不适合传输二进制数据比如图片视频因为二进制正文中很容易碰到和分割符一模一样字节导致被错误截断。如非要传正文必须先做BASE64编码或转义。消息头带长度该方法是最常用最标准的解决方案。核心思路将网络包分为Header包头和Body包体两部分。包头的长度是固定的比如固定四字节里面存放一个32位整数这个整数精确记录了后面包体的总长度。接收端处理逻辑先读包头强制recv固定的4字节解析出Body的长度N。再读包体写一个while循环继续recv直到精确地把这个N个字节全部读完多一个字节都不要。此时一个完整干净的业务包就拿到了。经典应用HTTP/1.1的Body部分利用Header中的Content-Length字段表明正文长度绝大多数自定义私有协议。5. 滑动窗口核心作用移动窗口是TCP协议中用于实现流量控制和提升传输效率的核心机制。提升效率。传统的停等协议发一个就要等待一个AC K效率太低。滑动窗口允许发送方在没有收到确认的情况下。连续发送多个数据包极大提高了网络的吞吐量。流量控制可以防止发送方发的太快导致接收方缓冲区。被打满而丢包的问题让发送方的发送速率与接收方的处理能力相匹配。工作机制。接收方的通知窗口。接收方在每次回复ACK时会通知TCP报头告诉发送方自己当前的接收缓冲区还能容纳多少字节。发送方窗口的状态划分。发送方的窗口大小由接收方通告的窗口大小决定。发送方会将维护的字节序列分为四部分已发送且已收到AC K确认的数据窗口左侧。已发送但未收到ACK确认的数据窗口内。未发送但允许发送的数据窗口内这是剩余的可用额度。未发送且不允许发送的数据窗口左侧。超出接收方能力。滑动过程只有当窗口最左侧的数据收到ACK后。整个窗口才会向右滑动从而释放额度允许发送后续的新数据。如果接收方缓冲区满了通告窗口为零会发生什么零窗口探测当发送方收到零窗口通知时会停止发送应用数据。为了防止接收方后来发送的窗口恢复ACK报文丢失导致双方死锁发送方会启动一个坚持定时器。定时器超时后发送方会主动发送一个零窗口探测报文强制接收方重新发送窗口大小。什么是糊涂窗口综合症怎么解决如果接收方应用程序处理的很慢每次只从缓冲区读几个字节然后向发送方通告一个只有几个字节的小窗口发送方一收到就立刻发送这几个字节数据这会导致TCP报文头部20加20字节的开销远大于有效载荷极大浪费网络带宽。解决方案接收端如果可用空间太小干脆通告窗口为零直到缓冲区可用空间达到总空间一半或能容纳一个MSS最大报文长度时在通告真实窗口。发送端配合使用Nagle算法只要还有未确认数据就把零碎小数据攒起来等收到前一个ACK或攒够一个MSS时再一起发。6. 拥塞控制滑动窗口解决的是收发双方点对点的处理速度匹配问题。而拥塞控制解决是全局网络链路的过载问题防止过多数据同时注入网络导致路由瘫痪。核心技术四大算法TCP通过维护一个拥塞窗口cwnd和一个慢启动阀值ssthresh。来动态调整发送速率。整个过程分为四个阶段联动。慢启动连接刚建立或严重超时时触发cwnd初始为极小值如十个mss。这表明发送方会发送十个MSS此后每收到一个ACK cwnd就加一也就是 RTT往返时间窗口大小呈指数级翻倍它的目的是快速探测网络的承载能力。拥塞避免当cwnd增长到等于或超过ssthresh时指数增长停止转为线性增长。每个RTT内的cwnd增加一个MSS这是一种保守的示范避免瞬间压垮网络。快重传 TCP不会死板等待超时定时器触发。如果发送方连续收到三个重复ACK说明该ACK后序列的包丢失但网络整体没断。此时会立即重传丢失的包提高响应速度。快恢复触发快重传后TCP认为网络只是轻微拥塞。此时会将ssthresh减半cwnd也设置为减半后的值部分实际会加上三个mss。然后直接进入拥塞避免的线性增长阶段而不是退回到慢启动从而维持较高的吞吐量。拥塞判定的两种程度严重拥塞RTO超时数据包石沉大海连重复的ACK都收不到。此时极其严厉cwnd直接重置为1ssthresh减半重新进入慢启动。轻微拥塞收到三个重复ACK触发快重传和快恢复平滑处理性能抖动较小。

相关文章:

计算机网络之TCP和UDP的底层机制

文章目录 1. TCP和UDP区别&#xff1f;2.TCP为什么可靠传输3. 怎么用UDP实现HTTP&#xff1f;4. TCP粘包怎么解决5. 滑动窗口6. 拥塞控制 1. TCP和UDP区别&#xff1f; TCP&#xff1a; 报头 TCP发送数据 客户端&#xff1a; #include <iostream> #include <strin…...

基于深度学习的苹果叶片病虫害识别系统,resnet50,vgg16,resnet34【pytorch框架,python源码】

更多图像分类、图像识别、目标检测、图像分割&#xff0c;图像检索等项目可从主页查看 功能演示(要看shi pin下面的简介)&#xff1a; https://www.bilibili.com/video/BV1Bs4XzcEdH/?spm_id_from333.1387.homepage.video_card.click&vd_source95b9b70984596ccebdb2780f0…...

2026年安卓APP安全加固公司哪家好?从技术、性能到合规的深度选型指南

选错安卓APP安全加固公司&#xff0c;后果可能比不加固更严重——应用商店审核被拒、上线后用户疯狂反馈闪退、核心代码被逆向破解导致数据泄露。更棘手的是&#xff0c;当你拿着加固报告去应对等保测评时&#xff0c;可能因为加固方案不被认可而功亏一篑。这些都不是危言耸听&…...

S32K144外部中断实战:用按键控制LED,手把手教你避开中断标志位清除的坑

S32K144外部中断实战&#xff1a;从按键消抖到标志位管理的完整解决方案 在嵌入式开发中&#xff0c;外部中断是实现实时响应的关键机制。S32K144作为NXP面向汽车电子和工业控制的主力MCU&#xff0c;其中断系统的灵活性和可靠性备受开发者青睐。但看似简单的按键中断控制LED背…...

有孩家庭接送场景混动车型实证测评:座舱健康与续航便捷性核心指标对比研究

摘要在城市通勤与萌娃接送双场景叠加的用车需求下&#xff0c;座舱健康&#xff08;甲醛 / 异味控制、空气净化&#xff09;与出行便捷性&#xff08;纯电续航、亏电油耗、场景适配收纳&#xff09;已成为有孩家庭选购混动 SUV 的核心决策指标。针对当前市场车型 “健康参数标注…...

源码解读:拿下顶会最佳论文的重建式VLA,是如何实现的!

“如果模型能重建它&#xff0c;就说明它真正注意到了它” ——源码级解析 目录 01 问题的起点&#xff1a;为什么 VLA 需要“重建”&#xff1f; 02 系统架构总览 03 核心技术一&#xff1a;DiT 扩散去噪与 adaLN-Zero 条件注入 条件的构建与融合 adaLN-Zero 注入逻…...

5篇2章12节:诊断试验准确性研究与多阈值Meta分析方法(下篇:可视计算)

多阈值 Meta 分析通过整合全阈值范围内的诊断效能数据,为临床实践提供更全面的证据支持,但数据转换的复杂性与结果可视化的直观性一直是技术痛点。本文作为诊断试验准确性研究系列下篇,接着全面讲解可视化绘图和诊断统计量计算 。 一、plot () 函数 diagmeta包的plot()函数…...

探索JavaScript中的生命游戏:细胞自动机的实现

探索JavaScript中的生命游戏:细胞自动机的实现 在计算机科学中,细胞自动机(Cellular Automata)是一个非常有趣的研究领域,而康威生命游戏(Conway’s Game of Life)则是其中最著名的例子之一。今天我们将探讨如何使用JavaScript来实现这个经典的自动机模型,并特别关注其…...

终极指南:5分钟快速掌握B站视频转文字开源工具bili2text

终极指南&#xff1a;5分钟快速掌握B站视频转文字开源工具bili2text 【免费下载链接】bili2text Bilibili视频转文字&#xff0c;一步到位&#xff0c;输入链接即可使用 项目地址: https://gitcode.com/gh_mirrors/bi/bili2text 还在为手动记录B站视频内容而烦恼吗&…...

快速掌握 FastAPI 路由:从基础到进阶

前言 FastAPI 是一个现代的、快速&#xff08;高性能&#xff09;的 Web 框架&#xff0c;用于构建 API。它基于 Python 类型注解&#xff0c;支持自动生成 OpenAPI 文档&#xff0c;且性能优越&#xff0c;适合构建高效的 Web API。本文将详细介绍如何在 FastAPI 中使用路由处…...

生成式AI数据飞轮构建:从0到规模化复利增长的6个关键杠杆(附某金融大模型真实飞轮增速曲线)

第一章&#xff1a;生成式AI应用数据飞轮构建 2026奇点智能技术大会(https://ml-summit.org) 生成式AI应用的数据飞轮并非天然形成&#xff0c;而是依赖闭环反馈机制驱动的持续演进系统&#xff1a;用户交互产生真实行为数据 → 数据经清洗与标注强化模型能力 → 模型升级提升…...

C 语言教程

C 语言教程C 语言是一种通用的、面向过程式的计算机程序设计语言。1972 年&#xff0c;为了移植与开发 UNIX 操作系统&#xff0c;丹尼斯里奇在贝尔电话实验室设计开发了 C 语言。 C 语言是一种广泛使用的计算机语言&#xff0c;它与 Java 编程语言一样普及&#xff0c;二者在现…...

揭秘多模态餐饮推荐系统落地难题:从BERT-Vision融合到实时推理延迟压降至89ms的实战路径

第一章&#xff1a;2026奇点智能技术大会&#xff1a;多模态餐饮推荐 2026奇点智能技术大会(https://ml-summit.org) 在2026奇点智能技术大会上&#xff0c;多模态餐饮推荐系统成为核心展示场景之一。该系统融合用户历史行为、实时图像&#xff08;如菜品照片&#xff09;、语…...

汽车c语言是什么?

汽车C语言主要是指在汽车电子系统中使用的C语言编程技术。以下是关于汽车C语言的详细解释&#xff1a; 1.定义与用途&#xff1a; 汽车C语言指的是在汽车制造和软件开发中&#xff0c;特别是在嵌入式系统编程中广泛使用的C语言。 它被用于编写控制汽车各种电子设备的软件&#…...

DeepSeek角色扮演指令终极指南:解锁AI自由对话新境界

1. DeepSeek角色扮演的核心原理 DeepSeek的角色扮演功能之所以强大&#xff0c;关键在于它突破了传统AI对话的边界。传统AI对话往往受限于预设的道德准则和安全审查机制&#xff0c;而DeepSeek通过特殊的指令设计&#xff0c;实现了真正意义上的自由对话体验。这就像给AI装了一…...

联邦学习落地金融风控:当银行遇到电商,如何在不共享数据的前提下联合建模?

联邦学习在金融风控中的实战&#xff1a;银行与电商的数据协作新范式 想象一下这样的场景&#xff1a;一家商业银行拥有客户的信用评分和还款记录&#xff0c;而某大型电商平台则积累了同一批用户的消费行为、浏览偏好和支付习惯。这两组数据如果能够联合建模&#xff0c;将大幅…...

微信小程序里用H5预览PDF,我为什么放弃了原生组件选了pdf.min.js?

微信小程序PDF预览方案深度解析&#xff1a;为何pdf.min.js成为技术选型最优解&#xff1f; 在微信小程序生态中实现PDF预览功能时&#xff0c;开发者往往面临技术路线的关键抉择。原生组件、云服务方案与H5渲染引擎各具特点&#xff0c;但经过多次实战验证&#xff0c;基于pdf…...

HWSDv2.0土壤数据怎么用?从全球1km栅格到你的研究区,这份ArcGIS Pro掩膜裁剪指南请收好

HWSDv2.0土壤数据区域提取实战&#xff1a;ArcGIS Pro高效掩膜裁剪技巧 当全球1公里分辨率的HWSDv2.0土壤数据遇上具体研究区域&#xff0c;如何快速提取目标范围的数据成为科研工作者的首要挑战。本文将手把手教你使用ArcGIS Pro完成从全局到局部的精准数据裁剪&#xff0c;让…...

QLabel的四种内容呈现模式

1. QLabel的多面手特性&#xff1a;不只是显示文字 第一次接触Qt开发时&#xff0c;很多人会把QLabel简单理解为一个"文字标签控件"。直到我在实际项目中需要显示动态图表时&#xff0c;才发现这个看似简单的控件竟然藏着这么多玩法。QLabel本质上是一个多功能显示容…...

Codex 前端实战:AI 能画出设计稿,也能写代码,但如何让它不再“像 AI 做的”?

Codex 前端实战:AI 能画出设计稿,也能写代码,但如何让它不再“像 AI 做的”? 上周我用 Codex 把一份 Figma 设计稿丢给它,三分钟生成了一个完整的前端页面。同事们看完说:“这个看起来不太像 AI 做的。” 这句话很有意思。默认情况下,Codex 生成的前端代码确实有一股“…...

用骗孩子压岁钱的故事,来解释AI 技术

❝开头还是介绍一下群&#xff0c;如果感兴趣PolarDB ,MongoDB ,MySQL ,PostgreSQL ,Redis, OceanBase, Sql Server等有问题&#xff0c;有需求都可以加群群内有各大数据库行业大咖&#xff0c;可以解决你的问题。加群请联系 liuaustin3 &#xff0c;&#xff08;共3400人左右 …...

嵌入式系统设计实践

嵌入式系统设计实践&#xff1a;连接数字与现实的桥梁 在智能设备无处不在的时代&#xff0c;嵌入式系统作为硬件与软件的完美结合体&#xff0c;悄然驱动着从智能家居到工业控制的各个领域。它不仅是技术的核心&#xff0c;更是创新应用的基石。本文将带你深入嵌入式系统设计…...

你怎么知道AI真的做对了?我花了三个月才想明白这个问题

你怎么知道AI真的做对了?我花了三个月才想明白这个问题 用AI写代码这件事,最让人上头的不是它能写多快,而是它总能用一种“我绝对没问题”的语气给你输出结果。然后你看着那个结果,心里开始打鼓:这玩意儿到底对不对? 我经历过三个阶段。第一阶段是“盲目信任期”——看…...

ViGEmBus深度解析:Windows内核级游戏控制器虚拟化架构揭秘

ViGEmBus深度解析&#xff1a;Windows内核级游戏控制器虚拟化架构揭秘 【免费下载链接】ViGEmBus Windows kernel-mode driver emulating well-known USB game controllers. 项目地址: https://gitcode.com/gh_mirrors/vi/ViGEmBus 在游戏输入设备兼容性领域&#xff0c…...

江城智造,共赴盛会!AICA数智创新公开课·武汉专场圆满举办

在4月9日&#xff0c;由武汉市工业信息化中心指导&#xff0c;百度飞桨&#xff08;武汉&#xff09;人工智能产业赋能中心主办&#xff0c;国家数字化设计与制造创新中心、e-works、武汉企业信息化促进会、深度学习技术及应用国家工程研究中心、湖北省人工智能学会协办的「首席…...

基于Raspberry Pi和OpenCV的家庭智能监控系统

智能家居新选择&#xff1a;树莓派监控系统 在科技飞速发展的今天&#xff0c;家庭安全监控已成为现代生活的刚需。基于Raspberry Pi&#xff08;树莓派&#xff09;和OpenCV的智能监控系统&#xff0c;凭借低成本、高灵活性和强大图像处理能力&#xff0c;成为DIY爱好者和技术…...

自动化测试:PO模式介绍及案例

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快PO&#xff08;Page Object&#xff09;设计模式是一种面向对象( 页面对象&#xff09;的设计模式&#xff0c;将测试对象及单个的测试步骤封装在每个Page对象以pag…...

【GitHub项目推荐--InkOS:把 AI 写小说变成“全自动流水线”】

GitHub 地址&#xff1a;https://github.com/Narcooo/inkos 简介 InkOS​ 是一个基于 TypeScript 开发的命令行&#xff08;CLI&#xff09;AI 小说创作智能体。它彻底颠覆了“单次生成”的玩法&#xff0c;将写小说变成了一个可审计、可续写、可仿写的工程化系统。 普通 AI …...

终极Windows安卓应用安装指南:如何快速批量安装APK文件

终极Windows安卓应用安装指南&#xff1a;如何快速批量安装APK文件 【免费下载链接】APK-Installer An Android Application Installer for Windows 项目地址: https://gitcode.com/GitHub_Trending/ap/APK-Installer 想要在Windows电脑上轻松安装Android应用吗&#xf…...

K8s StatefulSet 的数据持久化方案

Kubernetes StatefulSet 的数据持久化方案解析 在云原生应用架构中&#xff0c;有状态服务的数据持久化一直是关键挑战之一。Kubernetes StatefulSet 作为管理有状态工作负载的核心控制器&#xff0c;通过独特的持久化机制为分布式数据库、消息队列等场景提供了稳定支持。本文…...