总结自己在opc与自控开发的经验。首先介绍OPC DA模式下的OPC各种操作。

在使用opc时需要引用到 OPCDAAuto.dll 这个类库。

在项目引用后需要注册这个类库,否则程序跑起来会报错,“未找到工厂类 。。。”

将该dll文件放在任意目录下,建议在引用程序的的同级目录下。

在 cmd 控制台  输入regsvr32  Q:\PLCDataIntegration\packages\01OPCDaAuto\OPCDAAuto.dll

注册完成后电脑会提示注册成功,这时,就可以使用工具类中的方法啦。

1.定义相关变量

        private OPCServer opcServer;
private OPCGroups opcGroups;
private OPCGroup opcGroup;
private List<int> itemHandleClient = new List<int>();
private List<int> itemHandleServer = new List<int>();
private List<string> itemNames = new List<string>();
private List<model> modelValues = new List<model>();
private OPCItems opcItems;
private OPCItem opcItem;
private Dictionary<string, string> itemValues = new Dictionary<string, string>();

  

2.使用opc从plc中读取数据。这个是使用OPC DAAuto中的Connect方法。Connect之前要先

创建OPCServer 对象

opcServer = new OPCServer(); 

OPCServer.StartTime:服务器的启动时间

OPCServer.CurrentTime:服务器的当前时间,各个客户端可以通过这个属性值完成一些同步的操作

//strHostIP 主机IP,DA模式下通常为127.0.0.1;
//strHostName opc服务名,通常为字符串,例如kepsserver 的opc名称为 Kepware.KepServerEX.V6
 private bool ConnectServer(string strHostIP, string strHostName)
{
try
{
opcServer = new OPCServer();
opcServer.Connect(strHostName, strHostIP);
}
catch (Exception ex)
{
SaveCommand("连接到OPC服务器失败!" + ex.Message);
return false;
}
txtLog.Text += "连接到OPC服务器成功!" + "\r\n";
return true;
}

3.连接成功后就可以用opcServer这个对象了。

根据自己开发OPCServer 和OPCClient的经验,这里的OPCGroups与OPCGroup可以用树来理解,OPCGroups下可以有很多OPCGroup,OPCGroup下添加很多测点item。

OPC API 里AddItem 就是往OPCGroup下添加测点

4.OPCGroups与OPCGroup

  opcGroups = opcServer.OPCGroups;
 opcGroup = opcGroups.Add("YoursGroupName");

这里从OPCServer中获取OPCGroups信息,保持一致。

而OPCGroups下的OPCGroup可自行添加,然后自己注册的测点属性都在这个组下面

可以给OPCGroup设置如下属性,这将影响到这个组下面的测点数据采集频率等。

     private void SetGroupProperty(OPCGroup opcGroup, int updateRate)
{
opcGroup.IsActive = true;
opcGroup.DeadBand = 0f;
opcGroup.UpdateRate = updateRate;
opcGroup.IsSubscribed = true;
}
UpdateRate是一个整型值,影响组中测点数据更新频率。相关属性如下
OPCGroups.Count:下属组(Group)的数量
DefaultGroupIsActive(新添加的OPC组的活动状态的默认值。默认初始值是活动状态)
DefaultGroupUpdateRate(新添加的OPC组的默认数据更新周期,默认初始值是1000亳秒)、
DefaultGrouPDeadband( 新添加的OPC组的默认不敏感区的默认值,即能引起数据变化的最小数值百分比,默认值是0%)
DefaultGroupLocaleID(新添加的OPC组区域标识符的默认值)
DefaultGroupTimeBias(新添加的OPC组的时间偏差的默认值)等。

5.设置OPCGroup属性

  opcGroups = opcServer.OPCGroups;
opcGroup = opcGroups.Add("MYOPCGROUP");
SetGroupProperty(opcGroup, UpdateRateOPC);
opcItems = opcGroup.OPCItems;

6.注册opc监控测点。根据项目实际情况将监测点取出,以字符串数组传递。

        public void AddItems(string[] itemNamesAdded)
{
itemHandleServer.Clear(); for (int i = 0; i < itemNamesAdded.Length; i++)
{
itemNames.Add(itemNamesAdded[i]);
itemValues.Add(itemNamesAdded[i], "");
}
for (int j = 0; j < itemNamesAdded.Length; j++)
{
try
{
itemHandleClient.Add((itemHandleClient.Count == 0) ? 1 : (itemHandleClient[itemHandleClient.Count - 1] + 1));
opcItem = opcItems.AddItem(itemNamesAdded[j], itemHandleClient[itemHandleClient.Count - 1]);
itemHandleServer.Add(opcItem.ServerHandle);
}
catch (Exception ex)
{
SaveCommand(itemNamesAdded[j] + ex.Message);
}
}
}

7.注册读写或数据变化事件,opc的事件很多,常用的有以下几个。

opcGroup.AsyncWriteComplete //测点写入完成后触发
opcGroup.AsyncReadComplete //读取指令完成后触发
opcGroup.DataChange  //opcc测点数值变化后触发,通常用于实时数据上报

 //注册事件,然后在自己的方法里实现相关操作
opcGroup.AsyncWriteComplete += new DIOPCGroupEvent_AsyncWriteCompleteEventHandler(opcGroup_AsyncWriteComplete);

opcGroup.DataChange += new DIOPCGroupEvent_DataChangeEventHandler(KepGroup_DataChange);

opcGroup.AsyncReadComplete += new DIOPCGroupEvent_AsyncReadCompleteEventHandler(GroupAsyncReadComplete);

//实现方法 KepGroup_DataChange

        /// <summary>
/// 数据变动转储
/// </summary>
/// <param name="TransactionID"></param>
/// <param name="NumItems"></param>
/// <param name="ClientHandles"></param>
/// <param name="ItemValues"></param>
/// <param name="Qualities"></param>
/// <param name="TimeStamps"></param>
private void KepGroup_DataChange(int TransactionID, int NumItems, ref Array ClientHandles, ref Array ItemValues, ref Array Qualities, ref Array TimeStamps)
{
string itemID = string.Empty; try
{
string text = "hello rabbit";
int i = 0;
for (i = 1; i <= NumItems; i++)
{
//SetLogInfo($"{ItemValues.GetValue(i)}", Color.Green); if (ItemValues.GetValue(i) == null) { continue; } text = ItemValues.GetValue(i).ToString(); //SetLogInfo($"本次发布消息{text}", Color.Green); itemID = opcItems.GetOPCItem(itemHandleServer[(int)ClientHandles.GetValue(i) - 1]).ItemID; string message = itemID + ":" + text; MessageBox.Show(mesage);
} }
catch (Exception ex)
{
SaveCommand(ex.Message + itemID);
}
}

实现方法 GroupAsyncReadComplete

        private void GroupAsyncReadComplete(int TransactionID, int NumItems, ref Array ClientHandles, ref Array ItemValues, ref Array Qualities, ref Array TimeStamps, ref Array Errors)
{ //SaveCommand("**********GroupAsyncReadComplete***********"); string itemID = string.Empty;
string empty = string.Empty; try
{ int i = 0; for (i = 1; i <= NumItems; i++)
{
//非空判断,防止程序异常中断
if (ItemValues.GetValue(i) != null)
{
empty = ItemValues.GetValue(i).ToString();
} //获取监测点id
itemID = opcItems.GetOPCItem(itemHandleServer[(int)ClientHandles.GetValue(i) - 1]).ItemID; string text = itemID + ":" + empty; MessageBox.Show("读取完成,结果为"+text);
} MessageBox.Show($"本次读取完成,共{i}个", Color.Blue, 2);
}
catch (Exception ex)
{
MessageBox.Show($"GroupAsyncReadComplete:{ex.Message},itemID:{ itemID }");
}
}

7.使用监测数据,在成功读取到数据后就可以按自己的需求使用数据了,可以存到数据库,放到Redis,或者推到Rabbit等消息队列都没问题。

8.通过opc写入测点值实现,远程控制plc

其实远程控制,在代码层面来看并没有那么复杂,因为前面的工作都做好后,opc通道已经打开的情况下,控制只是一个写入操作。把指定的值写到指定的测点中,就是这么简单。实现控制难点不在控制本身,重点难点在于保证通信网络的稳定,和监测状态的

实时返回。总而言之,控制本身不复杂,复杂在安全性和稳定型的保证上。使用opc多为IT与OT融合的情景之下,IT平台本身缺乏对底层设备的有效监管,需要借助于opc客户端充当代理中介角色。这时OPC客户端本身的稳定性,和时效型显得尤为重要。这点

需要研发人员多多考虑。这点大家有好的方法可以评论分享下。

         //调用示例

string[] array2 = {"LCU1.PLC_GATE1_OPEN"};//点号开启1号闸门
          string[] array3 = {"1"};

         AsyncWrite(array2, array3);      

/// <summary>
/// 异步写方法,支持批量操作也可以调用opc单点读写方法
/// </summary>
/// <param name="writeItemNames"></param>
/// <param name="writeItemValues"></param>
public void AsyncWrite(string[] writeItemNames, string[] writeItemValues)
{
try
{
OPCItem[] array = new OPCItem[writeItemNames.Length];
for (int i = 0; i < writeItemNames.Length; i++)
{
for (int j = 0; j < itemNames.Count; j++)
{
if (itemNames[j] == writeItemNames[i])
{
array[i] = opcItems.GetOPCItem(itemHandleServer[j]);
break;
}
}
}
int[] array2 = new int[writeItemNames.Length + 1];
array2[0] = 0;
for (int k = 1; k < writeItemNames.Length + 1; k++)
{
array2[k] = array[k - 1].ServerHandle;
}
Array ServerHandles = array2;
object[] array3 = new object[writeItemNames.Length + 1];
array3[0] = "";
for (int l = 1; l < writeItemNames.Length + 1; l++)
{
array3[l] = writeItemValues[l - 1];
}
Array Values = array3;
opcGroup.AsyncWrite(writeItemNames.Length, ref ServerHandles, ref Values, out Array _, 2009, out int _);
GC.Collect();
}
catch (Exception ex)
{ MessageBox.Show(ex.Message);
}
}

至此OPC基本操作都完成了,OPC提供的官方API很深奥,有很多可以学的东西。OPC现在已经是自动化领域,跨域提供数据的潮流了,尤其时OPC UA发布后,自动化(OT)与信息化(IT)日趋融合,opc在传统自动化企业转型中应该会担任比较重要的角色。

所以OPC还是很值得一学的。

分享一首好听的歌 错都错了 ~~~

使用OPC与PLC通讯 一的更多相关文章

  1. PLC 通讯

    几个之前整理的高级语言与PLC通讯的资源下载链接:三菱:http://blog.sina.com.cn/s/blog_16d7d3ecb0102x6wj.html倍福:http://bbs.elecf ...

  2. WeinView 与 MITSUBISHI FX 系列 PLC 通讯范例

    1. 范例操作概述 此范例将介绍如何快捷简易地建立WEINVIEW HMI与MITSUBISHI FX系列 PLC通讯. 注意事项:通讯参数设置,通讯线接法. 2. 规划说明 (1) 新建简单 PLC ...

  3. C#与西门子PLC通讯

    1.0  通讯组件概述 通讯组件用于PC与可编程控制器(PLC).智能仪表等进行数据通讯,适用于基于PC高级语言的工业自动化控制系统.组件采用动态链接库文件(*.DLL)的形式,在PC系统的项目工程里 ...

  4. 【转】常用PLC通讯协议

    三菱FX系列PLC通讯测试 发送帧(Hex): 起始(STX) 02 命令(CMD) 30 首地址(ADDRESS) 30 30 41 30 字节数(BYTES) 30 31 终止(ETX) 03 校 ...

  5. 对S7通信的连接的理解以及对比CAN通信协议来理解PLC通讯

    对S7通信的连接的理解以及对比CAN通信协议来理解PLC通讯. 对功能块 SFB12 和 SFB13 的R_ID参数的理解 ? 对于同一个数据包.发送方与接收方的R_ID应该相同. 用下图解释 双向连 ...

  6. 威纶通 与 信捷XC\XD系列PLC 通讯

    第一次使用信捷XD系列PLC正式做个项目,用的触摸屏为威纶通的 MT6071iP (注意:下面内容同样适用于 信捷XC系列PLC ,除信捷XC与XD系列编程软件不一样,其余接线设置实测均一样 ) 目前 ...

  7. 基于TCP/IP的Matlab Modbus与M340 PLC通讯

    本人原创,代码拿出来供大家交流学习经验,勿作他用. 废话不多说,代码直接上. 1.创建链接 function link = connect_create(client_addr,port) %**** ...

  8. 欧姆龙plc通讯协议格式

    欧姆龙CPM1A型plc与上位计算机通信的顺序是上位机先发出命令信息给PLC,PLC返回响应信息给上位 机.每次通信发送/接受的一组数据称为一"帧".帧由少于131个字符的数据构成 ...

  9. 基于ModBus-TCP/IT 台达PLC 通讯协议解析

    客户端发送:19 B2 00 00 00 06 06 03 00 27 00 02 上面是modbus客户端发出的报文内容,为modbus tcp/ip协议格式,其前面的六个字节为头字节( heade ...

随机推荐

  1. Java学习(十六)

    今天先学了文本标签 <p> <strong>永远不要相信诺克萨斯人的血条!</strong><!--表示一段内容的重要性--> <br /> ...

  2. 来了!公开揭密团队成员开发鸿蒙 OpenHarmony 的完整过程(收获官方7000奖金和开发板等,1w字用心总结)

    背景 随着 OpenHarmony 组件开发大赛结果公布,我们的团队成员被告知获得了二等奖,在开心之余也想将我们这段时间宝贵的开发经验写下来与大家分享,当我们看到参赛通知的时候已经是 9 月中旬的时候 ...

  3. Hadoop整体概述

    目录 前言 core-site.xml hdfs-site.xml mapred-site.xml yarn-site.xml 一.HDFS HDFS的设计理念 HDFS的缺点 1.NameNode ...

  4. 【Git 系列】基础知识全集

    Git 是一种分布式版本控制系统,它可以不受网络连接的限制,加上其它众多优点,目前已经成为程序开发人员做项目版本管理时的首选,非开发人员也可以用 Git 来做自己的文档版本管理工具. 一.Git 基础 ...

  5. [bzoj1858]序列操作

    考虑建立一棵线段树,维护:1.左端点的连续1和:2.右端点的连续1和:3.最长1的连续子序列:4.1的个数:5.将0和1交换后上面的四项:6.懒标记具体实现中,需要注意细节,可以看代码(比较短) 1 ...

  6. springboot和springcloud版本上的选择

    现在的springboot项目和cloud版本都是更新很快,但我们开发不是版本越新越好,我们要把版本对应起来,那么我们怎么去关联呢? springboot和springcloud不是越新越好,clou ...

  7. vue3 高阶 API 大汇总,强到离谱

    高阶函数是什么呢? 高阶函数英文名叫:Higher Order function ,一个函数可以接收一个或多个函数作为输入,或者输出一个函数,至少满足上述条件之一的函数,叫做高阶函数. 前言 本篇内容 ...

  8. 洛谷 P6276 - [USACO20OPEN]Exercise P(组合数学+DP)

    洛谷题面传送门 废了,又不会做/ll orz czx 写的什么神仙题解,根本看不懂(%%%%%%%%% 首先显然一个排列的贡献为其所有置换环的乘积.考虑如何算之. 碰到很多数的 LCM 之积只有两种可 ...

  9. 洛谷 P6189 - [NOI Online #1 入门组]跑步(根号分治+背包)

    题面传送门 题意: 求有多少个数列 \(x\) 满足: \(\sum x_i=n\) \(x_i\geq x_{i+1}\) 答案对 \(p\) 取模. ...你确定这叫"入门"组 ...

  10. 自定义 Word 模板

    自定义 Word 模板 目录 必要设置 样式设置 标题样式 多级列表 封面 正文 引用目录 页码 页眉 图标 自定义模板保存 样式导入和导出 批量删除多余空白段落 必要设置 显示所有格式标记 选择&q ...