OPENCV——RV1126+OPENCV对视频流进行膨胀、腐蚀
一、视频膨胀操作的大体流程图本章节是利用RV1126的视频流结合OPENCV的API对视频流进行膨胀然后对其进行编码保存。要完成这个功能我们首先要初始化VI、VENC的模块并且使能然后需要创建两个线程。第一个线程是opencv_vi_dliate_thread它的主要功能是获取VI原始数据并用OPENCV转换成Mat矩阵然后用dilate对其VI数据进行膨胀然后把VI数据发送到VENC编码器。第二个线程是GET_VENC_STREAM_THREAD它主要是获取H264的VENC码流数据并且保存到H264文件。二、具体代码实现// Copyright 2020 Fuzhou Rockchip Electronics Co., Ltd. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include assert.h #include fcntl.h #include getopt.h #include opencv2/imgproc.hpp #include pthread.h #include signal.h #include stdbool.h #include stdio.h #include stdlib.h #include time.h #include unistd.h // #include common/sample_common.h #include rkmedia_api.h #include opencv2/core.hpp // #include opencv2/imgoroc.hpp #include opencv2/highgui.hpp #include opencv2/opencv.hpp using namespace cv; using namespace std; #define CAMERA_PATH rkispp_scale0 #define CAMERA_ID 0 #define CAMERA_CHN 0 #define VENC_CHN 0 #define WIDTH 1920 #define HEIGHT 1080 void *opencv_vi_dliate_thread(void *args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VI, CAMERA_CHN, -1);//获取VI数据 if (!mb) { printf(Get Vi stream break...\n); break; } Mat rv1126_vi_mat Mat(HEIGHT, WIDTH, CV_8UC1, RK_MPI_MB_GetPtr(mb)); //把VI数据转换成OPENCV的Mat矩阵 Mat rv1126_vi_structure getStructuringElement(MORPH_RECT, Size(15, 15));//获取内核内核的形状是矩形长度大小是15 * 15 dilate(rv1126_vi_mat, rv1126_vi_mat, rv1126_vi_structure);//对Mat矩阵进行dilate膨胀 RK_MPI_SYS_SendMediaBuffer(RK_ID_VENC, VENC_CHN, mb);//把膨胀后的数据传输到VENC编码器 RK_MPI_MB_ReleaseBuffer(mb);//释放资源 } return NULL; } void *get_venc_stream_thread(void * args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; FILE *opencv_dliate_file fopen(test_opencv_dliate.h264, w); while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VENC, VENC_CHN, -1); if (!mb) { printf(Get Venc stream break...\n); break; } printf(Get Dlite_Venc Stream Success...\n); fwrite(RK_MPI_MB_GetPtr(mb), RK_MPI_MB_GetSize(mb), 1, opencv_dliate_file); RK_MPI_MB_ReleaseBuffer(mb); } return NULL; } int main() { int ret; VI_CHN_ATTR_S vi_chn_attr; vi_chn_attr.pcVideoNode CAMERA_PATH; // Path vi_chn_attr.u32Width 1920; // Width vi_chn_attr.u32Height 1080; // Height vi_chn_attr.enPixFmt IMAGE_TYPE_NV12; // ImageType vi_chn_attr.enBufType VI_CHN_BUF_TYPE_MMAP; // BufType vi_chn_attr.u32BufCnt 3; // Cnt vi_chn_attr.enWorkMode VI_WORK_MODE_NORMAL; // Mode ret RK_MPI_VI_SetChnAttr(CAMERA_ID, CAMERA_CHN, vi_chn_attr); if (ret) { printf(Vi Set Attr Failed.....\n); return 0; } else { printf(Vi Set Attr Success.....\n); } ret RK_MPI_VI_EnableChn(CAMERA_ID, CAMERA_CHN); if (ret) { printf(Vi Enable Attr Failed.....\n); return 0; } else { printf(Vi Enable Attr Success.....\n); } VENC_CHN_ATTR_S venc_chn_attr; memset(venc_chn_attr, 0, sizeof(VENC_CHN_ATTR_S)); venc_chn_attr.stVencAttr.u32PicWidth 1920; venc_chn_attr.stVencAttr.u32PicHeight 1080; venc_chn_attr.stVencAttr.u32VirWidth 1920; venc_chn_attr.stVencAttr.u32VirHeight 1080; venc_chn_attr.stVencAttr.imageType IMAGE_TYPE_NV12; venc_chn_attr.stVencAttr.enType RK_CODEC_TYPE_H264; venc_chn_attr.stVencAttr.u32Profile 66; venc_chn_attr.stRcAttr.enRcMode VENC_RC_MODE_H264CBR; venc_chn_attr.stRcAttr.stH264Cbr.u32Gop 25; venc_chn_attr.stRcAttr.stH264Cbr.u32BitRate 1920 * 1080 * 3; venc_chn_attr.stRcAttr.stH264Cbr.fr32DstFrameRateDen 1; venc_chn_attr.stRcAttr.stH264Cbr.fr32DstFrameRateNum 25; venc_chn_attr.stRcAttr.stH264Cbr.u32SrcFrameRateDen 1; venc_chn_attr.stRcAttr.stH264Cbr.u32SrcFrameRateNum 25; ret RK_MPI_VENC_CreateChn(VENC_CHN, venc_chn_attr); if (ret) { printf(ERROR: Create venc failed!\n); exit(0); } ret RK_MPI_VI_StartStream(CAMERA_ID, CAMERA_CHN); if (ret) { printf(start vi stream failed.....\n); } else { printf(start vi stream success.....\n); } pthread_t pid; pthread_create(pid, NULL, opencv_vi_dliate_thread, NULL);//膨胀处理线程 pthread_create(pid, NULL, get_venc_stream_thread, NULL);//获取VENC线程 while (1) { sleep(2); } RK_MPI_VENC_DestroyChn(VENC_CHN); RK_MPI_VI_DisableChn(CAMERA_ID, CAMERA_CHN); return 0; }2.1. RV1126模块初始化并启动VI工作RV1126模块的初始化包括VI模块的初始化(RK_MPI_VI_SetChnAttr)、使能VI模块(RK_MPI_VI_EnableChn)、VENC模块的初始化(RK_MPI_VENC_CreateChn)、启动VI工作(RK_MPI_VI_StartStream)。关于这方面的参数设置我们就不详细说了因为这方面的内容之前的课程已经详细说过。2.2. opencv_dliate_vi_thread线程的讲解void *opencv_vi_dliate_thread(void *args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VI, CAMERA_CHN, -1);//获取VI数据 if (!mb) { printf(Get Vi stream break...\n); break; } Mat rv1126_vi_mat Mat(HEIGHT, WIDTH, CV_8UC1, RK_MPI_MB_GetPtr(mb)); //把VI数据转换成OPENCV的Mat矩阵 Mat rv1126_vi_structure getStructuringElement(MORPH_RECT, Size(15, 15));//获取内核内核的形状是矩形长度大小是15 * 15 dilate(rv1126_vi_mat, rv1126_vi_mat, rv1126_vi_structure);//对Mat矩阵进行dilate膨胀 RK_MPI_SYS_SendMediaBuffer(RK_ID_VENC, VENC_CHN, mb);//把膨胀后的数据传输到VENC编码器 RK_MPI_MB_ReleaseBuffer(mb);//释放资源 } return NULL; }上面是opencv_dliate_vi_thread线程的具体实现首先我们要通过RK_MPI_SYS_GetMediaBuffer获取每一帧的VI视频原始数据然后把每一帧的原始数据通过OPENCV转换成Mat矩阵。Mat矩阵的转换用构造器就可以Mat rv1126_vi_mat Mat(HEIGHT, WIDTH, CV_8UC1, RK_MPI_MB_GetPtr(mb))第一个参数是HEIGHT1080第二个参数WIDTH1920第三个参数图像格式CV_8UC1第四个参数:具体的图像数据RK_MPI_MB_GetPtr(mb)。然后调用dilate的对其Mat矩阵进行膨胀这里的膨胀参数我们设置成矩形和Size(15,15)(具体的如下Mat rv1126_vi_structure getStructuringElement(MORPH_RECT, Size(15, 15));)2.3. get_venc_stream_thread线程的讲解void *get_venc_stream_thread(void * args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; FILE *opencv_dliate_file fopen(test_opencv_dliate.h264, w); while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VENC, VENC_CHN, -1); if (!mb) { printf(Get Venc stream break...\n); break; } printf(Get Dlite_Venc Stream Success...\n); fwrite(RK_MPI_MB_GetPtr(mb), RK_MPI_MB_GetSize(mb), 1, opencv_dliate_file); RK_MPI_MB_ReleaseBuffer(mb); } return NULL; }上面是get_venc_stream_thread的具体实现在这个线程里面要通过RK_MPI_SYS_GetMediaBuffer获取每一帧H264的编码数据然后用fwrite写入经过上面的编码后输出的H264文件。这个视频流更加的明亮整个视频流的细节都变大了三、视频腐蚀操作的大体流程图本章节是利用RV1126的视频流结合OPENCV的API对视频流进行腐蚀然后对其进行编码保存。要完成这个功能我们首先要初始化VI、VENC的模块并且使能然后需要创建两个线程。第一个线程是opencv_vi_erode_thread它的主要功能是获取VI原始数据并用OPENCV转换成Mat矩阵然后用erode对其VI数据进行腐蚀然后把VI数据发送到VENC编码器。第二个线程是GET_VENC_STREAM_THREAD它主要是获取H264的VENC码流数据并且保存到H264文件。四、具体代码实现#include assert.h #include fcntl.h #include getopt.h #include opencv2/imgproc.hpp #include pthread.h #include signal.h #include stdbool.h #include stdio.h #include stdlib.h #include time.h #include unistd.h // #include common/sample_common.h #include rkmedia_api.h #include opencv2/core.hpp // #include opencv2/imgoroc.hpp #include opencv2/highgui.hpp #include opencv2/opencv.hpp using namespace cv; using namespace std; #define CAMERA_PATH rkispp_scale0 #define CAMERA_ID 0 #define CAMERA_CHN 0 #define VENC_CHN 0 #define WIDTH 1920 #define HEIGHT 1080 void *opencv_vi_erode_thread(void *args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VI, CAMERA_CHN, -1);//获取VI数据 if (!mb) { printf(Get Vi Stream break.....\n); break; } Mat rv1126_mat Mat(HEIGHT, WIDTH, CV_8UC1, RK_MPI_MB_GetPtr(mb));//把VI数据转换成OPENCV的Mat矩阵 Mat rv1126_structure getStructuringElement(MORPH_RECT, Size(15, 15));//获取内核内核的形状是矩形长度大小是15 * 15 erode(rv1126_mat, rv1126_mat, rv1126_structure);//对Mat矩阵进行erode腐蚀 RK_MPI_SYS_SendMediaBuffer(RK_ID_VENC, VENC_CHN, mb);//把腐蚀后的数据传输到VENC编码器 RK_MPI_MB_ReleaseBuffer(mb);//释放资源 } return NULL; } void *get_venc_stream_thread(void *args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; FILE * opencv_erode_h264_file fopen(opencv_erode.h264, w); while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VENC, VENC_CHN, -1); if (!mb) { printf(Get Venc Stream break.....\n); break; } printf(Get ERODE_STREAM Success...\n); fwrite(RK_MPI_MB_GetPtr(mb), RK_MPI_MB_GetSize(mb), 1, opencv_erode_h264_file); RK_MPI_MB_ReleaseBuffer(mb); } return NULL; } int main() { int ret; VI_CHN_ATTR_S vi_chn_attr; vi_chn_attr.pcVideoNode CAMERA_PATH; // Path vi_chn_attr.u32Width 1920; // Width vi_chn_attr.u32Height 1080; // Height vi_chn_attr.enPixFmt IMAGE_TYPE_NV12; // ImageType vi_chn_attr.enBufType VI_CHN_BUF_TYPE_MMAP; // BufType vi_chn_attr.u32BufCnt 3; // Cnt vi_chn_attr.enWorkMode VI_WORK_MODE_NORMAL; // Mode ret RK_MPI_VI_SetChnAttr(CAMERA_ID, CAMERA_CHN, vi_chn_attr); if (ret) { printf(Vi Set Attr Failed.....\n); return 0; } else { printf(Vi Set Attr Success.....\n); } ret RK_MPI_VI_EnableChn(CAMERA_ID, CAMERA_CHN); if (ret) { printf(Vi Enable Attr Failed.....\n); return 0; } else { printf(Vi Enable Attr Success.....\n); } VENC_CHN_ATTR_S venc_chn_attr; memset(venc_chn_attr, 0, sizeof(VENC_CHN_ATTR_S)); venc_chn_attr.stVencAttr.u32PicWidth 1920; venc_chn_attr.stVencAttr.u32PicHeight 1080; venc_chn_attr.stVencAttr.u32VirWidth 1920; venc_chn_attr.stVencAttr.u32VirHeight 1080; venc_chn_attr.stVencAttr.imageType IMAGE_TYPE_NV12; venc_chn_attr.stVencAttr.enType RK_CODEC_TYPE_H264; venc_chn_attr.stVencAttr.u32Profile 66; venc_chn_attr.stRcAttr.enRcMode VENC_RC_MODE_H264CBR; venc_chn_attr.stRcAttr.stH264Cbr.u32Gop 25; venc_chn_attr.stRcAttr.stH264Cbr.u32BitRate 1920 * 1080 * 3; venc_chn_attr.stRcAttr.stH264Cbr.fr32DstFrameRateDen 1; venc_chn_attr.stRcAttr.stH264Cbr.fr32DstFrameRateNum 25; venc_chn_attr.stRcAttr.stH264Cbr.u32SrcFrameRateDen 1; venc_chn_attr.stRcAttr.stH264Cbr.u32SrcFrameRateNum 25; ret RK_MPI_VENC_CreateChn(VENC_CHN, venc_chn_attr); if (ret) { printf(ERROR: Create venc failed!\n); exit(0); } ret RK_MPI_VI_StartStream(CAMERA_ID, CAMERA_CHN); if (ret) { printf(ERROR: Start Vi failed.....!\n); return -1; } else { printf(SUCCESS: Start Vi success.....!\n); } pthread_t pid; pthread_create(pid, NULL, opencv_vi_erode_thread, NULL);//腐蚀处理线程 pthread_create(pid, NULL, get_venc_stream_thread, NULL);//获取VENC线程 while (1) { sleep(2); } RK_MPI_VENC_DestroyChn(VENC_CHN); RK_MPI_VI_DisableChn(CAMERA_ID, CAMERA_CHN); return 0; }4.1. RV1126模块初始化并启动VI工作RV1126模块的初始化包括VI模块的初始化(RK_MPI_VI_SetChnAttr)、使能VI模块(RK_MPI_VI_EnableChn)、VENC模块的初始化(RK_MPI_VENC_CreateChn)、启动VI工作(RK_MPI_VI_StartStream)。关于这方面的参数设置我们就不详细说了因为这方面的内容之前的课程已经详细说过。2.2. opencv_erode_vi_thread线程的讲解void *opencv_vi_erode_thread(void *args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VI, CAMERA_CHN, -1);//获取VI数据 if (!mb) { printf(Get Vi Stream break.....\n); break; } Mat rv1126_mat Mat(HEIGHT, WIDTH, CV_8UC1, RK_MPI_MB_GetPtr(mb));//把VI数据转换成OPENCV的Mat矩阵 Mat rv1126_structure getStructuringElement(MORPH_RECT, Size(15, 15));//获取内核内核的形状是矩形长度大小是15 * 15 erode(rv1126_mat, rv1126_mat, rv1126_structure);//对Mat矩阵进行erode腐蚀 RK_MPI_SYS_SendMediaBuffer(RK_ID_VENC, VENC_CHN, mb);//把腐蚀后的数据传输到VENC编码器 RK_MPI_MB_ReleaseBuffer(mb);//释放资源 } return NULL; }上面是opencv_erode_vi_thread线程的具体实现首先我们要通过RK_MPI_SYS_GetMediaBuffer获取每一帧的VI视频原始数据然后把每一帧的原始数据通过OPENCV转换成Mat矩阵。Mat矩阵的转换用构造器就可以Mat rv1126_mat Mat(HEIGHT, WIDTH, CV_8UC1, RK_MPI_MB_GetPtr(mb));第一个参数是HEIGHT1080第二个参数WIDTH1920第三个参数图像格式CV_8UC1第四个参数:具体的图像数据RK_MPI_MB_GetPtr(mb)。然后调用erode的对其Mat矩阵进行腐蚀这里的膨胀参数我们设置成矩形和Size(15,15)具体的(Mat rv1126_structure getStructuringElement(MORPH_RECT, Size(15, 15)))。2.3. get_venc_stream_thread线程的讲解void *get_venc_stream_thread(void *args) { pthread_detach(pthread_self()); MEDIA_BUFFER mb NULL; FILE * opencv_erode_h264_file fopen(opencv_erode.h264, w); while (1) { mb RK_MPI_SYS_GetMediaBuffer(RK_ID_VENC, VENC_CHN, -1); if (!mb) { printf(Get Venc Stream break.....\n); break; } printf(Get ERODE_STREAM Success...\n); fwrite(RK_MPI_MB_GetPtr(mb), RK_MPI_MB_GetSize(mb), 1, opencv_erode_h264_file); RK_MPI_MB_ReleaseBuffer(mb); } return NULL; }上面是get_venc_stream_thread的具体实现在这个线程里面要通过RK_MPI_SYS_GetMediaBuffer获取每一帧H264的编码数据然后用fwrite写入。经过上面的编码后我们来看看输出的H264文件。这个视频流明显更加暗淡并且细节都变小了

相关新闻

基于ATtiny817与SD卡的低功耗温度记录仪设计与实现

基于ATtiny817与SD卡的低功耗温度记录仪设计与实现

1. 项目缘起与核心价值最近在整理工作室的物料,翻出来一堆闲置的ATtiny817芯片和几张老旧的MicroSD卡。看着这些“电子垃圾”,我就在想,能不能用它们干点正经事?比如,做一个极简、低功耗、能长时间独立工作的温度记录仪…

2026/7/1 11:29:11阅读更多 →
LPrint:重新定义标签打印的技术架构与无驱动打印新范式

LPrint:重新定义标签打印的技术架构与无驱动打印新范式

LPrint:重新定义标签打印的技术架构与无驱动打印新范式 【免费下载链接】lprint A Label Printer Application 项目地址: https://gitcode.com/gh_mirrors/lp/lprint 在零售、物流、制造等行业中,标签和收据打印是业务运营的核心环节。传统打印解…

2026/7/1 11:29:11阅读更多 →
【ChatGPT隐私保护终极指南】:20年安全专家亲授5大高危数据泄露场景与零信任防护实战方案

【ChatGPT隐私保护终极指南】:20年安全专家亲授5大高危数据泄露场景与零信任防护实战方案

更多请点击: https://codechina.net 第一章:ChatGPT隐私保护的认知重构与风险觉醒 当用户向ChatGPT输入“我的身份证号是11010119900307281X,帮我生成一份购房贷款评估报告”时,看似便捷的交互背后,数据已悄然脱离个人…

2026/7/1 11:29:11阅读更多 →
LX Music Desktop:一站式开源音乐播放器的革命性体验

LX Music Desktop:一站式开源音乐播放器的革命性体验

LX Music Desktop:一站式开源音乐播放器的革命性体验 【免费下载链接】lx-music-desktop 一个基于 Electron 的音乐软件 项目地址: https://gitcode.com/GitHub_Trending/lx/lx-music-desktop 在数字音乐时代,我们常常面临一个困境:音…

2026/7/1 12:39:48阅读更多 →
SLO2016与PIC32MZ2048EFM144在工业通信中的高效组合

SLO2016与PIC32MZ2048EFM144在工业通信中的高效组合

1. SLO2016与PIC32MZ2048EFM144的黄金组合解析 在工业通信和嵌入式系统领域,数据传输的可靠性与实时性始终是开发者面临的核心挑战。SLO2016作为一款专业级串行通信协议芯片,与Microchip公司推出的PIC32MZ2048EFM144高性能微控制器相结合,构成…

2026/7/1 12:39:48阅读更多 →
Linux/Windows双平台VMware Tools安装失败全场景排查手册(含ESXi 8.0+兼容性避坑清单)

Linux/Windows双平台VMware Tools安装失败全场景排查手册(含ESXi 8.0+兼容性避坑清单)

更多请点击: https://kaifayun.com 第一章:VMware Tools安装失败的典型现象与诊断定位 VMware Tools 是提升虚拟机性能与集成度的关键组件,但其安装过程常因环境差异而失败。常见现象包括:安装程序无响应、进度条卡在“正在配置驱…

2026/7/1 12:39:48阅读更多 →
终极解决方案:用Obsidian Better Export PDF插件高效管理知识输出

终极解决方案:用Obsidian Better Export PDF插件高效管理知识输出

终极解决方案:用Obsidian Better Export PDF插件高效管理知识输出 【免费下载链接】obsidian-better-export-pdf Obsidian PDF export enhancement plugin 项目地址: https://gitcode.com/gh_mirrors/ob/obsidian-better-export-pdf 你是否曾为将Obsidian笔记…

2026/7/1 12:39:48阅读更多 →
Space Thumbnails:Windows资源管理器的3D模型预览革命,让文件浏览进入立体时代

Space Thumbnails:Windows资源管理器的3D模型预览革命,让文件浏览进入立体时代

Space Thumbnails:Windows资源管理器的3D模型预览革命,让文件浏览进入立体时代 【免费下载链接】space-thumbnails Generates preview thumbnails for 3D model files. Provide a Windows Explorer extensions that adds preview thumbnails for 3D mode…

2026/7/1 12:39:48阅读更多 →
3分钟解锁中兴光猫:永久Telnet权限获取实战指南

3分钟解锁中兴光猫:永久Telnet权限获取实战指南

3分钟解锁中兴光猫:永久Telnet权限获取实战指南 【免费下载链接】zteOnu A tool that can open ZTE onu device factory mode 项目地址: https://gitcode.com/gh_mirrors/zt/zteOnu 你是否曾经因为无法访问光猫的高级设置而感到困扰?想要配置端口…

2026/7/1 12:34:47阅读更多 →
AI Coding 六个月真实ROI账本:产品经理的血泪教训,研发的冷静忠告

AI Coding 六个月真实ROI账本:产品经理的血泪教训,研发的冷静忠告

6个月前的2025年12月,Boris Cherny 公开宣布自己卸载了 IDE。一时间,Vibe Coding 成了全行业最热的话题。6个月后,当我们回过头来拉一份真实账本,发现事情远没有"一句话生成一个App"那么浪漫。本文从产品经理和研发两个…

2026/7/1 4:42:14阅读更多 →
审计来了,数据权限全开——审计走了,怎么确保权限全部关掉?

审计来了,数据权限全开——审计走了,怎么确保权限全部关掉?

引言:审计结束三个月了,审计员的权限还没关某城商行每年按照监管要求开展至少一次数据安全审计。审计期间,内审部门需要抽样检查各类业务数据——交易流水、客户信息、员工操作日志、权限配置记录。这些数据分布在不同系统中,审计…

2026/7/1 5:19:01阅读更多 →
YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

如果你在部署 YOLOv8 时,发现推理速度只有可怜的 1-2 FPS,而别人的演示视频却能跑到 30 FPS 以上,那么问题很可能不在模型本身,而在于你的整个处理链路。很多开发者拿到一个训练好的 YOLOv8 模型后,会直接使用官方示例…

2026/7/1 0:01:44阅读更多 →
Coze与Dify对比指南:低代码AI应用开发从入门到实战

Coze与Dify对比指南:低代码AI应用开发从入门到实战

1. 从零到一:为什么你需要了解 Coze 和 Dify?如果你对 AI 应用开发感兴趣,但一看到“大模型”、“智能体”、“工作流”这些词就头疼,觉得门槛太高,那这篇文章就是为你准备的。很多开发者,包括我自己&#…

2026/7/1 0:01:44阅读更多 →
AI生图工具怎么选?2026年6月版实测对比

AI生图工具怎么选?2026年6月版实测对比

做自媒体的朋友应该都有体会:配图一直是个让人头疼的问题。2026年,AI生图工具已经非常成熟了,但工具太多反而不知道怎么选。以下是截至2026年6月我对主流AI生图工具的实测对比。Midjourney V8.1:速度之王2026年6月11日&#xff0c…

2026/7/1 0:01:44阅读更多 →
YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

YOLOv8推理性能优化:从1.2FPS到35FPS的全链路加速实践

如果你在部署 YOLOv8 时,发现推理速度只有可怜的 1-2 FPS,而别人的演示视频却能跑到 30 FPS 以上,那么问题很可能不在模型本身,而在于你的整个处理链路。很多开发者拿到一个训练好的 YOLOv8 模型后,会直接使用官方示例…

2026/7/1 0:01:44阅读更多 →
Coze与Dify对比指南:低代码AI应用开发从入门到实战

Coze与Dify对比指南:低代码AI应用开发从入门到实战

1. 从零到一:为什么你需要了解 Coze 和 Dify?如果你对 AI 应用开发感兴趣,但一看到“大模型”、“智能体”、“工作流”这些词就头疼,觉得门槛太高,那这篇文章就是为你准备的。很多开发者,包括我自己&#…

2026/7/1 0:01:44阅读更多 →
AI生图工具怎么选?2026年6月版实测对比

AI生图工具怎么选?2026年6月版实测对比

做自媒体的朋友应该都有体会:配图一直是个让人头疼的问题。2026年,AI生图工具已经非常成熟了,但工具太多反而不知道怎么选。以下是截至2026年6月我对主流AI生图工具的实测对比。Midjourney V8.1:速度之王2026年6月11日&#xff0c…

2026/7/1 0:01:44阅读更多 →