(OPC Client .NET 开发类库)网上很多网友都有提过,.NET开发OPC Client不外乎下面三种方法
1. 背景
OPC Data Access 规范是基于COM/DCOM定义的,因此大多数的OPC DA Server和client都是基于C++开发的,因为C++对COM/DCOM有最好的支持。现在,随着微软的开发平台渐渐的转移到.NET框架上,好多OPC Client程序都需要建立在.NET平台上,用.NET提供的技术开发OPC Client就成为一种需求。网上很多网友都有提过,.NET开发OPC Client不外乎下面三种方法:
- 使用OPCNetAPI 2.0,需要用到OPCNetAPI.dll,OPCNetAPI.Com.dll;
- 使用自动化接口,需要用到OPCDAAuto.dll;
- 使用自定义接口,需要用到多个.NET Wrapper:OpcRcw.Ae.dll,OpcRcw.Batch.dll,OpcRcw.Comn.dll,OpcRcw.Da.dll,OpcRcw.Dx.dll,OpcRcw.Hda.dll,OpcRcw.Sec.dll;
OPCNetAPI 2.0由OPC foundation提供,只有注册会员才能得到,是需要付费的。其他的dll不需要付费,很容易得到。网上有网友已经介绍过使用OPCDAAuto.dll开发.NET Client的方法, 这种方法的优点是比较简单,缺点是不够灵活。本文使用自定义接口,借助OpcRcw.Da.dll,开发出一个OPC .NET Client的类库,可供其他client程序调用。
必要文件:
OpcRcw.Comn.dll --- 包含对IConnectionPointContainer的包装。
OpcRcw.Da.dll ---.NET 对OPC COM 接口 定义的包装。
适应版本:
OPC Data Access specification 2.05
说明:
该类库正在开发中,这是第一个版本,只实现了一些基本功能,好多功能如OPC Browse等还未实现,代码也未经过测试,存在bug在所难免,感兴趣的朋友请继续关注。。。
2. VS2008工程项目文件

- 基本类库视图
下图是OpcDa.Client组件实现的基本类库:


3. 类库实现的基本功能
OpcServer:
|
Connect |
连接OPC Server。 |
|
Disconnect |
断开Server。 |
|
GetStatus |
获得Server的当前状态,返回ServerStatus。 |
|
AddGroup |
添加group |
|
RemoveGroup |
删除group |
|
FindGroupByName |
通过名字获取OpcGroup对象 |
Opc Group:
|
AddItems |
添加Opc Items到组 |
|
RemoveItems |
删除items |
|
AsyncRead |
异步读取Items,调用IOPCAsyncIO2::Read接口 |
|
AsyncWrite |
异步写items,调用IOPCAsyncIO2::Write接口 |
|
Refresh |
刷新当前group,调用IOPCAsyncIO2::Refresh接口 |
|
GetState |
获得当前group状态,返回GroupState |
|
SetState |
设置当前group状态,返回设置后的group状态 |
|
DataChanged |
事件,客户端注册,可用来接收OnDataChange事件 |
4. 类库使用方法
1) 连接OPC DA server
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
string serverProgID = "OPCSample.OpcDa20Server.1"; // opc da sample server prog idstring hostName = string.Empty; //local serverType tp = Type.GetTypeFromProgID(serverProgID);this._opcServer = new OpcServer(tp.GUID.ToString(), hostName);try{ this._opcServer.Connect();}catch (Exception ex){ MessageBox.Show(ex.Message, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);} |
2) 断开OPC Server
|
1
2
3
4
5
|
if (this._opcServer != null) { this._opcServer.Disconnect(); this._opcServer = null; } |
3) 添加Group
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
string groupName = "grp_0"; // group nameint updateRate = 1000;bool active = true;try{ OpcGroup grp = this._opcServer.AddGroup(groupName, updateRate, active); grp.DataChanged += OnDataChange; //register OnDataChange Event}catch (Exception ex){ MessageBox.Show(ex.Message, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);} |
其中OnDataChange 定义为DataChangedEventHandler类型:
|
1
2
3
|
public delegate void DataChangedEventHandler(object subscriptionHandle, object requestHandle, ItemValueResult[] values); private void OnDataChange(object subscriptionHandle, object requestHandle, ItemValueResult[] values); |
4) 删除Group
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
try{ OpcGroup grp = this._opcServer.FindGroupByName(groupName ); if (grp != null) { grp.DataChanged -= OnDataChange; //unregister OnDataChange Event this._opcServer.RemoveGroup(grp); }}catch (Exception ex){ MessageBox.Show(ex.Message, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);} |
5) 添加Items
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
List<string> items = new List<string>();Items.Add("itemname");OpcGroup grp = this._opcServer.FindGroupByName(groupName);if (grp != null){ try { ItemResult[] results = grp.AddItems(items.ToArray()); foreach (ItemResult result in results) { if (result.ResultID.Failed()) { string message = "Failed to add item \'" + result.ItemName + "\'" + " Error: " + result.ResultID.Name; MessageBox.Show(message); } else { AddItemToList(result); // add item to view list } } } // end try catch (Exception ex) { MessageBox.Show(ex.Message, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error); } // end catch} // end if |
6) 删除Items
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
// remove items// List<object> itemsClientHandle = new List<object>(); //if (itemsClientHandle.Count > 0){ try { // get group OpcGroup grp = this._opcServer.FindGroupByName(groupName); if (grp != null) { IdentifiedResult[] results = grp.RemoveItems(itemsClientHandle.ToArray()); for (int i = 0; i < results.Length; i++) { if (results[i].ResultID.Succeeded()) { // remove opc item from server successfully, remove it from list RemoveItemFromList(results[i]); } else { string message = "Remove item \'" + results[i].ItemName + "\' error: " + results[i].ResultID.ToString(); MessageBox.Show(message); } } } } catch (Exception ex) { MessageBox.Show(ex.Message, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error); } // end catch} |
7) 异步读取Items
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
List<object> clientHandles = new List<object>();foreach (ListViewItem lvItem in items){ ItemResult item = (ItemResult)lvItem.Tag; if (item != null) { clientHandles.Add(item.ClientHandle); }}if (clientHandles.Count > 0){ // get group OpcGroup grp = this._opcServer.FindGroupByName(groupName); if (grp != null) { try { IdentifiedResult[] results = grp.AsyncRead(clientHandles.ToArray(), ++this._handle, new ReadCompleteEventHandler(OnReadComplete), out this._request); for (int i = 0; i < results.Length; i++) { if (results[i].ResultID.Failed()) { string message = "Failed to read item \'" + results[i].ItemName + "\' error: " + results[i].ResultID.ToString(); MessageBox.Show(message); } } } catch (Exception ex) { MessageBox.Show(ex.Message); } }} |
其中OnReadComplete用来接收异步读取的数据结果,其定义为:
|
1
|
private void OnReadComplete(object clientHandle, ItemValueResult[] values); |
8) 异步写Items
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
// get group OpcGroup grp = this._opcServer.FindGroupByName(groupName); if (grp != null) { try { IdentifiedResult[] results = grp.AsyncWrite(new ItemValue[] { itemValue }, ++_handle, new WriteCompleteEventHandler(this.OnWriteComplete), out _request); for (int i = 0; i < results.Length; i++) { if (results[i].ResultID.Failed()) { string message = "Failed to write item \'" + results[i].ItemName + "\' error: " + results[i].ResultID.ToString(); MessageBox.Show(message); } } } catch (Exception ex) { MessageBox.Show(ex.Message); } } |
其中OnWriteComplete用来接收异步写的返回结果,其定义为:
private void OnWriteComplete(object clientHandle, IdentifiedResult[] results);
类库的具体使用举例可以参照OpcTestClient程序。
5. OpcTestClient 用户界面

OPC 接口 .NET 包装组件下载:
源代码下载:
(OPC Client .NET 开发类库)网上很多网友都有提过,.NET开发OPC Client不外乎下面三种方法的更多相关文章
- 体温数据上传程序开发+获取时间的三种方法+DB Browser下载及安装
今天开始了体温上传程序的开发 今日所学: 获取时间 (21条消息) (转)安卓获取时间的三种方法_sharpeha的博客-CSDN博客_安卓获取时间 DB Browser安装教程 (20条消息) sq ...
- C#中??和?分别是什么意思? 在ASP.NET开发中一些单词的标准缩写 C#SESSION丢失问题的解决办法 在C#中INTERFACE与ABSTRACT CLASS的区别 SQL命令语句小技巧 JQUERY判断CHECKBOX是否选中三种方法 JS中!=、==、!==、===的用法和区别 在对象比较中,对象相等和对象一致分别指的是什么?
C#中??和?分别是什么意思? 在C#中??和?分别是什么意思? 1. 可空类型修饰符(?):引用类型可以使用空引用表示一个不存在的值,而值类型通常不能表示为空.例如:string str=null; ...
- Java Web开发中用Tomcat部署项目的三种方法
第一种方法:在tomcat中的conf目录中,在server.xml中的,<host/>节点中添加: <Context path="/hello" docBase ...
- LwIP协议栈开发嵌入式网络的三种方法分析
LwIP协议栈开发嵌入式网络的三种方法分析 摘要 轻量级的TCP/IP协议栈LwIP,提供了三种应用程序设计方法,且很容易被移植到多任务的操作系统中.本文结合μC/OS-II这一实时操作系统,以 ...
- struts2开发action 的三种方法以及通配符、路径匹配原则、常量
struts2开发action 的三种方法 1.继承ActionSupport public class UserAction extends ActionSupport { // Action中业务 ...
- 痞子衡嵌入式:在IAR开发环境下将关键函数重定向到RAM中执行的三种方法
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是在IAR开发环境下将关键函数重定向到RAM中执行的三种方法. 嵌入式项目里应用程序代码正常是放在 Flash 中执行的,但有时候也需要将 ...
- Android开发中完全退出程序的三种方法
参考: http://android.tgbus.com/Android/tutorial/201108/363511.shtml Android程序有很多Activity,比如说主窗口A,调用了子窗 ...
- 安卓开发笔记(十八):实现button按钮事件的三种方法
Android开发中有三种主要的方式用于设置View的点击事件,1.创建内部类:2.主类中实现OnClickListener接口:3.使用匿名内部类.这三种方式都用到了OnClickListener接 ...
- Servlet开发的三种方法
第一种 实现 Servlet 接口,需要覆写 Servlet 的5个方法,并将ServletConfig对象保存到类级变量中 package app01a; import java.io.IOExce ...
随机推荐
- JDK的BIO, NIO, AIO
背景知识点我 1. BIO JDK5之前, JDK的IO模式只有BIO(同步阻塞)问题: 因为阻塞的存在, 需对每个请求开启一个线程. 过多的线程切换影响操作系统性能解决: 使用线程池, 处理不过来的 ...
- C++复习12.程序内存管理
程序内存管理 20131006 一个程序在运行期间的内存是如何的对编写程序至关重要,之前整理的C++内存管理的知识和Java程序内存管理的知识.今天我们系统的整理一下程序的内存. 1.一个程序的内存有 ...
- PHP回调函数call_user_func()和call_user_func_array()的使用
call_user_func():把第一个参数作为回调函数调用 用法:call_user_func ( callable $callback [, mixed $parameter [, mixed ...
- OC-字符串、数组、字典总结
#import <Foundation/Foundation.h> /* ∆以NSMutable开头的都是可变类型,以NSMutable开头定义的对象都是可以对数据进行修改: NSStri ...
- 《Effective C++》第3章 资源管理(2)-读书笔记
章节回顾: <Effective C++>第1章 让自己习惯C++-读书笔记 <Effective C++>第2章 构造/析构/赋值运算(1)-读书笔记 <Effecti ...
- Android 屏幕密度适配
Android Icon Size and Location for Apps 分辨率 DPI Density scale 1dp对应像素 1dp对应物理尺寸 Location Icon Size ...
- continue的作用
特别有用,用于循环中-跳过不满足某个条件的某轮循环continue后面的语句
- Kotlin Reference (六) Control Flow
most from reference if表达式 在kotlin中,if是一个表达式,即它返回一个值.kotlin中没有Java中的三元运算符. // Traditional usage var m ...
- 【Keras学习】常见问题与解答
Keras FAQ:常见问题 如何引用Keras? 如果Keras对你的研究有帮助的话,请在你的文章中引用Keras.这里是一个使用BibTex的例子 @misc{chollet2015keras, ...
- 求二叉树的深度 python
二叉树有深度和高度两个属性,一个节点的深度指的是从根节点到该节点路径的长度,根节点的深度为1:一个节点的高度指的是从该节点到叶子节点所有路径上包含节点个数的最大值.叶子节点的高度为1,往上节点的高度依 ...