模块----modbus_tk中TCP协议简单应用

2024-05-08 19:18

本文主要是介绍模块----modbus_tk中TCP协议简单应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


被告知说会考有关modbus_tk这个模块的东东,我和小伙伴们都惊呆了,很是捉鸡啊~~~

关于这个模块,基本上就是零基础,一步步来。以下都是我的个人见解,若有错误请大胆地指出来吧~~



=====================================================================================================================

第一步  做好基本准备工作

=====================================================================================================================


1.查阅modbus本身相关的资料,弄清楚它大概是什么东东

2.安装modbus_tool,里面有个POLL和SLAVE,尽情去调试吧,有利于弄清楚modbus的结构,还可以测试程序哦~~

楼主当时是没有弄modbus_tool,直接进程序瞎倒腾,后来被高手指点才跑去用modbus_tool,果然豁然开朗




=====================================================================================================================

第二步  安装modbus_tk

=====================================================================================================================


在linux和windows上安装会有点不一样,其实就是windows在前面多了一个步骤。

----------------------------------------------------------------------------------------------------------------

linux:

在官网下载完modbus_tk,解压进去就会发现有个setup.py的文件,直接在此目录下打开终端,运行python setup.py  

就会有个提示help告诉你怎么安装,然后用install的那个参数就行了。


windows:

先下载安装一个Modbus Tools.7z的东东,之后就和在linux的操作一样一样的了。

这个东西是安装python其它模块的一个辅助软件,就是说有了这个东西,它才能去装其它的一些模块

linux 本来就有带Modbus Tools,所以可以直接安装

注:这个Modbus Tools.7z是python里面的一个模块,不是前面说的modbus_tool哦~




=====================================================================================================================

第三步  看示例吧·~

=====================================================================================================================


示例一   前辈的示例 (被我改了一些,原始版的木有了,懒得去再打一次,将就用修改版的吧)

------------------------------------------------------------------------------------------------------------

得把slave先运行,才能用master读取或者修改。

#!/usr/bin/python
# -*- coding: utf-8 -*-
# filename: slave.py# ------------------------------------------------------------------------------
# 导入外部模块
# ------------------------------------------------------------------------------import struct
import modbus_tk.defines
import modbus_tk.modbus
import modbus_tk.modbus_tcp
import time
import random# ------------------------------------------------------------------------------
# 主程序
# ------------------------------------------------------------------------------try:server = modbus_tk.modbus_tcp.TcpServer()#注意,若是在linux里面运行,端口就不能不写了,否则就要用root才能跑,其它用户只能用1024以上的端口#这里的端口和地址都是默认的,地址是本地:#原来的程序:server = modbus_tk.modbus_tcp.TcpServer(port=502, address='0.0.0.0', timeout_in_sec=3)server.start()slave_1 = server.add_slave(1)slave_1.add_block('block1', modbus_tk.defines.ANALOG_INPUTS, 0, 2)#给slave_1添加一个模块(模块名,只读,地址,长度)while 1:bb = random.random()aa = struct.unpack('>HH', struct.pack('>f', bb))print 'bb:', bbprint "aa:", aaslave_1.set_values('block1', 0, aa)print '========='time.sleep(3)
except:print '============error==========='
finally:print '=========stop========'server.stop()

#!/usr/bin/python
# -*- coding: utf-8 -*-
# filename: master.py# ------------------------------------------------------------------------------
# 导入外部模块
# ------------------------------------------------------------------------------import modbus_tk
import modbus_tk.defines
import modbus_tk.modbus
import modbus_tk.modbus_tcp
import struct
import time# ------------------------------------------------------------------------------
# 主程序
# ------------------------------------------------------------------------------try:while 1:master = modbus_tk.modbus_tcp.TcpMaster()#('0.0.0.0', 502)   (master ip 地址,端口)#注意用linux的童鞋们,端口小于1024得用root才能跑起来哦~#为这个问题楼主曾经也折腾了一阵,后来才发现居然是这个原因·····#这里有个问题就是原来我是用'0.0.0.0'这个地址的。#但是它应该不能算一个ip地址了,这个是代表现有的ip地址均可的意思,有种通配符的感觉#在slave中可以用它,但是在master中不行,一定要给它指明一个具体的地址,看来它也有选择困难症master.set_timeout(3)  #timeout表示若超过3秒没有连接上slave就会自动断开#set_timeout(3)不知是何意----重新把timeout时间设置成3秒,在生成master那里就可以进行初始化定义的,若没有自定义就会用默认值aa = master.execute(1, modbus_tk.defines.ANALOG_INPUTS, 0, 2)#(slave id,只读,block地址,长度:即字节乘个数)print 'aa:', aa, 'size', len(aa)bb = struct.unpack('>f', struct.pack('>HH', aa[0], aa[1]))#在slave中,是先打包成'>f'在以'>HH'解包的,在master中刚好相反#struct.unpack里解出来的是一个元组,可以用bb, = 或者输出bb[0]print 'bb:', bb[0]time.sleep(3)
except Exception as e:print '=========error========='print e
finally:print '=======stop======='

查看端口是否已经开放               (用root权限)
/etc/init.d/iptables status
/sbin/iptables -I INPUT -p tcp --dport 5023 -j ACCEPT   # 开启5023端口
/etc/rc.d/init.d/iptables save #保存配置
/etc/rc.d/init.d/iptables restart #重启服务 



示例二   官网的示例       很坑爹····好多错误····不知道怎么会放在官网上的·····果断也被我改了
---------------------------------------------------------------------------------------------------------------------

#!/usr/bin/python
# -*- coding: utf-8 -*-
# filename: server_example.py# ------------------------------------------------------------------------------
# 导入外部模块
# ------------------------------------------------------------------------------import modbus_tk
import modbus_tk.modbus_tcp as modbus_tcp
import modbus_tk.defines as mdef
import threading
import sys# ------------------------------------------------------------------------------
# 主程序
# ------------------------------------------------------------------------------logger = modbus_tk.utils.create_logger(name='console', record_format='%(message)s')if __name__ == '__main__':try:logger.info("running...")logger.info("enter 'quit' for closing the server")server = modbus_tcp.TcpServer(port=1152)server.start()#sever.start()可以在加了slave和模块以后,也可以不加直接裸奔~#creates a slave with id 0slave1 = server.add_slave(1)#add 2 blocks of holding registersslave1.add_block('a', mdef.HOLDING_REGISTERS, 0, 100)  #address 0, length 100slave1.add_block('b', mdef.HOLDING_REGISTERS, 200, 20) #address 200, length 20#create another slave with id 5slave5 = server.add_slave(5)slave5.add_block('c', mdef.COILS, 0, 100)slave5.add_block('d', mdef.HOLDING_REGISTERS, 0, 100)#set the values of registers at address 0slave1.set_values('a', 0, range(100))while True:cmd = sys.stdin.readline()if cmd.find('quit')==0:#当输入的只有quit,程序就会退出sys.stdout.write('bye-bye\r\n')sys.exit(0)finally:server.stop()

#!/usr/bin/python
# -*- coding: utf-8 -*-
# filename: master_example.py# ------------------------------------------------------------------------------
# 导入外部模块
# ------------------------------------------------------------------------------
import modbus_tk
import modbus_tk.defines as cst
import modbus_tk.modbus_tcp as modbus_tcp# ------------------------------------------------------------------------------
# 主程序
# ------------------------------------------------------------------------------
if __name__ == '__main__':try:#Connect to the slavemaster = modbus_tcp.TcpMaster(port=1152)print master.execute(1, cst.READ_HOLDING_REGISTERS, 10, 3)master.execute(1, cst.WRITE_MULTIPLE_REGISTERS, 200, output_value=xrange(12))print master.execute(1, cst.READ_HOLDING_REGISTERS, 10, 3)except modbus_tk.modbus.ModbusError, e:print "Modbus error", e.get_exception_code()except Exception, e2:print "Error", str(e2)


示例三   modbus_tk文件中本身自带的example
---------------------------------------------------------------------------------------------------------------------

slave

#!/usr/bin/env python
# -*- coding: utf_8 -*-
"""Modbus TestKit: Implementation of Modbus protocol in python(C)2009 - Luc Jean - luc.jean@gmail.com(C)2009 - Apidev - http://www.apidev.frThis is distributed under GNU LGPL license, see license.txt
"""import sys#add logging capability
import logging
import threadingimport modbus_tk
import modbus_tk.defines as cst
import modbus_tk.modbus as modbus
import modbus_tk.modbus_tcp as modbus_tcplogger = modbus_tk.utils.create_logger(name="console", record_format="%(message)s")if __name__ == "__main__":try:#Create the serverserver = modbus_tcp.TcpServer(port=1500)logger.info("running...")logger.info("enter 'quit' for closing the server")server.start()slave_1 = server.add_slave(1)slave_1.add_block('0', cst.COILS, 0, 100)slave_1.set_values('0', 0, range(100))#range(100)为0-99的列表while True:cmd = sys.stdin.readline()#这里是进入程序以后,读取每次输入的命令:以回车来判断命令是否输入完毕args = cmd.split(' ')if cmd.find('quit')==0:#当输入的只有quit,程序就会退出sys.stdout.write('bye-bye\r\n')breakelif args[0]=='add_slave':slave_id = int(args[1])#slave id只能为int?是的server.add_slave(slave_id)sys.stdout.write('done: slave %d added\r\n' % (slave_id))#write到哪里去了?屏幕输出去了~~elif args[0]=='add_block':slave_id = int(args[1])name = args[2]block_type = int(args[3])starting_address = int(args[4])length = int(args[5])slave = server.get_slave(slave_id)slave.add_block(name, block_type, starting_address, length)sys.stdout.write('done: block %s added\r\n' % (name))elif args[0]=='set_values':slave_id = int(args[1])name = args[2]#这个name是slave_id对应的block的名字address = int(args[3])values = []#这里的values只可少设不可多设,否则就报错退出程序for v in args[4:]:#value只能是int,就算是其他数字类型也报错#囧 好奇怪,明明平时int(0.1)也是可以的values.append(int(v))slave = server.get_slave(slave_id)slave.set_values(name, address, values)values = slave.get_values(name, address, len(values))sys.stdout.write('done: values written: %s\r\n' % (str(values)))elif args[0]=='get_values':slave_id = int(args[1])name = args[2]address = int(args[3])length = int(args[4])slave = server.get_slave(slave_id)values = slave.get_values(name, address, length)sys.stdout.write('done: values read: %s\r\n' % (str(values)))else:sys.stdout.write("unknown command %s\r\n" % (args[0]))finally:server.stop()

master

#!/usr/bin/env python
# -*- coding: utf_8 -*-
"""Modbus TestKit: Implementation of Modbus protocol in python(C)2009 - Luc Jean - luc.jean@gmail.com(C)2009 - Apidev - http://www.apidev.frThis is distributed under GNU LGPL license, see license.txt
"""import sys#add logging capability
import loggingimport modbus_tk
import modbus_tk.defines as cst
import modbus_tk.modbus_tcp as modbus_tcplogger = modbus_tk.utils.create_logger("console")if __name__ == "__main__":try:#Connect to the slavemaster = modbus_tcp.TcpMaster(port=1500)master.set_timeout(5.0)logger.info("connected")#logger.info(master.execute(1, cst.READ_HOLDING_REGISTERS, 100, 100))#(slave id,Modbus数据模型,block地址,长度:即字节乘个数)#同一个slave下,不同block的Modbus数据模型可以是不一样的##################################################################################################
#下面的例子都需要自己在slave里面添加slave(需要改下面的id)或block才能实现#-------------------------------------------------------------------------------------------------print cst.COILS                              #1-----1位读写print cst.READ_COILS                         #1-----1位只读print cst.WRITE_SINGLE_COIL                  #5-----1位只写print cst.WRITE_MULTIPLE_COILS               #15----1位写logger.info(master.execute(1, cst.READ_COILS, 0, 10))#一位:所以个数是10,字节是1     若给非0值,均会以1存入   默认值是(0,0,0,0,0,0,0,0,0,0)#只写的,这里会输出modbus的可识别的格式'>HH',即有两个元素的元组:a = logger.info(master.execute(1, cst.WRITE_SINGLE_COIL, 7, output_value=0))#将slave的地址7的value改成1。logger.info(master.execute(1, cst.READ_COILS, 0, 10))#logger.info(master.execute(1, cst.WRITE_MULTIPLE_COILS, 5, output_value=xrange(12)))print 'a =', a # a=None#-------------------------------------------------------------------------------------------------print cst.DISCRETE_INPUTS                    #2-----1位只读print cst.READ_DISCRETE_INPUTS               #2-----1位只读#logger.info(master.execute(1, cst.READ_DISCRETE_INPUTS, 0, 8))#一位:所以个数是8,字节是1     若给非0值,均会以1存入    默认值是(0,0,0,0,0,0,0,0)#-------------------------------------------------------------------------------------------------print cst.HOLDING_REGISTERS           #3-----16位读写print cst.READ_HOLDING_REGISTERS      #3-----16位只读print cst.WRITE_SINGLE_REGISTER       #6-----16位写print cst.WRITE_MULTIPLE_REGISTERS    #16----16位写#这里读取的是之前的信息#logger.info(master.execute(1, cst.READ_HOLDING_REGISTERS, 0, 3))#16位:所以个数是3,字节是16     若给负值,会出错         默认值是(0,0,0)#logger.info(master.execute(1, cst.READ_HOLDING_REGISTERS, 0, 100))#logger.info(master.execute(1, cst.WRITE_MULTIPLE_REGISTERS, 0, output_value=[50, 1, 1, 1, 1, 1, 1, 1]))#从地址0开始,后面8为全是1#logger.info(master.execute(1, cst.WRITE_MULTIPLE_REGISTERS, 5, output_value=xrange(12)))#从地址5开始,后面12位是xrange(12)#这里读取的是改变后的信息#logger.info(master.execute(1, cst.READ_HOLDING_REGISTERS, 0, 3))#logger.info(master.execute(1, cst.READ_HOLDING_REGISTERS, 0, 100))#logger.info(master.execute(1, cst.WRITE_SINGLE_REGISTER, 5, output_value=1))#-------------------------------------------------------------------------------------------------print cst.ANALOG_INPUTS               #4-----16位只读print cst.READ_INPUT_REGISTERS        #4-----16位只读#logger.info(master.execute(1, cst.READ_INPUT_REGISTERS, 10, 12))#16位:所以个数是3,字节是16     若给负值,会出错         默认值是(0,0,0,0,0,0,0,0,0,0,0,0)##################################################################################################except modbus_tk.modbus.ModbusError, e:logger.error("%s- Code=%d" % (e, e.get_exception_code()))


更多资料   后来才发现原来modbus_tk文件中自带的doc文件夹下面有源代码的   囧   看那个应该会很不错



=====================================================================================================================
第四步  小总结
=====================================================================================================================
1.在linux上运行除了要注意端口,还要注意的就是防火墙的问题,我是直接把防火墙给关了的。


2.modbus本身可以传输16位无符号整数,如果要传输浮点数或者是32位无符号整数就要进行打包,具体看struct模块。


3.server和master的功能码对应表
===================================================================
对于server(slave):


  modbus_tk.defines.COILS:               1位读写                  1
  modbus_tk.defines.DISCRETE_INPUTS:     1位只读                  2
  modbus_tk.defines.HOLDING_REGISTERS:   16位读写                 3
  modbus_tk.defines.ANALOG_INPUTS:       16位只读                 4
===================================================================
对于master(要于server对应起来):


  modbus_tk.defines.COILS:                       1-----1位读写
  modbus_tk.defines.READ_COILS                   1-----1位只读
  modbus_tk.defines.WRITE_SINGLE_COIL            5-----1位只写,一个个写的,更稳定,但效率低
  modbus_tk.defines.WRITE_MULTIPLE_COILS         15----1位只写,多个一起写,稳定性一般,效率高
-------------------------------------------------------------------
  modbus_tk.defines.DISCRETE_INPUTS:             2-----1位只读
  modbus_tk.defines.READ_DISCRETE_INPUTS         2-----1位只读
 
-------------------------------------------------------------------
  modbus_tk.defines.HOLDING_REGISTERS:           3-----16位读写
  modbus_tk.defines.READ_HOLDING_REGISTERS       3-----16位只读
  modbus_tk.defines.WRITE_SINGLE_REGISTER        6-----16位写,一个个写的,更稳定,但效率低
  modbus_tk.defines.WRITE_MULTIPLE_REGISTERS     16----16位写,多个一起写,稳定性一般,效率高
-------------------------------------------------------------------
  modbus_tk.defines.ANALOG_INPUTS:               4-----16位只读
  modbus_tk.defines.READ_INPUT_REGISTERS         4-----16位只读


===================================================================


这篇关于模块----modbus_tk中TCP协议简单应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/971204

相关文章

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

redis群集简单部署过程

《redis群集简单部署过程》文章介绍了Redis,一个高性能的键值存储系统,其支持多种数据结构和命令,它还讨论了Redis的服务器端架构、数据存储和获取、协议和命令、高可用性方案、缓存机制以及监控和... 目录Redis介绍1. 基本概念2. 服务器端3. 存储和获取数据4. 协议和命令5. 高可用性6.

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav

Python利用自带模块实现屏幕像素高效操作

《Python利用自带模块实现屏幕像素高效操作》这篇文章主要为大家详细介绍了Python如何利用自带模块实现屏幕像素高效操作,文中的示例代码讲解详,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1、获取屏幕放缩比例2、获取屏幕指定坐标处像素颜色3、一个简单的使用案例4、总结1、获取屏幕放缩比例from

nginx-rtmp-module模块实现视频点播的示例代码

《nginx-rtmp-module模块实现视频点播的示例代码》本文主要介绍了nginx-rtmp-module模块实现视频点播,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录预置条件Nginx点播基本配置点播远程文件指定多个播放位置参考预置条件配置点播服务器 192.

5分钟获取deepseek api并搭建简易问答应用

《5分钟获取deepseekapi并搭建简易问答应用》本文主要介绍了5分钟获取deepseekapi并搭建简易问答应用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需... 目录1、获取api2、获取base_url和chat_model3、配置模型参数方法一:终端中临时将加

JavaScript中的isTrusted属性及其应用场景详解

《JavaScript中的isTrusted属性及其应用场景详解》在现代Web开发中,JavaScript是构建交互式应用的核心语言,随着前端技术的不断发展,开发者需要处理越来越多的复杂场景,例如事件... 目录引言一、问题背景二、isTrusted 属性的来源与作用1. isTrusted 的定义2. 为

Python调用另一个py文件并传递参数常见的方法及其应用场景

《Python调用另一个py文件并传递参数常见的方法及其应用场景》:本文主要介绍在Python中调用另一个py文件并传递参数的几种常见方法,包括使用import语句、exec函数、subproce... 目录前言1. 使用import语句1.1 基本用法1.2 导入特定函数1.3 处理文件路径2. 使用ex

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像