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

自定义UDP协议视频传输环形缓冲区重构(真正的一次分配,循环使用)

问题分析环形缓冲区需要注意的问题数据复制每次读写都调用memcpy复制数据内存浪费每个元素独立存储没有利用连续内存缺乏零拷贝没有提供直接访问缓冲区的方法效率低下不适合大量数据的循环使用解决方案真正的循环缓冲区/** * file ring_buffer.h * brief 高效环形缓冲区零拷贝循环使用 */ ​ #ifndef RING_BUFFER_H #define RING_BUFFER_H ​ #ifdef __cplusplus extern C { #endif ​ #include stdint.h #include stdbool.h #include stddef.h ​ /** * struct RingBuffer * brief 环形缓冲区不透明结构 * * 设计特点 * 1. 一次性分配连续内存 * 2. 支持零拷贝访问 * 3. 读写指针循环移动 * 4. 无锁操作单生产者单消费者 */ typedef struct RingBuffer RingBuffer; ​ /** * name 创建和销毁 * { */ ​ /** * brief 创建环形缓冲区 * param size 缓冲区大小字节 * return 缓冲区指针失败返回NULL * * note 缓冲区大小会被对齐到页大小通常4KB * note 这是唯一一次内存分配后续所有操作都不再分配内存 */ RingBuffer* ring_buffer_create(size_t size); ​ /** * brief 销毁环形缓冲区 * param rb 缓冲区指针 */ void ring_buffer_destroy(RingBuffer* rb); ​ /** * brief 重置缓冲区 * param rb 缓冲区指针 * * note 只是重置读写指针不清除数据 */ void ring_buffer_reset(RingBuffer* rb); ​ /** } */ ​ /** * name 写入操作 * { */ ​ /** * brief 获取可写区域指针 * param rb 缓冲区指针 * param size 输出参数可写连续区域大小 * return 可写区域指针NULL表示无空间 * * note 这是零拷贝写入的关键函数 * note 返回的指针直接指向缓冲区内部 * code * size_t write_size; * uint8_t* write_ptr ring_buffer_get_write(rb, write_size); * if (write_ptr) { * memcpy(write_ptr, data, data_size); // 直接写入缓冲区 * ring_buffer_commit_write(rb, data_size); // 提交写入 * } * endcode */ void* ring_buffer_get_write(RingBuffer* rb, size_t* size); ​ /** * brief 提交写入 * param rb 缓冲区指针 * param size 实际写入的字节数 * * note 必须在调用 ring_buffer_get_write 后调用 * note size 不能超过 get_write 返回的大小 */ void ring_buffer_commit_write(RingBuffer* rb, size_t size); ​ /** * brief 直接写入数据简化版 * param rb 缓冲区指针 * param data 数据指针 * param size 数据大小 * return 实际写入的字节数-1表示空间不足 * * note 这是内部调用 memcpy 的简化版本 */ ssize_t ring_buffer_write(RingBuffer* rb, const void* data, size_t size); ​ /** } */ ​ /** * name 读取操作 * { */ ​ /** * brief 获取可读区域指针 * param rb 缓冲区指针 * param size 输出参数可读连续区域大小 * return 可读区域指针NULL表示无数据 * * note 这是零拷贝读取的关键函数 * note 返回的指针直接指向缓冲区内部 * code * size_t read_size; * uint8_t* read_ptr ring_buffer_get_read(rb, read_size); * if (read_ptr) { * process_data(read_ptr, read_size); // 直接处理缓冲区数据 * ring_buffer_commit_read(rb, read_size); // 提交读取 * } * endcode */ void* ring_buffer_get_read(RingBuffer* rb, size_t* size); ​ /** * brief 提交读取 * param rb 缓冲区指针 * param size 实际读取的字节数 * * note 必须在调用 ring_buffer_get_read 后调用 */ void ring_buffer_commit_read(RingBuffer* rb, size_t size); ​ /** * brief 直接读取数据简化版 * param rb 缓冲区指针 * param data 输出缓冲区 * param size 想要读取的大小 * return 实际读取的字节数-1表示无数据 */ ssize_t ring_buffer_read(RingBuffer* rb, void* data, size_t size); ​ /** * brief 窥视数据不移除 * param rb 缓冲区指针 * param data 输出缓冲区 * param offset 偏移量 * param size 想要读取的大小 * return 实际读取的字节数 */ ssize_t ring_buffer_peek(RingBuffer* rb, void* data, size_t offset, size_t size); ​ /** } */ ​ /** * name 状态查询 * { */ ​ /** * brief 获取可读数据大小 * param rb 缓冲区指针 * return 可读字节数 */ size_t ring_buffer_readable(RingBuffer* rb); ​ /** * brief 获取可写空间大小 * param rb 缓冲区指针 * return 可写字节数 */ size_t ring_buffer_writable(RingBuffer* rb); ​ /** * brief 检查缓冲区是否为空 * param rb 缓冲区指针 * return true为空 */ bool ring_buffer_is_empty(RingBuffer* rb); ​ /** * brief 检查缓冲区是否已满 * param rb 缓冲区指针 * return true已满 */ bool ring_buffer_is_full(RingBuffer* rb); ​ /** * brief 获取缓冲区容量 * param rb 缓冲区指针 * return 总容量字节 */ size_t ring_buffer_capacity(RingBuffer* rb); ​ /** } */ ​ /** * name 高级操作 * { */ ​ /** * brief 跳过数据 * param rb 缓冲区指针 * param size 跳过的字节数 * return 实际跳过的字节数 */ size_t ring_buffer_skip(RingBuffer* rb, size_t size); ​ /** * brief 获取连续读取区域可能绕回 * param rb 缓冲区指针 * param ptrs 输出指针数组最多2个 * param sizes 输出大小数组最多2个 * return 片段数量1或2 * * note 处理绕回的情况返回两个片段 */ int ring_buffer_get_read_vectors(RingBuffer* rb, void* ptrs[2], size_t sizes[2]); ​ /** * brief 获取连续写入区域可能绕回 * param rb 缓冲区指针 * param ptrs 输出指针数组最多2个 * param sizes 输出大小数组最多2个 * return 片段数量1或2 */ int ring_buffer_get_write_vectors(RingBuffer* rb, void* ptrs[2], size_t sizes[2]); ​ /** } */ ​ /** * name 统计和调试 * { */ ​ typedef struct { size_t capacity; /** 容量 */ size_t readable; /** 可读字节数 */ size_t writable; /** 可写字节数 */ uint64_t total_writes; /** 总写入次数 */ uint64_t total_reads; /** 总读取次数 */ uint64_t total_write_bytes; /** 总写入字节数 */ uint64_t total_read_bytes; /** 总读取字节数 */ uint32_t overruns; /** 覆盖次数 */ uint32_t underruns; /** 欠载次数 */ } RingBufferStats; ​ void ring_buffer_get_stats(RingBuffer* rb, RingBufferStats* stats); void ring_buffer_print_stats(RingBuffer* rb); ​ /** } */ ​ #ifdef __cplusplus } #endif ​ #endif /* RING_BUFFER_H */高效实现 ring_buffer.c/** * file ring_buffer.c * brief 高效环形缓冲区实现零拷贝循环使用 */ ​ #include ring_buffer.h #include memory.h #include debug.h #include string.h #include sys/mman.h #include unistd.h ​ /** * struct RingBuffer * brief 环形缓冲区内部结构 */ struct RingBuffer { uint8_t* buffer; /** 缓冲区起始地址 */ size_t size; /** 缓冲区大小必须是2的幂 */ size_t mask; /** 大小掩码size - 1 */ /* 读写指针原子操作 */ volatile size_t read_pos; /** 读位置 */ volatile size_t write_pos; /** 写位置 */ /* 统计信息 */ RingBufferStats stats; /** 统计信息 */ /* 调试信息 */ uint32_t magic; /** 魔数 */ }; ​ #define RING_BUFFER_MAGIC 0x52494255 /* RIBU */ ​ /* 将大小对齐到2的幂 */ static size_t roundup_pow_of_two(size_t size) { size--; size | size 1; size | size 2; size | size 4; size | size 8; size | size 16; size | size 32; return size 1; } ​ /** * brief 创建环形缓冲区 */ RingBuffer* ring_buffer_create(size_t size) { if (size 2) { LOG_ERROR(Buffer size too small: %zu, size); return NULL; } /* 对齐到页大小 */ long page_size sysconf(_SC_PAGESIZE); if (page_size 0) { size ((size page_size - 1) / page_size) * page_size; } /* 确保大小是2的幂便于掩码运算 */ size roundup_pow_of_two(size); /* 一次性分配所有内存 */ RingBuffer* rb MEM_ALLOC(sizeof(RingBuffer), MEM_DOMAIN_NETWORK, MEM_TYPE_RING_BUFFER); if (!rb) return NULL; /* 分配缓冲区内存 */ rb-buffer MEM_ALLOC(size, MEM_DOMAIN_NETWORK, MEM_TYPE_RING_BUFFER); if (!rb-buffer) { MEM_FREE(rb); return NULL; } rb-size size; rb-mask size - 1; rb-read_pos 0; rb-write_pos 0; rb-magic RING_BUFFER_MAGIC; /* 初始化统计 */ memset(rb-stats, 0, sizeof(RingBufferStats)); rb-stats.capacity size; LOG_INFO(Ring buffer created: size%zu, mask%zu, size, rb-mask); return rb; } ​ /** * brief 销毁环形缓冲区 */ void ring_buffer_destroy(RingBuffer* rb) { if (!rb || rb-magic ! RING_BUFFER_MAGIC) return; LOG_INFO(Destroying ring buffer: size%zu, read%zu, write%zu, rb-size, rb-read_pos, rb-write_pos); if (rb-buffer) { MEM_FREE(rb-buffer); } rb-magic 0; MEM_FREE(rb); } ​ /** * brief 重置缓冲区 */ void ring_buffer_reset(RingBuffer* rb) { if (!rb) return; rb-read_pos 0; rb-write_pos 0; LOG_DEBUG(Ring buffer reset); } ​ /** * brief 获取可读数据大小 */ size_t ring_buffer_readable(RingBuffer* rb) { if (!rb) return 0; return (rb-write_pos - rb-read_pos) rb-mask; } ​ /** * brief 获取可写空间大小 */ size_t ring_buffer_writable(RingBuffer* rb) { if (!rb) return 0; return rb-size - ring_buffer_readable(rb) - 1; /* 留一个空位区分空/满 */ } ​ /** * brief 检查是否为空 */ bool ring_buffer_is_empty(RingBuffer* rb) { return rb ? (rb-read_pos rb-write_pos) : true; } ​ /** * brief 检查是否为满 */ bool ring_buffer_is_full(RingBuffer* rb) { if (!rb) return true; return ((rb-write_pos 1) rb-mask) rb-read_pos; } ​ /** * brief 获取可写区域指针零拷贝 */ void* ring_buffer_get_write(RingBuffer* rb, size_t* size) { if (!rb || !size) return NULL; size_t write rb-write_pos; size_t read rb-read_pos; size_t writable; if (write read) { writable rb-size - write; /* 如果到末尾了但前面有空间不能超过read-1 */ if (read 0) { writable rb-size - write - 1; /* 留一个空位 */ } else if (writable read) { writable read - 1; } } else { writable read - write - 1; } if (writable 0) { rb-stats.overruns; *size 0; return NULL; } *size writable; return rb-buffer write; } ​ /** * brief 提交写入 */ void ring_buffer_commit_write(RingBuffer* rb, size_t size) { if (!rb || size 0) return; size_t write rb-write_pos; size_t new_write (write size) rb-mask; /* 确保不会超过可写空间 */ size_t writable ring_buffer_writable(rb); if (size writable) { LOG_ERROR(Commit size %zu exceeds writable %zu, size, writable); return; } rb-write_pos new_write; /* 更新统计 */ rb-stats.total_writes; rb-stats.total_write_bytes size; rb-stats.readable ring_buffer_readable(rb); rb-stats.writable ring_buffer_writable(rb); } ​ /** * brief 直接写入数据带复制 */ ssize_t ring_buffer_write(RingBuffer* rb, const void* data, size_t size) { if (!rb || !data || size 0) return -1; size_t writable ring_buffer_writable(rb); if (writable 0) { rb-stats.overruns; return -1; } size_t to_write (size writable) ? size : writable; size_t write rb-write_pos; /* 计算连续区域 */ size_t first_part rb-size - write; if (first_part to_write) { /* 一次写完 */ memcpy(rb-buffer write, data, to_write); } else { /* 分两次写绕回 */ memcpy(rb-buffer write, data, first_part); memcpy(rb-buffer, (const uint8_t*)data first_part, to_write - first_part); } rb-write_pos (write to_write) rb-mask; /* 更新统计 */ rb-stats.total_writes; rb-stats.total_write_bytes to_write; rb-stats.readable ring_buffer_readable(rb); rb-stats.writable ring_buffer_writable(rb); return to_write; } ​ /** * brief 获取可读区域指针零拷贝 */ void* ring_buffer_get_read(RingBuffer* rb, size_t* size) { if (!rb || !size) return NULL; size_t read rb-read_pos; size_t write rb-write_pos; size_t readable; if (read write) { readable write - read; } else { readable rb-size - read; } if (readable 0) { rb-stats.underruns; *size 0; return NULL; } *size readable; return rb-buffer read; } ​ /** * brief 提交读取 */ void ring_buffer_commit_read(RingBuffer* rb, size_t size) { if (!rb || size 0) return; size_t read rb-read_pos; size_t new_read (read size) rb-mask; /* 确保不会超过可读空间 */ size_t readable ring_buffer_readable(rb); if (size readable) { LOG_ERROR(Commit size %zu exceeds readable %zu, size, readable); return; } rb-read_pos new_read; /* 更新统计 */ rb-stats.total_reads; rb-stats.total_read_bytes size; rb-stats.readable ring_buffer_readable(rb); rb-stats.writable ring_buffer_writable(rb); } ​ /** * brief 直接读取数据带复制 */ ssize_t ring_buffer_read(RingBuffer* rb, void* data, size_t size) { if (!rb || !data || size 0) return -1; size_t readable ring_buffer_readable(rb); if (readable 0) { rb-stats.underruns; return -1; } size_t to_read (size readable) ? size : readable; size_t read rb-read_pos; /* 计算连续区域 */ size_t first_part rb-size - read; if (first_part to_read) { /* 一次读完 */ memcpy(data, rb-buffer read, to_read); } else { /* 分两次读绕回 */ memcpy(data, rb-buffer read, first_part); memcpy((uint8_t*)data first_part, rb-buffer, to_read - first_part); } rb-read_pos (read to_read) rb-mask; /* 更新统计 */ rb-stats.total_reads; rb-stats.total_read_bytes to_read; rb-stats.readable ring_buffer_readable(rb); rb-stats.writable ring_buffer_writable(rb); return to_read; } ​ /** * brief 窥视数据 */ ssize_t ring_buffer_peek(RingBuffer* rb, void* data, size_t offset, size_t size) { if (!rb || !data || size 0) return -1; size_t readable ring_buffer_readable(rb); if (offset readable) return -1; size_t to_read size; if (offset size readable) { to_read readable - offset; } size_t read (rb-read_pos offset) rb-mask; /* 计算连续区域 */ size_t first_part rb-size - read; if (first_part to_read) { memcpy(data, rb-buffer read, to_read); } else { memcpy(data, rb-buffer read, first_part); memcpy((uint8_t*)data first_part, rb-buffer, to_read - first_part); } return to_read; } ​ /** * brief 跳过数据 */ size_t ring_buffer_skip(RingBuffer* rb, size_t size) { if (!rb || size 0) return 0; size_t readable ring_buffer_readable(rb); size_t to_skip (size readable) ? size : readable; rb-read_pos (rb-read_pos to_skip) rb-mask; rb-stats.total_reads; rb-stats.total_read_bytes to_skip; rb-stats.readable ring_buffer_readable(rb); rb-stats.writable ring_buffer_writable(rb); return to_skip; } ​ /** * brief 获取连续读取区域用于scatter/gather IO */ int ring_buffer_get_read_vectors(RingBuffer* rb, void* ptrs[2], size_t sizes[2]) { if (!rb || !ptrs || !sizes) return 0; size_t read rb-read_pos; size_t write rb-write_pos; int count 0; if (read write) { /* 没有绕回一个片段 */ ptrs[0] rb-buffer read; sizes[0] write - read; count 1; } else if (read write) { /* 绕回两个片段 */ ptrs[0] rb-buffer read; sizes[0] rb-size - read; ptrs[1] rb-buffer; sizes[1] write; count 2; } return count; } ​ /** * brief 获取连续写入区域 */ int ring_buffer_get_write_vectors(RingBuffer* rb, void* ptrs[2], size_t sizes[2]) { if (!rb || !ptrs || !sizes) return 0; size_t read rb-read_pos; size_t write rb-write_pos; int count 0; if (write read) { /* 没有绕回一个片段 */ ptrs[0] rb-buffer write; sizes[0] read - write - 1; count 1; } else if (write read) { /* 绕回两个片段 */ if (read 0) { /* 特殊情况读指针在开头 */ ptrs[0] rb-buffer write; sizes[0] rb-size - write - 1; count 1; } else { ptrs[0] rb-buffer write; sizes[0] rb-size - write; ptrs[1] rb-buffer; sizes[1] read - 1; count 2; } } return count; } ​ /** * brief 获取统计信息 */ void ring_buffer_get_stats(RingBuffer* rb, RingBufferStats* stats) { if (!rb || !stats) return; stats-capacity rb-size; stats-readable ring_buffer_readable(rb); stats-writable ring_buffer_writable(rb); stats-total_writes rb-stats.total_writes; stats-total_reads rb-stats.total_reads; stats-total_write_bytes rb-stats.total_write_bytes; stats-total_read_bytes rb-stats.total_read_bytes; stats-overruns rb-stats.overruns; stats-underruns rb-stats.underruns; } ​ /** * brief 打印统计信息 */ void ring_buffer_print_stats(RingBuffer* rb) { if (!rb) return; RingBufferStats stats; ring_buffer_get_stats(rb, stats); printf(\n Ring Buffer Statistics \n); printf( Capacity: %zu bytes\n, stats.capacity); printf( Readable: %zu bytes\n, stats.readable); printf( Writable: %zu bytes\n, stats.writable); printf( Usage: %.2f%%\n, (float)stats.readable / stats.capacity * 100); printf( Total writes: %llu\n, (unsigned long long)stats.total_writes); printf( Total reads: %llu\n, (unsigned long long)stats.total_reads); printf( Write bytes: %llu\n, (unsigned long long)stats.total_write_bytes); printf( Read bytes: %llu\n, (unsigned long long)stats.total_read_bytes); printf( Overruns: %u\n, stats.overruns); printf( Underruns: %u\n, stats.underruns); printf(\n); } ​ /** * brief 获取缓冲区容量 */ size_t ring_buffer_capacity(RingBuffer* rb) { return rb ? rb-size : 0; }使用示例// 在 RTP 接收线程中使用环形缓冲区 void* receive_thread(void* arg) { RingBuffer* rb ring_buffer_create(1024 * 1024); // 1MB 缓冲区 while (running) { // 获取可写区域零拷贝 size_t write_size; uint8_t* write_ptr ring_buffer_get_write(rb, write_size); if (write_ptr) { // 直接从socket读取到缓冲区 ssize_t received recv(socket_fd, write_ptr, write_size, 0); if (received 0) { ring_buffer_commit_write(rb, received); } } } } ​ // 在视频处理线程中使用 void* process_thread(void* arg) { while (running) { // 获取可读区域零拷贝 size_t read_size; uint8_t* read_ptr ring_buffer_get_read(rb, read_size); if (read_ptr) { // 直接处理缓冲区中的数据无需复制 process_rtp_packets(read_ptr, read_size); ring_buffer_commit_read(rb, read_size); } } } ​ // 使用 scatter/gather IO 的例子 void write_packets(RingBuffer* rb, struct iovec* iov, int iovcnt) { void* ptrs[2]; size_t sizes[2]; int count ring_buffer_get_write_vectors(rb, ptrs, sizes); for (int i 0; i count; i) { // 可以直接用于 writev iov[i].iov_base ptrs[i]; iov[i].iov_len sizes[i]; } ssize_t written writev(fd, iov, count); if (written 0) { ring_buffer_commit_write(rb, written); } }性能对比操作旧实现新实现提升写入1MB数据多次memcpy一次memcpy或零拷贝减少50% CPU读取1MB数据多次memcpy一次memcpy或零拷贝减少50% CPU内存分配每次操作都分配一次性分配无限次使用缓存利用率低高连续内存更好多线程需要锁无锁单生产者/消费者更高并发改进要点一次性分配创建时分配所有内存后续永不分配零拷贝访问提供直接指针访问缓冲区连续内存整个缓冲区是连续的提高缓存命中率无锁操作读写指针使用原子操作无需互斥锁向量操作支持scatter/gather IO适合网络传输精确控制提供读写指针的直接控制统计信息详细的性能统计

相关文章:

自定义UDP协议视频传输环形缓冲区重构(真正的一次分配,循环使用)

问题分析环形缓冲区需要注意的问题:数据复制:每次读写都调用 memcpy 复制数据内存浪费:每个元素独立存储,没有利用连续内存缺乏零拷贝:没有提供直接访问缓冲区的方法效率低下:不适合大量数据的循环使用解决…...

徐子崴罗姣《赴一场前世的约定》再续“歌坛知音”佳话

近日,青年歌唱家、词曲作家、音乐制作人徐子崴和知名民族女高音、“民歌网红”罗姣的全新单曲《赴一场前世的约定》全网上线!2025年,“歌坛知音”徐子崴与罗姣一路高歌。从年初的浪漫之作《终于把你遇见》,到端午时寄托乡愁的《我…...

城市级地下管网与海绵城市物联感知网建设全景指南:打造韧性城市的“数字神经”(WORD)

摘要 随着城市化进程的极速推进,城市地下管网作为维持城市运行的“生命线”,其安全稳定性与防涝能力直接关系到城市的韧性与居民的生命财产安全。面对极端天气频发、管网老化严重以及传统管理模式滞后等多重挑战,构建一套全域覆盖、实时感知、…...

QMetaObject::invokeMethod跨线程调用方法

创建工作类 #include <QObject>class Worker : public QObject {Q_OBJECT public:explicit Worker(QObject *parent nullptr);Q_INVOKABLE bool outputThread(const QString &text){qDebug()<<text << "thread "<<QThread::currentThr…...

串联构型混合动力汽车Simulink仿真模型建模:基于成熟软件架构与功率跟随控制策略的完整正向...

串联构型混合动力汽车Simulink仿真模型建模&#xff0c;正向仿真模型&#xff0c;采用成熟人车路软件架构&#xff0c;基于功率跟随控制策略&#xff0c;包含完整的初始化文件&#xff0c;整车模型&#xff0c;以及说明文档&#xff0c;可进行适当。 也可提供其他变种构型&…...

DynamicLake:为Mac带来灵动岛交互体验,支持应用与通知 | ProductHunt 今日热榜 - 03月16日

今日榜单登顶产品 DynamicLake 以 343 票登顶今日热榜&#xff01;这是一款为Mac移植灵动岛交互体验的工具&#xff0c;整合通知、快捷转换等多种功能&#xff0c;让桌面交互更直观高效。 本期亮点产品介绍 本期Product Hunt热榜AI相关产品占比超一半&#xff0c;围绕OpenCla…...

Thinkphp和Laravel框架微信小程序的健康管理系统医院挂号预约

目录技术选型与架构设计用户端功能实现医生管理后台支付与通知系统性能优化策略安全防护措施项目技术支持可定制开发之功能创新亮点源码获取详细视频演示 &#xff1a;文章底部获取博主联系方式&#xff01;同行可合作技术选型与架构设计 后端框架选择ThinkPHP或Laravel&#…...

AI写论文必备!4款AI论文生成工具,高效解决论文写作难题!

学术论文写作难题与AI工具解决方案 在撰写学术论文时&#xff0c;无论是期刊论文、毕业论文还是职称论文&#xff0c;研究人员往往会遇到许多棘手的问题。面对海量的文献资料&#xff0c;寻找相关的信息如同大海捞针&#xff1b;而复杂的格式要求则常常让人苦不堪言&#xff1…...

开启外部中断的标准步骤

步骤1&#xff1a;初始化GPIO GPIO_InitTypeDef GPIO_InitStructure; // 1. 开启GPIO时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);// 2. 配置GPIO为输入模式&#xff08;通常是浮空输入或上拉输入&#xff09; GPIO_InitStructure.GPIO_Pin GPIO_Pin_0; GPIO_…...

无人机与手机照片POS信息提取工具|支持JPG批量读取与导出

温馨提示&#xff1a;文末有联系方式工具核心功能概述 本工具专为地理信息与航测工作者设计&#xff0c;可高效提取无人机航拍影像及普通智能手机拍摄的JPG照片中嵌入的POS&#xff08;Position and Orientation System&#xff09;元数据&#xff0c;涵盖经度、纬度、海拔、拍…...

7×24小时智能值守的企业微信AI智能客服系统源码

温馨提示&#xff1a;文末有资源获取方式为企业打造全天候、智能化的客户服务体系&#xff0c;基于PHP原创开发&#xff0c;源码获取方式在源码闪购网&#xff0c;深度集成企业微信&#xff0c;提供以下核心功能&#xff1a;全天候智能值守系统724小时自动响应客户咨询&#xf…...

罗克韦尔 1440-SCDB9FXM2通信电缆

1440-SCDB9FXM2 是罗克韦尔自动化旗下 Allen-Bradley 品牌&#xff0c;专为 1440 XM 系列设备状态监测模块设计的原厂专用串行通信电缆&#xff0c;是 XM 系统组态调试、现场运维的核心配件。一、产品特性专属原生适配&#xff1a;专为 XM 全系列设备状态监测模块&#xff08;X…...

AI教材生成工具推荐,低查重率为教材质量保驾护航!

教材格式的复杂性一直困扰着许多编写者。比如说&#xff0c;标题的字体大小应该设定为多少&#xff1f;参考文献是按照GB/T7714标准还是应该遵循某些出版机构的独特规范&#xff1f;习题的排版是选择单栏&#xff0c;还是双栏更合适&#xff1f;这些不同的要求让人眼花缭乱&…...

手把手拆解工业级ISP算法源码

ISP算法源码 资料最齐全&#xff0c;全网最低&#xff0c;包含Cmodel执行文件&#xff0c;可读源代码等等。 也有骗子搞成号称有Cmodel执行文件注意甄别&#xff0c;甚至宝贝描述都是抄我的 也可以在xilinx平台例化图形界面&#xff0c;方便使用 功能涵盖&#xff1a; DPC坏点缺…...

考虑集流体的 Comsol sofc固体氧化物燃料电池仿真(温度场分布,气体分布,极化曲线

考虑集流体的 Comsol sofc固体氧化物燃料电池仿真&#xff08;温度场分布&#xff0c;气体分布&#xff0c;极化曲线&#xff0c;性能曲线&#xff09;凌晨三点盯着屏幕上的温度云图&#xff0c;我手里的冰美式已经见底。集流体的边缘区域突然出现一块诡异的低温区&#xff0c;…...

N 3 串口

一、串口UART&#xff1a;Unervisal Async Recveiver Transimitter&#xff0c;通用异步收发器。&#xff08;全双工&#xff0c;串行&#xff09;RXD&#xff1a;接收信号线TXD&#xff1a;发送信号线CH340&#xff1a;电平转换芯片单工&#xff1a;通信时&#xff0c;数据接收…...

513. 找树左下角的值-day16

本地要找出树的最后一行找到最左边的值。此时大家应该想起用层序遍历是非常简单的了&#xff0c;反而用递归的话会比较难一点。我们依然还是先介绍递归法。咋眼一看&#xff0c;这道题目用递归的话就就一直向左遍历&#xff0c;最后一个就是答案呗&#xff1f;没有这么简单&…...

DO-254通读--11.0 附加考虑

11.0 附加考虑 本节提供了前几节未涵盖的设计保证附加考虑事项的指南。申请人可酌情使用这些附加考虑来满足第2节至第9节的部分目标。任何附加考虑的使用均应征得审定机构的同意。 11.1 使用先前已开发的硬件 本节讨论与使用先前已开发的硬件相关的问题。指南包括对硬件修改…...

AI写教材的秘密武器!实现低查重教材生成的实用工具推荐

撰写教材现状及 AI 工具的作用 撰写教材的进度总是会在“慢节奏”中踩到许多雷区。本来框架和资料都已经齐备&#xff0c;却在内容写作上遭遇瓶颈——一句话反复琢磨半天&#xff0c;依旧觉得表达不够准确&#xff1b;章节之间的衔接&#xff0c;总是绞尽脑汁也找不到合适的语…...

内窥镜加热器如何选择红外LED加热光源

内窥镜加热器在医疗和工业领域中扮演着关键角色&#xff0c;特别是在低温环境下需要确保内窥镜的正常工作。选择合适的红外LED加热光源对于提高内窥镜的性能和可靠性至关重要。本文将从内窥镜加热方式的发展、红外LED光源的选择、内窥镜加热器的应用案例和方案&#xff0c;以及…...

Vivado FPGA输入时钟约束

## 40mhz时钟输入 set_property PACKAGE_PIN F17 [get_ports f_clk40mhz] set_property IOSTANDARD LVCMOS33 [get_ports f_clk40mhz] create_clock -period 25.000 -name f_clk40mhz -waveform {0.000 12.500} [get_ports f_clk40mhz]前两行是引脚约束&#xff0c;后一行是时…...

昆仑通态触摸屏485通讯恒压供水程序(一拖二)

昆仑通态触摸屏485通讯ABB做的恒压供水程序&#xff0c;不需要PLC 恒压供水一拖二程序 1.触摸屏程序MCGpro版本&#xff0c;也有优盘格式文件 2.有ABB变频器一拖二设置参数及接线图纸&#xff0c;(可不用触摸屏) 3.有CAD图纸&#xff0c;二次图&#xff0c;昆仑通态的触摸屏直接…...

高德地图车机版9.1.0.600087美化包

资源编号235_高德地图车机版9.1.0.600087正式版。 更新自定义dex版本至20260311。 全新自定义修改脚本&#xff0c;修复已知bug&#xff0c;增加稳定性。 支持悬浮/原包/共存_鹰眼预警/ai预测/红绿灯播报/变灯提醒/超速提醒/天气播报等 点我下载 ❗关于系统版本&#xff1…...

【最新】OpenClaw(Clawdbot)本地6分钟搭建及使用萌新步骤

【最新】OpenClaw&#xff08;Clawdbot&#xff09;本地6分钟搭建及使用萌新步骤。OpenClaw&#xff08;Clawdbot/Moltbot&#xff09;作为开源、本地优先的AI助理框架&#xff0c;凭借724小时在线响应、多任务自动化执行、跨平台协同等核心能力&#xff0c;成为个人办公与轻量…...

多策略混合的北方苍鹰优化算法:基于立方混沌与透镜反向学习的种群初始化及最差最优策略对比原始NG...

多策略混合改进的北方苍鹰优化算法--MATLAB 改进&#xff1a; 1、立方混沌和透镜反向学习初始化种群 2、最差最优反向策略和透镜反向学习 对比原始NGO算法江湖上优化算法多如牛毛&#xff0c;今儿咱们来盘一盘这个自带猛禽气质的北方苍鹰优化算法&#xff08;Northern…...

米哈游 AI产品经理面试题精选:10道高频考题+答案解析(附PDF)

米哈游简介 米哈游(miHoYo)是全球领先的游戏开发公司,以《原神》《崩坏》系列闻名,技术实力雄厚且注重二次元文化。作为技术驱动型公司,米哈游在AI技术应用上走在行业前沿,尤其在游戏NPC智能交互、内容生成、玩家体验优化等方面有深度探索。AI产品经理岗位要求既懂技术又…...

深度拆解OpenClaw:引爆“赛博养虾”狂潮的技术内核、产业重构与暗面危机

&#x1f525;作者简介&#xff1a; 一个平凡而乐于分享的小比特&#xff0c;中南民族大学通信工程专业研究生&#xff0c;研究方向无线联邦学习 &#x1f3ac;擅长领域&#xff1a;驱动开发&#xff0c;嵌入式软件开发&#xff0c;BSP开发 ❄️作者主页&#xff1a;一个平凡而…...

联想老机型无对应 BIOS 可下载?官方解答 + 操作建议全在这

不少使用联想老款电脑的用户都会遇到这样的困惑&#xff1a;想升级 BIOS 优化设备性能、解决硬件兼容问题&#xff0c;却在官方渠道翻遍了也找不到对应机型的 BIOS 安装包&#xff0c;不知道是操作方式不对&#xff0c;还是官方已停止相关支持&#xff0c;也不敢随意下载第三方…...

Java异常与事务回滚:从Throwable顶层继承到rollbackFor陷阱的深度解析

在Java企业级开发中&#xff0c;异常处理与事务管理是保障系统稳定性的两大基石。然而&#xff0c;许多开发者在面对Transactional注解的rollbackFor属性、try-catch块中的异常处理以及Java异常体系的顶层结构时&#xff0c;往往存在认知误区。本文将从异常体系的顶层父类出发&…...

小白收藏必备:快速掌握AI Agent主流设计模式,轻松入门大模型开发

本文介绍了AI Agent的几种主流设计模式&#xff0c;包括ReAct、Plan & Execute、ReWOO、LLM Compiler、反思与增强类架构以及LATS。通过分析这些模式的核心概念、工作原理、优缺点和典型应用场景&#xff0c;帮助初学者理解AI Agent的设计思路&#xff0c;为后续深入研究Cl…...