康耐视智能相机(Insight)通过ModbusTCP发送字符串到倍福(BECKHOFF)PLC中
文章目录
- 1.背景
- 2.分析
- 3.实现
- 3.1.PLC的ModbusTCP_Server
- 3.1.1.安装TF6250-Modbus-TCP
- 3.1.2.PLC设置
- 3.2.智能相机的ModbusTCP_Client
- 3.2.1.了解ModbusTCP的协议
- 3.2.2.根据协议写代码
- 3.2.2.1.纯函数代码
- 3.2.2.2.脚本代码
- 3.2.3.非脚本处理时的代码逻辑图
- 3.2.4.关于代码的问题及解答
- 4.总结
1.背景
目前有个需求,要求康耐视智能相机(Insight)在每次触发完成作业后,将一串字符串通过ModbusTCP发送至倍福的PLC中。此时PLC作为Modbus的server,智能相机作为Modbus的client,智能相机主动发送数据给PLC(写PLC的Modbus的保持寄存器(Holding Registers))。
目前已经实现了,效果如下:
代码我已经上传到这里了【康耐视智能相机ModbusTCP发送字符串代码 】,不需要下载积分。造福大家。
2.分析
主要有两部分的功能要实现:PLC的ModbusTCP_Server的实现、智能相机的ModbusTCP_Client的实现。
有了server和client,通讯起来就没啥问题了。至于其中涉及的实时性的问题,暂时先不考虑。
3.实现
经过查阅资料,倍福PLC作为ModbusTCP Server的设置过程很简单,而智能相机的设置过程则非常麻烦。下面来详细介绍。
3.1.PLC的ModbusTCP_Server
3.1.1.安装TF6250-Modbus-TCP
根据从这里【TwinCAT 3 Modbus TCP使用方法】查阅得到的资料,只要下载并安装TF6250-Modbus-TCP这个软件包即可。


TF6250-Modbus-TCP的下载地址为【TF6250 | TwinCAT 3 Modbus TCP】,下载的时候需要先登录倍福的账号,没有的话注册一个即可,就是它要求的密码复杂度比较高,要字母+数字+特殊符号+大小写。
具体安装步骤,可以查看前面提到的那个网站的【TwinCAT 3 Modbus TCP使用方法.docx】

3.1.2.PLC设置
首先,先激活一下试用


然后写程序

PROGRAM MAIN
VARarr1 AT %MB0:ARRAY[1..2] OF WORD; (*保持寄存器 起始地址为12288 (0x3000)*)arr2 AT %MB10:ARRAY[1..2] OF BYTE; (**)str1 AT %MB0: STRING;byteArr1 AT %MB0:ARRAY[0..9] OF BYTE;byteArr2 AT %MB0:ARRAY[1..10] OF BYTE;
END_VAR
为了方便观察,这里准备了字符串以及byte数组,这样子的话,既可以看到字符串,又可以看到字符串对应的16进制值。

至此,ModbusTCPServer的设置已经完成了,内存的映射等其他操作,系统已经帮忙处理了。

这里需要注意的是,modbus寄存器与PLC的地址对应关系。这个关系我们可以通过查阅官方文档【TF6250 | TwinCAT 3 Modbus TCP Default Configuration】得知:
保持寄存器对应的是Output registers
所以我们读写保持寄存器时,地址偏移要设置成12288(0x3000)。前面的文档有说要+1,但是我测试不用加1也行,可能具体得看实际情况吧。

ok,倍福PLC这边已经设置完成了,这时其实就已经可以用你趁手的Modbus调试工具测试一下与PLC的通讯了。我这边测试的话,貌似slave ID设置成0、1或者其他任意值都可以。

3.2.智能相机的ModbusTCP_Client
智能相机这边就相当的麻烦了。因为我们用的相机的固件版本比较低,不直接支持Modbus(貌似新的固件版本也只是支持ModbusServer,而非ModbusClient),得靠我们自己通过TCPDevice根据Modbus的协议来手动编码。
行吧,那就自己手动搞呗。
3.2.1.了解ModbusTCP的协议
首先,了解一下ModbusTCP的协议,【Modbus TCP协议说明】、【ModbusTCP数据帧】、【如何看懂Modbus数据帧?】。了解其报文结构是非常重要的,因为只有了解其协议规定,我们才能确定为了达到我们的效果需要发送多长的数据,每个数据又应该是如何取值。
简单地说,我们可以总结modbusTCP的数据帧有以下几个特点
1.与通过串口发送的数据帧相比,不用携带校验码。这个应该是因为底层TCP/IP通讯本身就已经是可靠通讯(自带校验及重发)。挺好,我们不用再用CRC16算法来算效验码了。
2.与通过串口发送的数据帧相比,要在前面增加7个字节的MBAP报文头。
3.报文头中的事务处理标识需要累加。实际上,经过测试,不累加也行。后面的程序我都没有对其累加。
3.2.2.根据协议写代码
ok,我们动手。
经过测试,将字符串转化成需要发送出去的数据这部分,代码有两种实现方式:一种是纯利用智能相机提供的函数进行处理,另外一种是通过js脚本实现。
建议先用纯函数实现,然后再用脚本实现,这样理解起来会更加深刻。
3.2.2.1.纯函数代码
先直接放出程序,再依次解释每个函数的作用。

| 左侧文字 | 代码 | 说明 |
|---|---|---|
| '待发送的字符串 | EditString(255) | 字符串控件,可以在运行时输入字符串 |
| '字符串长度 | Len(J26) | 获取字符串的长度 |
| '长度是否为奇数 | Mod(J27,2) | 判断是否为奇数 |
| '减少最后一个字符的字符串 | Left(J26, J27-1) | 原来的字符串移除最后一个字符形成的字符串 |
| '最后一个字符 | Right(J26,1) | 原来字符串的最后的那个字符 |
| '偶数长度部分的字符串 | If(J28,J29,J26) | 原来字符串的的偶数长度部分的字符串,比如字符串长度为5,那么取前面4个;假如为6,那就全取。 |
| '编码1(偶数部分的字符串) | BStringf(0, “%~s”, J31) | 把字符串编码成Binary结构体 |
| '编码2(2byte的0) | BStringf(0, “%h”, 0) | 长度为2,且两个数据都是0的Binary结构体 |
| '编码3(0,字符) | BStringf(0, “%c%s”, 0, J30) | 长度为2,第1个字节为0,第2个字节为字符串最后一个字符,的Binary结构体 |
| '最终发送的编码(奇数长度时) | BStringf(0,“%b%b”,J32,J34) | 字符长度为奇数时,最终需要发送的Binary数据结构体;偶数部分+0+最后一个字符 |
| '最终发送的编码(偶数长度时) | BStringf(0,“%b%b”,J32,J33) | 字符长度为偶数时,最终需要发送的Binary数据结构体;偶数部分+ 0 + 0 |
| '实际需要发送的编码 | $If(J28,J35,J36) | 根据原始字符串的长度是奇数还是偶数,确定发送的Binary数据结构体; |
| '需要发送的编码长度 | BLen(J37) | 实际需要发送的编码的长度 |
| '计算Modbus的数据长度 | 1 + 1 + 2 + 2 + 1 + J38 | 1设备ID+1字节功能码+2字节寄存器地址+2字节寄存器数量+1字节寄存器数据长度+寄存器数据本身的长度 |
| '手动触发发送 | Button(“触发”,-1) | 按钮,触发发送 |
| '设备 | TCPDevice(“127.0.0.1”,502,0,4,1000,255) | 设备,IP:127.0.0.1,端口:502,modbus(二进制)发送 |
| '读写 | QueryDevice(M29,M30,0,0,0,0,0,M26,0,0x10,0x30,0,0,J38/2,J38,J37) | 发送数据,且读取接收到的数据。M26是【'计算Modbus的数据长度】;0是slaveID;0x10表示写保持寄存器;0x30 0x00表示PLC的寄存器地址; J38/2表示寄存器的数量,因为我们的数据长度是以字节为单位的,而寄存器是16bit大小的,可以存储2个字节,因此要除以2; J38要写到n个寄存器的数据的大小;,J37,要写到n个寄存器的数据。 |
3.2.2.2.脚本代码

部分代码解释和上一小节一样,主要解释一下脚本部分
| 左侧文字 | 代码 | 说明 |
|---|---|---|
| ''将字符串通过脚本处理得到编码 | Script($J$47) | 通过脚本处理字符串。脚本的输入参数为字符串,返回处理完成的Binary结构体 |
脚本截图

脚本:
function Script() {
}
module.exports = Script;function stringToUint8ArrayWithSwap(str) {var dataArrLen = str.length;if (str.length % 2 !== 0) {dataArrLen += 1; // 奇数就补一个0}else{dataArrLen += 2; // 偶数就补两个0}var dataArray = new Uint8Array(dataArrLen); // 数据会默认初始化为0for (var i = 0; i < str.length; i++) {dataArray[i] = str.charCodeAt(i); // 将字符拷贝到数组中} // 每两个数据一组,交换组中的元素for (var i = 0; i < dataArrLen; i += 2) {// 交换位置 i 和 i+1 的数字[dataArray[i], dataArray[i + 1]] = [dataArray[i + 1], dataArray[i]];}return dataArray;
}Script.prototype.run = function (arg0) {return stringToUint8ArrayWithSwap(arg0)
}
3.2.3.非脚本处理时的代码逻辑图

3.2.4.关于代码的问题及解答
1.为啥要根据字符串长度的奇偶来执行不同的处理办法?
因为我们最终执行的modbus功能是写保持寄存器,而保持寄存器的大小为2字节,每次写的话都只能写n个寄存器(n为整数),也就是2n个字节的数据。所以,每次写的数据必须是偶数个数据才能填充完n个寄存器。不允许写半个寄存器。
2.为啥要 BStringf(0, “%~s”, J31) 中要选 %~s这种格式?
因为假如选了%s这种格式的话,数据是按小端发送过去的,但是,PLC那边接收到数据是按大端处理的,这就导致你发了字符串"123456"过去,PLC那边存储且显示出来的是"214365",每两个字节内部互相交换。
3.既然是大小端的问题,为啥不通过设置 BStringf中的第一个参数来处理?
因为它只对数值型数据,比如double、int等类型起作用,对字符串这种连续、独立、可变长度的类型无效。只能选%~s这种格式来处理。
3.字符长度为奇数时,为啥要将最后一个字符单独拿出来,然后插入一个0,然后再把这个字符补回去。
首先,补0是必须的,因为任何字符串都需要用0来作为结束符,不然无法确定一段字符在何处结束。至于为啥要把这个0补在最后一个字符的前面(插队),还是前面说的大小端的问题。我们发送过去的数据,PLC那边会每两个字节交替存储,比如说问哦我们发了 0x00 0x01两个数据过去,存在PLC寄存器的顺序会自动调整为0x01 0x00,而我们的0x00需要存放在物理地址的高位(也就是后面),因此需要做一个插入操作。
4.总结
通过底层的方式实现数据传输,学是能够学到好多东西,但是就是贼麻烦,且鲁棒性差。不知道为啥Modbus这么通用的功能,康耐视智能相机为啥就是不支持。
参考:
【TwinCAT 3 Modbus TCP使用方法】
【Modbus TCP协议说明】
【ModbusTCP数据帧】
【如何看懂Modbus数据帧?】
相关文章:
康耐视智能相机(Insight)通过ModbusTCP发送字符串到倍福(BECKHOFF)PLC中
文章目录 1.背景2.分析3.实现3.1.PLC的ModbusTCP_Server3.1.1.安装TF6250-Modbus-TCP3.1.2.PLC设置 3.2.智能相机的ModbusTCP_Client3.2.1.了解ModbusTCP的协议3.2.2.根据协议写代码3.2.2.1.纯函数代码3.2.2.2.脚本代码 3.2.3.非脚本处理时的代码逻辑图3.2.4.关于代码的问题及解…...
TIFS投稿记录(IEEE Transactions on Information Forensics Security)
毕竟是CCF A类期刊,TIFS审稿有点慢,记录最近一篇论文的投稿时间线。 2024年10月27日:提交。 2024年11月12日:分配DE。 2024年12月3日:AE与SAE还未分配。发邮件催了催。 2024年12月5日:SAE已分配。AE: Not A…...
极越汽车,加速跌落
文丨梅元知 9月,极越销量2605辆;10月进一步攀升到3107辆,尽管11月略有回落,销量跌至2485辆,但对于一个品牌影响力尚未完全建立、销售渠道有限的新品牌而言,这样的表现已实属不易。然而,就在看似…...
深入解析MySQL事务隔离级别与锁机制在银行账户业务中的应用
一、引言 在金融行业,尤其是银行账户业务中,数据的一致性和安全性至关重要。MySQL作为一种广泛使用的数据库,其事务隔离级别和锁机制在保证数据一致性方面发挥着重要作用。本文将针对银行账户查询与转账业务,探讨如何运用事务锁来…...
postman可以通的请求,前端通不了(前端添加Content-type,后端收不到请求)
接口完成之后,自己使用postman测试了一下,没有问题; 可是在和小组前端调试接口的时候,他却说访问不了; 信息如下:(我自己写的一个打印请求信息的拦截器) 发现报错信息是: Content type ‘application/x-www-form-urlencoded;charset=UTF-8’ not supported 也就是说…...
【Java计算机毕业设计】基于SSM+VUE宠物领养管理系统【源代码+数据库+LW文档+开题报告+答辩稿+部署教程+代码讲解】
源代码数据库LW文档(1万字以上)开题报告答辩稿 部署教程代码讲解代码时间修改教程 一、开发工具、运行环境、开发技术 开发工具 1、操作系统:Window操作系统 2、开发工具:IntelliJ IDEA或者Eclipse 3、数据库存储:…...
排队论、负载均衡和任务调度关系
目录 排队论、负载均衡和任务调度关系 一、排队论 二、负载均衡 三、任务调度 四、总结 排队论、负载均衡和任务调度关系 排队论为负载均衡和任务调度提供了数学理论和方法支持 排队论、负载均衡和任务调度是三个相关但不同的概念。以下是对这三个概念的详细解释和它们之…...
智能客户服务:科技赋能下的新体验
在当今这个数字化时代,客户服务已经不仅仅是简单的售后服务,它已竞争的关键要素之一。随着人工智能、大数据、云计算等技术的飞速发展,智能客户服务正逐步改变着传统的服务模式,为企业和消费者带来了前所未有的新体验。 一、智能客…...
代码随想录第45天
115.不同的子序列 class Solution:def numDistinct(self, s: str, t: str) -> int:n1 len(s)n2 len(t)dp [[0] * (n1 1) for _ in range(n2 1)]for j in range(n1 1):dp[0][j] 1for i in range(1, n2 1):for j in range(1, n1 1):if t[i - 1] s[j - 1]:dp[i][j]…...
前端项目初始化搭建(二)
一、使用 Vite 创建 Vue 3 TypeScript 项目 PS E:\web\cursor-project\web> npm create vitelatest yf-blog -- --template vue-ts> npx > create-vite yf-blog --template vue-tsScaffolding project in E:\web\cursor-project\web\yf-blog...Done. Now run:cd yf-…...
3D 目标检测:从萌芽到前沿的技术演进之路
亲爱的小伙伴们😘,在求知的漫漫旅途中,若你对深度学习的奥秘、JAVA 、PYTHON与SAP 的奇妙世界,亦或是读研论文的撰写攻略有所探寻🧐,那不妨给我一个小小的关注吧🥰。我会精心筹备,在…...
Apifox 产品更新|支持发布多个文档站、文档站支持 Algolia 搜索配置、从返回响应直接设置断言
看看本次 这次版本更新主要涵盖的重点内容,有没有你所关注的功能特性: 「发布文档」升级为「发布文档站」 支持发布多个文档站 文档站支持 Algolia 搜索配置 支持从返回响应直接设置断言 用户反馈优化 解决恢复退出 App 时未关闭的标签页可能导致内存…...
Linux内核结构及源码概述
参考:深入分析LINUX内核源码 深入分析Linux内核源码 (kerneltravel.net) Linux 是一个庞大、高效而复杂的操作系统,虽然它的开发起始于 Linus Torvalds 一个人,但随着时间的推移,越来越多的人加入了 Linux 的开发和对它的不断完善…...
《探索C++在3D重建中的算法与技术要点》
3D重建作为计算机视觉领域的重要技术,在诸多行业有着广泛应用,而C以其高效性和对底层硬件的良好控制,成为实现3D重建算法的常用语言。以下是利用C进行3D重建的一些常见算法和技术要点。 多视图立体视觉算法 多视图立体视觉是3D重建的基础算…...
【老白学 Java】数字格式化
数字格式化 文章来源:《Head First Java》修炼感悟。 很多时候需要对数字或日期进行格式化操作,来达到某些输出效果。Java 的 Formatter 类提供了很多扩展性功能用于字符串的格式化,只要调用 String 静态方法 format() ,传入参数…...
useCallback和forwardRef的联合使用
文章目录 一、useCallback二、forwardRef 总结了useCallback、forwardRef中的deps,以及操作子组建时会遇到数据流不同步的问题 一、useCallback useCallback可以缓存函数,这样避免组建更新导致的函数重建;useCallback在函数更新以后会在deps中…...
C# .NET CORE 开发问题汇总
1. error MSB4803: .NET Core 版本的 MSBuild 不支持“ResolveComReference”。请使用 .NET Framework 版本的 MSBuild。 引用了一个COM组件, 使用donet 命令时,提示不支持, 可以先将项目设置为x86以构建, 将COM引用添加到核心项目中,构建它,在obj\x86\…...
【C语言】拆数字组成最大数
相信你是最棒哒!!! 文章目录 题目描述 正确代码 法一注释版 简洁版 法二注释版 简洁版 题目描述 任意输入一个自然数,输出该自然数的各位数字组成的最大数。例如,输入 1593 ,则输出为 9531 。 输入描述 …...
【Git系列】根据提交打印邮箱
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...
Nginx在处理客户端请求的并发性发面是否依赖Linux的多线程原理
Nginx在处理客户端请求的并发性发面是否依赖Linux的多线程原理 Nginx 在处理客户端请求的并发性方面,并不依赖于 Linux 的多线程原理。 Nginx 的并发处理主要基于 事件驱动模型 和 异步非阻塞 I/O,而不是传统的多线程或多进程模型。 Nginx 的并发处理模…...
OpenClaw+千问3.5-9B学习助手:自动整理笔记与生成习题
OpenClaw千问3.5-9B学习助手:自动整理笔记与生成习题 1. 为什么需要AI学习助手? 去年备考PMP证书时,我每天要处理上百页PDF讲义。最痛苦的莫过于手动整理重点和制作复习卡片——复制粘贴到半夜,第二天发现漏了关键图表ÿ…...
算法复杂度的视觉化表达与教学研究的技术
引言算法复杂度作为计算机科学核心概念,其抽象性常导致学习障碍。视觉化表达与教学研究旨在通过直观手段提升理解效率。本大纲从理论基础、视觉化工具、教学方法、案例分析和未来方向展开。理论基础算法复杂度定义与分类(时间/空间复杂度) 大…...
绕过Boss直聘反爬:用Selenium+本地Chrome Profile实现稳定数据采集(附防封号心得)
企业招聘数据采集实战:基于用户行为模拟的合规解决方案 在数字化招聘时代,市场情报分析已成为企业人力资源战略的重要组成部分。许多技术团队希望通过自动化手段获取公开的招聘平台数据,用于行业人才分布分析、薪资水平调研和技能需求趋势预测…...
Limine混合ISO制作教程:BIOS/UEFI双启动镜像的完整流程
Limine混合ISO制作教程:BIOS/UEFI双启动镜像的完整流程 【免费下载链接】limine Modern, advanced, portable, multiprotocol bootloader and boot manager. 项目地址: https://gitcode.com/gh_mirrors/li/limine Limine是一款现代化、先进的可移植多协议引导…...
XXL-JOB调度中心集群部署实战:从单机到高可用的完整配置指南
XXL-JOB调度中心集群部署实战:从单机到高可用的完整配置指南 在当今企业级应用架构中,任务调度系统扮演着至关重要的角色。无论是日常的报表生成、数据同步,还是复杂的业务流水线处理,都需要一个可靠、高效的调度引擎来支撑。XXL-…...
Debian根文件系统定制:从零构建到实战优化
1. Debian根文件系统入门指南 第一次听说"根文件系统"这个概念时,我也是一头雾水。简单来说,它就像是你电脑的操作系统"骨架"——包含了启动、运行和管理系统所需的所有核心文件和目录。想象一下盖房子,根文件系统就是地…...
AI FUTURE北京亦庄AI未来大会在京启幕
4月8日,AI FUTURE北京亦庄AI未来大会在北京经济技术开发区(简称“北京经开区”,又称“北京亦庄”)北京智慧电竞赛事中心正式启幕。这场为期两天的人工智能盛会,以“让每个人看AI的另一面”为主题,集高端论坛…...
00 | 从零打造Claude Code:AI编程Agent完整解析(一)——引言篇
从零打造Claude Code:AI编程Agent完整解析(一)——引言篇 声明: 📝 作者:甜城瑞庄的核桃(ZMJ) 原创学习笔记,欢迎分享,但请保留作者信息及原文链接哦~ 本系列文章深度解析如何从零开始构建一个类似Claude Code的AI编程助手,涵盖Agent循环、工具系统、提示词工程、权限…...
企业官网源码_公司网站模板_自适应手机端
一、源码下载平台:企业建站的“数字工具箱” 1. 开源生态驱动创新 GitHub、Gitee等全球开源代码托管平台,汇聚了数百万企业级项目。以GitHub为例,其企业官网源码库涵盖电商、教育、金融等20余个行业,包含完整的前端框架…...
AI Agent在物流与运输中的应用:路径优化与调度自动化
AI Agent在物流与运输中的应用:路径优化与调度自动化 引言 在当今快速发展的商业环境中,物流与运输行业正面临着前所未有的挑战。随着电子商务的爆发式增长,消费者对配送速度、成本和可靠性的要求越来越高。同时,全球化供应链的复杂性、燃油价格的波动以及环保法规的日益…...
