基于PTP实现主机与相机系统时钟同步功能

基于PTP实现主机与相机系统时钟同步功能

  • 一、PTP简介
  • 二、工业相机PTP功能支持
  • 三、工业相机时间戳介绍
    • 3.1基本概念
    • 3.2海康工业相机时间戳介绍
      • 3.2.1相机参数时间戳
      • 3.2.2图像嵌入式时间戳
      • 3.2.3相机event事件时间戳
      • 3.2.4各种时间戳的时序关系
      • 3.2.5通过工业相机SDK获取相机时间戳
  • 四、通过PTPD实现主机与相机系统时钟同步
    • 4.1 ptpd服务部署
    • 4.2 PTP时钟同步测试

一、PTP简介

PTP概览: PTP是一种网络时间同步协议,设计用于实现局域网络中设备间的高精度时间同步,通常能达到亚微秒乃至纳秒级的同步精度。它主要服务于对时间同步有严格要求的应用场景,如金融交易、电信网络、科学研究、工业自动化等。
核心特点:

  • 硬件级实现:PTP利用网络接口卡(NIC)的硬件时间戳功能直接在数据链路层(MAC层)进行时间戳标记,减少了操作系统和协议栈带来的延迟,提升了同步精度。
  • 主从架构:网络中存在一个或多个主时钟(通常通过Best Master Clock算法选举),其他设备作为从时钟,根据主时钟的信号调整自己的时间。
  • 时间延迟补偿:PTP协议通过测量和补偿消息的传播延迟,包括往返时间(Round Trip Time, RTT)的计算,来精确调整从时钟的时间。
  • PTPv2(IEEE 1588-2008):相较于最初的PTPv1,PTPv2增加了对多种时钟类型的支持、改善了时钟模型、增强了时间戳机制,并引入透明时钟(Transparent Clocks)概念,进一步提高了同步精度。

PTP作为一种高精度时间同步技术,能够实现亚微秒乃至约30纳秒的精准同步,但这需要网络中的交换机等节点具备PTP支持,以确保纳秒级的同步精度。尽管传统的NTP能满足多数应用场景,提供大约5毫秒内的时钟同步,非极高端需求外,PTP并非标配之选,因其对硬件和网络配置有着更高要求。
PTP协议在IEEE 1588标准中定义,主要通过两种方式在网络中传输数据包:

  • 基于以太网的数据链路层(MAC层)直接封装:这是PTPv2推荐的方式之一,允许时间同步消息直接在数据链路层传输,绕过了网络层(IP)和传输层(TCP/UDP),从而减少延迟和抖动。这种方式下,PTP报文类型和时间戳在MAC层处理,减少了操作系统和协议栈引入的不确定性。
  • 基于UDP/IP的承载:当直接MAC层传输不可行或不适用时,PTP可以利用UDP/IP协议作为传输载体。PTP事件消息(如Sync和Follow_Up)通常使用UDP端口319,而PTP通用(或普通)消息(如Delay_Req和Delay_Resp)使用UDP端口320。这种方式虽然方便与现有网络基础设施集成,但由于经过了完整的网络协议栈,可能引入更多的时延和不确定性,降低了同步精度。

而想要在UDP上承载PTP,需要确保网络设备(如交换机和终端设备)能够支持PTP,并正确配置网络接口以使用UDP端口转发PTP消息。在软件层面,可以利用编程语言(如C、C++、Python或Go)中的网络编程库来发送和接收这些UDP数据包,并根据PTP协议规范处理时间戳和同步算法。

二、工业相机PTP功能支持

海康工业网口相机支持PTP时钟同步功能,可以通过PTP协议,实现多场景的高精度时间戳同步,例如:多个工业相机之间、工业相机与主机时间、工业相机与硬件授时模块(北斗/GPS授时服务器)等.
海康相机可以通过简单的参数配置,在参数配置Transport layer下启用IEEE 1588 V2(即PTP v2)功能,通过设置相机为主时钟(Master)或从时钟(Slave),并利用交换机等网络设备实现纳秒级的同步拍照。因此,在适当的网络配置下,海康工业相机能够利用PTP功能满足对时间同步有严格要求的应用场景。
在这里插入图片描述
海康工业相机支持的PTP状态模式如下:

  • Master(主钟态): 成为Master的PTP设备承担起为网络中的其他设备提供时间基准的责任。它会定期广播时间同步消息(Sync)和对准消息(Follow_Up),帮助网络中的Slave设备与其时间保持一致。Master通常由网络中的设备基于Best Master Clock (BMC)算法选举产生,考虑到了时钟的精度、稳定性等多种因素。
  • Slave(从钟态): 当一个PTP设备确定自身应遵循网络中的另一台设备(即Master)作为时间参考时,它会转变为Slave状态。在此状态下,设备会积极接收来自Master的同步和跟随(Delay_Resp)消息,通过这些信息调整自己的时间,以尽量减小与Master之间的时间偏差,实现精确的时间同步。
  • Uncalibrated(未校准态): 当一个PTP时钟刚开始运行或刚被重置时,它通常处于未校准态。在这个状态下,时钟尚未完成与网络中其他时钟的对齐过程,因此其时间和频率的准确性无法保证。设备在这一阶段主要是在收集网络信息、识别其他PTP设备,并准备进入更高级的状态进行时间同步。
  • Passive(被动态): 被动态是指PTP时钟虽然监听网络上的同步和对准消息,但并不参与主时钟的选择过程,也不进行任何时间调整。它是一个观察者角色,可能用于监控网络状况或准备进入Slave或Master状态而不立即参与同步过程。
    需要注意的是,这些状态之间的转换是动态的,依据设备的性能、网络条件以及PTP协议的内部机制来决定。PTP协议的设计旨在通过这些状态的有序变迁,确保网络中的所有时钟能够高效、准确地达到时间同步

三、工业相机时间戳介绍

工业相机中的时间戳是一种记录图像获取精确时刻的功能,它是工业视觉系统和高精度测量应用中的核心要素。以下是对工业相机时间戳的详细介绍:

3.1基本概念

时间戳指的是附着在每一帧图像数据上的时间信息,通常表示相对于相机内部时钟的相对时间(非UTC时间)。这个时间戳使得用户可以准确知道每幅图像的捕获时间,对于需要精确同步多相机系统、分析动态过程或与外部事件关联的应用至关重要。
a) 实现机制

  • 硬件时间戳:工业相机通常内置高精度振荡器(如温补晶振、铷钟或铯钟)来生成时间基准,确保即使在没有外部时间源的情况下也能维持较高的时间精度。图像传感器在捕获图像的同时,硬件直接在图像数据包中嵌入时间戳,这种机制减少了软件处理的延迟,提高了时间记录的即时性和准确性。
  • 软件时间戳:在一些应用场景中,时间戳也可以通过软件方式在图像数据处理链路的后期添加。虽然这种方法简便,但由于涉及到软件处理流程,相较于硬件时间戳可能会引入额外的延迟。
    b) 精度与同步
    工业相机的时间戳精度可达到微秒甚至纳秒级,这对于需要极高时间分辨率的应用(如高速运动分析、机器视觉引导的机器人操作)至关重要。
    为了在多相机系统中实现精确同步,可以使用诸如 Precision Time Protocol (PTP) 或 GenICam的Time Stamp over Ethernet (ToE) 功能,通过网络基础设施(交换机网络)来同步所有相机的时间基准。
    c) 应用场景
    • 生产制造:在自动化装配线中,时间戳帮助追踪每个部件通过各个检查点的确切时间,用于质量控制和过程优化。
    • 运动捕捉:在体育分析、生物力学研究中,高精度时间戳确保了动作的精确计时和分析。
    • 科学研究:如天文观测、粒子物理实验中,时间戳对于精确记录和分析瞬时事件至关重要。
    • 安全监控:在安全和监控应用中,时间戳是事件重建和证据记录的关键。

3.2海康工业相机时间戳介绍

海康工业相机支持多种途径获取相机内部时间戳,主要分类如下:

  • 相机参数时间戳:用户主动读取相机参数TimestampValue,获取访问当前参数时刻的相机内部时间戳
  • 图像嵌入式时间戳:产生图像前,相机内部自动记录时间,通过嵌入到图像帧信息内部,方便用户获取
  • Event事件时间戳:记录相机内部各种事件时刻的时间戳

三种类型的时间戳,都是基于相机内部时钟产生相对时间,由硬件晶振生成时间基准,开始时间一般为相机上电时刻

3.2.1相机参数时间戳

相机参数说明
GevTimestampTickFrequency相机时间戳晶振频率,其倒数即为时间戳的单位,如时钟频率为 100MHz 时相机时间戳单位为 10ns,不同相机该频率可能不同,需要结合该参数具体值来确认
GevTimestampControlLatch获取当前的时间戳值,时间戳值并不会自动更新,使用时需要执行该命令进 行获取
GevTimestampValue时间戳值,需搭配GevTimestampControlLatch获取,本身不会主动更新

3.2.2图像嵌入式时间戳

图像嵌入式时间戳,用于记录相机的拍照时刻。
通常需要结合sdk接口,在_MV_FRAME_OUT_INFO_EX_帧信息结构体获取得到;
在这里插入图片描述
也可以通过Wireshark抓包,获取相机的图像数据流GVSP包,在leader头包中获取;
在这里插入图片描述
在这里插入图片描述

3.2.3相机event事件时间戳

海康工业相机支持多种事件,每一种事件产生,都会有时间戳记录

事件类型简介
Acquisition Start采集开始
Acquisition End采集结束
Frame Start帧开始
Frame End帧结束
Frame Burst Start帧触发开始
Frame Burst End帧触发结束
Exposure Start曝光开始
Exposure End曝光结束
Line0 Rising EdgeLine 0 上升沿
Line0 Falling EdgeLine 0 下降沿
Frame Start Over Trigger帧开始过触发
Over Run过载
Stream Transfer Overflow相机缓存内图像被覆盖
Frame Trigger Wait帧触发等待,相机可被触发时输出event
Software Active软触发有效
Image Error图像错误

在MVS中,设置操作方法:

  1. 在相机参数中,开启所要选择的事件,以Exposure Start事件为例
    在这里插入图片描述
  2. 开始相机取流曝光,在事件监视器中,获取事件信息
    在这里插入图片描述

3.2.4各种时间戳的时序关系

以相机常用的时间戳为例,梳理了下工业相机比较场景的时间戳关系,以相机相对时间为基准,具体如下图:
在这里插入图片描述
3. Line0RisingEdge:触发信号通过line0给到相机
4. FrameStart:图像输出开始
5. nDevTimeStamp:时间戳打包到leader包时刻
6. ExposureStart:sensor开始曝光时刻
7. ExposureEnd:sensor结束曝光时刻
8. nHoststamp: 图像头包到达主机时间,主机时间,非相机时间戳
9. FrameEnd:相机端图像输出结束

3.2.5通过工业相机SDK获取相机时间戳

  1. 事件时间戳
// ch:开启相机Event功能 | en:Set Event  On
char const *nEvent_ID[]={"Line0RisingEdge","FrameStart","FrameEnd", "ExposureStart","ExposureEnd"};
for(int i=0;i<sizeof(nEvent_ID)/sizeof(nEvent_ID[0]);i++)
{nRet = MV_CC_SetEnumValueByString(handle, "EventSelector",nEvent_ID[i]);if (MV_OK != nRet){printf("Set Event Selector fail! nRet [0x%x],nEvent_ID:%d\n", nRet,i);}nRet = MV_CC_SetEnumValueByString(handle, "EventNotification", "On");if (MV_OK != nRet){printf("Set Event Notification fail! nRet [0x%x]\n", nRet);}// ch:注册event事件回调函数nRet = MV_CC_RegisterEventCallBackEx(handle, nEvent_ID[i],EventCallBack, handle);if (MV_OK != nRet){printf("Register Event CallBack fail! nRet [0x%x]\n", nRet);}	
}		

相机事件配置与事件回调函数注册

void __stdcall EventCallBack(MV_EVENT_OUT_INFO * pEventInfo, void* pUser)
{if (pEventInfo){int64_t nBlockId = pEventInfo->nBlockIdHigh;nBlockId = (nBlockId << 32) + pEventInfo->nBlockIdLow;int64_t nTimestamp = pEventInfo->nTimestampHigh;nTimestamp = (nTimestamp << 32) + pEventInfo->nTimestampLow;printf("EventName[%s],Timestamp[" "%" PRIu64 "]\n",pEventInfo->EventName,nTimestamp);}
}

事件回调函数内部处理

  1. 图像时间戳
void __stdcall ImageCallBackEx(unsigned char * pData, MV_FRAME_OUT_INFO_EX* pFrameInfo, void* pUser)
{uint64_t m_DevTimeStamp=0;//设备产生图像的时间int64_t  m_HostTimeStamp =0;//图像头包到达主机时间if (pFrameInfo){m_DevTimeStamp = pFrameInfo->nDevTimeStampHigh;m_DevTimeStamp = (m_DevTimeStamp << 32)+pFrameInfo->nDevTimeStampLow;m_HostTimeStamp=pFrameInfo->nHostTimeStamp;printf("GetOneFrame, Width[%d], Height[%d], nFrameNum[%d],nDevTimeStamp[" "%" PRIu64 "],nHostTimeStamp[" "%" PRIu64 "]\n",pFrameInfo->nWidth, pFrameInfo->nHeight, pFrameInfo->nFrameNum,m_DevTimeStamp,m_HostTimeStamp);}
}

通过回调函数取流并通过帧信息结构体中获取nDevTimeStamp、nHoststamp
3. 相机参数时间戳获取

//获取相机当前时间戳
//仅网口相机支持
MVCC_INTVALUE_EX nGevTimestampValue;
nRet = MV_CC_SetCommandValue(handle, "GevTimestampControlLatch");
if (MV_OK != nRet)
{printf("Gev TimestampControlLatch fail! nRet [%x]\n", nRet);//break; 
}else
{nRet = MV_CC_GetIntValueEx(handle,"GevTimestampValue",&nGevTimestampValue);if (MV_OK == nRet){printf("Get GevTimestampValue =  [%lld] Success!\n",(long long)nGevTimestampValue.nCurValue);}
}

四、通过PTPD实现主机与相机系统时钟同步

4.1 ptpd服务部署

参考文章:ptpd安装部署
ptpd开源链接:github
ptpd通常指的是实现Precision Time Protocol (PTP) 协议的守护进程,用于在计算机网络中提供高精度的时间同步服务;

  1. 查看网卡与ip
1|sudo apt-get install net-tools
2|ifconfig

2、查看网卡支持项

1|sudo apt-get install ethtool
2|sudo ethtool -T ens33

在这里插入图片描述
软件时间戳需要包括参数

SOF_TIMESTAMPING_SOFTWARE
SOF_TIMESTAMPING_TX_SOFTWARE
SOF_TIMESTAMPING_RX_SOFTWARE

硬件时间戳需要包括参数

SOF_TIMESTAMPING_RAW_HARDWARE
SOF_TIMESTAMPING_TX_HARDWARE
SOF_TIMESTAMPING_RX_HARDWARE
  1. 查看NTP(网络时间同步)状态,并关闭
timedatectl  status
sudo timedatectl set-ntp false

在这里插入图片描述

  1. ptpd安装部署
sudo apt-get install ptpd
  1. 选定主机网卡ens33,开启ptpd作为主时钟master
sudo ptpd -M -i ens33

开启ptpd后,可在进程中发现ptpd相关进程
在这里插入图片描述

  1. 主机作为从时钟(相机或其他设备作为主设备)
sudo ptpd -g -i ens33
注:步骤4和5加入-C参数的话,会在前台运行,并打印输出,如在主时钟端:

例:sudo ptpd -M -C -i ens33

在这里插入图片描述
7. 关闭进程
测试结束时,可直接使用kill命令,结束进程中的ptpd,恢复NTP服务

sudo pkill ptpd
sudo timedatectl set-ntp true

4.2 PTP时钟同步测试

PTP时钟同步,可以搭配不同的硬件有多种Master、Slave组合,如下表所示

MasterSlave
电脑主机工业相机、硬件同步设备、电脑主机
工业相机电脑主机、工业相机、硬件同步设备
硬件同步设备(GPS等)工业相机、电脑主机、硬件同步设备

本文主要基于PTPD介绍表格中第一种情况,即Ubuntu主机,作为Master、工业相机作为Slave模式
测试方法:

  1. Ubuntu开启Master模式
sudo ptpd -M -i ens33
  1. 工业相机开启1588模式
    在这里插入图片描述
  2. 使用工业相机时间戳,测试同步效果如下
    在这里插入图片描述
    按照时间维度,对时间戳进行排序与计算
时间戳类型时间戳ID时间戳时间差us(100MHz)ns说明
EventLine0RisingEdge1716878696804977842///
EventFrameStart171687869680499002212.1812180FrameStart- Line0RisingEdge
ImagenDevTimeStamp17168786968049900820.0660nDevTimeStamp- FrameStart
EventExposureStart17168786968049911821.11100ExposureStart- nDevTimeStamp
EventExposureEnd17168786968106586225667.445667440ExposureEnd- ExposureStart
HostnHostTimeStamp1716878696811341.378341378nHostTimeStamp- ExposureEnd
EventFrameEnd171687869683618655225186.55225186552FrameEnd- nHostTimeStamp

HostTimeStamp与相机的各种时间戳时间单位基本一致
关闭ptpd进程服务,相机断电重新测试

sudo pkill ptpd

在这里插入图片描述

时间戳类型时间戳ID时间戳时间差us(100MHz)ns说明
EventLine0RisingEdge32279293504138///
EventFrameStart322792935092385.15100FrameStart- Line0RisingEdge
ImagenDevTimeStamp322792935092980.0660nDevTimeStamp- FrameStart
EventExposureStart322792935103981.11100ExposureStart- nDevTimeStamp
EventExposureEnd322792991779705667.5725667572ExposureEnd- ExposureStart
HostnHostTimeStamp1716872579325///
EventFrameEnd3227932470638125528.4125528411FrameEnd- ExposureEnd

关闭ptpd服务后,主机时间nHostTimeStamp不与相机时间同步

其他:

  1. 如需实现工业相机与工业相机之间PTP同步,只需要选定一台相机做master,先开启1588参数使能接口,其他相机后开启,会自动协商成slave模式;如需实现1588触发同步功能,建议参考海康工业相机ActionCommand功能
  2. 如需在windows上实现ptpd等类似功能,需要找能够在windows上运行的ptp开源项目;或者使用硬件设备作为master,相机只需要作为slave在windows上面运行
    测试代码基于ubuntu 系统,海康工业相机示例程序Grab_ImageCallback修改,可自行下载测试

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

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

相关文章

JS实现彩色图片转换为黑白图片

1. 使用 Canvas 研究 canvas 时发现一个有趣的现象——将彩色图片巧妙地转换为黑白图片。以下是实现这一功能的简洁代码示例&#xff1a; <div style"display: flex"><img src"./panda.jpeg" /><button onclick"change()">转…

香港Web3媒体:Techub News

Techub News&#xff1a;香港领先&#xff0c;世界一流的科技媒体平台 在数字化时代&#xff0c;Web3技术的崛起为媒体行业注入了新的活力。作为香港领先的Web3媒体平台&#xff0c;Techub News凭借其专业的团队、丰富的资源和创新的业务模式&#xff0c;成为了行业内的佼佼者。…

Idea java.lang.RuntimeException: java.lang.OutOfMemoryError: Java heap space 解决

咱们平时的开发过程中一定会进行本地调试&#xff0c;今天我也是安装了新的idea之后拉了一个比较大的项目进行本地调试的时候报错。报错信息如下&#xff1a; java: java.lang.OutOfMemoryError: Java heap space java.lang.RuntimeException: java.lang.OutOfMemoryError: Ja…

浅谈网络安全态势感知

前言 网络空间环境日趋复杂&#xff0c;随着网络攻击种类和频次的增加&#xff0c;自建强有力的网络安全防御系统成为一个国家发展战略的一部分&#xff0c;而网络态势感知是实现网络安全主动防御的重要基础和前提。 什么是网络安全态势感知&#xff1f; 态势感知一词来源于对…

【NumPy】全面解析mean函数:高效计算数组平均值的方法

&#x1f9d1; 博主简介&#xff1a;阿里巴巴嵌入式技术专家&#xff0c;深耕嵌入式人工智能领域&#xff0c;具备多年的嵌入式硬件产品研发管理经验。 &#x1f4d2; 博客介绍&#xff1a;分享嵌入式开发领域的相关知识、经验、思考和感悟&#xff0c;欢迎关注。提供嵌入式方向…

【class19】人工智能初步---语音识别(5)

【class19】 上节课&#xff0c;我们学习了&#xff1a;语音识别模型的结构和原理&#xff0c;同时调用创建好的AipSpeech客户端实现了语音转文字功能。 本节课&#xff0c;我们将初识字幕&#xff0c;学习这些知识点&#xff1a;1. srt字幕 2. 获取时间数据 …

Java开发-面试题-0001-String、StringBuilder、StringBuffer的区别

Java开发-面试题-0001-String、StringBuilder、StringBuffer的区别 更多内容欢迎关注我&#xff08;持续更新中&#xff0c;欢迎Star✨&#xff09; Github&#xff1a;CodeZeng1998/Java-Developer-Work-Note 技术公众号&#xff1a;CodeZeng1998&#xff08;纯纯技术文&am…

【2024.5.29数据库MYSQL史上最详细基础学习汇总】

初识数据库 什么是数据库: DB的全称是database,即数据库的意思。数据库实际上就是一个文件集合,是一个存储数据的仓库,数据库是按照特定的格式把数据存储起来,用户可以对存储的数据进行增删改查操作; 什么是关系型数据库(SQL)? 关系型数据库是依据关系模型来创建的…

如何恢复被盗的加密货币?

本世纪&#xff0c;网络犯罪的首要目标是加密货币。 这要归功于加密货币的日益普及和价值&#xff0c;网络犯罪分子已经认识到经济收益的潜力&#xff0c;并将重点转向利用这种数字资产中的漏洞。 在今天的文章中&#xff0c;我们将讨论加密货币恢复和被盗加密货币恢复。 我们…

IPFoxy Tips:海外代理IP适用的8个跨境出海业务

在当今数字化时代&#xff0c;互联网已经成为商业和个人生活不可或缺的一部分。IP代理作为出海业务的神器之一&#xff0c;备受跨境出海业务人员关注。IPFoxy动态、静态纯净代理IP也根据业务需求的不同&#xff0c;分为静态住宅、动态住宅、静态IPv4、静态IPv6四种类型代理。那…

【软件测试】LoadRunner参数化属性设置_单个参数

目录 为什么使用参数化属性详解Select next rowSequential&#xff08;顺序&#xff09;Random&#xff08;随机&#xff09;Unique&#xff08;唯一&#xff09;Same line as XXX&#xff08;和XXX属性的取值方式一样&#xff09; Update value onEach iteration&#xff08;每…

HDTune和CrystalDiskInfo硬盘检测S.M.A.R.T.参数当前值最差值阈值

高亮颜色说明&#xff1a;突出重点 个人觉得&#xff0c;&#xff1a;待核准个人观点是否有误 高亮颜色超链接 文章目录 S.M.A.R.T.监控技术磁盘健康状态监测,硬盘检测硬盘检测工具 HD Tune硬盘检测工具 CrystalDiskInfo 当前值最差值阈值原始值的含义二级标题待补充待补充 开头…

RedHat9 | 配置与管理DNS服务器

一、 知识预备 1、DNS服务器的分类 主DNS服务器 主DNS服务器复制维护所管辖域的域名服务信息&#xff0c;它从域管理员构造的本地磁盘文件中的加载域信息。该文件包含服务器具有管理权的的一部分域结构的精确信息&#xff0c;配置主域服务器需要一整套配置文件&#xff1a; …

《QT实用小工具·六十八》基于QMenu开发的炫酷菜单栏

1、概述 源码放在文章末尾 该项目基于QMenu实现了炫酷的菜单栏效果&#xff0c;包含了如下功能&#xff1a; 1、实现了类似word菜单栏的效果&#xff0c;可以在菜单栏中横向添加不同的菜单 2、鼠标点击菜单可以展开菜单栏&#xff0c;再次点击菜单可以收起菜单栏 3、鼠标点击笑…

Chromium源码学习(1)—— 拉取源码,编译

阅读建议&#xff1a;先简单过一下整个文章目录结构&#xff0c;大致了解一下各个步骤在干什么&#xff0c;然后在上手操作可能会事半功倍。也许你遇到的有些问题文章中已经提及到了&#xff0c;但是由于你没有往下看导致卡进度。 Chromium简介 Chromium项目于2008年发布&…

postman教程-7-文件上传接口

领取资料&#xff0c;咨询答疑&#xff0c;请➕wei: June__Go 上一小节我们学习了postman发送get请求的方法&#xff0c;本小节我们讲解一下postman文件上传接口的请求方法。 postman文件上传的方式大概有两种&#xff0c;一种是form-data类型上传文件&#xff0c;一种是bin…

IEnumerable 、 IEnumerator,yield return

自定义迭代类 》》》using System.Collections; using System.Collections; using System.Runtime.CompilerServices;namespace ConsoleApp1 {// 可迭代对象 标记此类可迭代 继承IEnumerable 类是可以迭代public class SpecificEnumerable : IEnumerable{private readonly …

RAG 高级应用:基于 Nougat、HTML 转换与 GPT-4o 解析复杂 PDF 内嵌表格

一、前言 RAG&#xff08;检索增强生成&#xff09;应用最具挑战性的方面之一是如何处理复杂文档的内容&#xff0c;例如 PDF 文档中的图像和表格&#xff0c;因为这些内容不像传统文本那样容易解析和检索。前面我们有介绍过如何使用 LlamaIndex 提供的 LlamaParse 技术解析复…

GD32F103RCT6/GD32F303RCT6(10)独立看门狗/窗口看门狗实验

本文章基于兆易创新GD32 MCU所提供的2.2.4版本库函数开发 后续项目主要在下面该专栏中发布&#xff1a; 手把手教你嵌入式国产化_不及你的温柔的博客-CSDN博客 感兴趣的点个关注收藏一下吧! 电机驱动开发可以跳转&#xff1a; 手把手教你嵌入式国产化-实战项目-无刷电机驱动&am…

集合的综合练习

自动点名器1&#xff1a;班级里有N个学生&#xff0c;实现随机点名器 public class test {public static void main(String [] args) {ArrayList<String> listnew ArrayList<>();//创建一个集合//在集合中添加元素Collections.addAll(list, "李明",&quo…