Java实现OPC通信,看这一篇就够了

1.PLC和OPC

使用的PLC:西门子的S7 300,具体型号如下图

使用的OPC server软件:

  • 模拟仿真用的 MatrikonOPCSimulation(50M),百度网盘,密码: mcur
  • 项目使用KEPServer V6(450M,中文):百度网盘 ,密码: ykj2
  • Github上的:资料下载

2.连接测试

什么是OPC

OPC是工业控制和生产自动化领域中使用的硬件和软件的接口标准,以便有效地在应用和过程控制设备之间读写数据。O代表OLE(对象链接和嵌入),P (process过程),C (control控制)。

OPC标准采用C/S模式,OPC服务器负责向OPC客户端不断的提供数据。

OPC服务器包括3类对象(Object):服务器对象(Server)、组对象(Group)和项对象(Item)。

来源:OPC-(二)-什么是OPC

OPC server软件使用

Server和Client

要实现的是Client(Java)和Client(PLC)之间的通信

中间借助OPCServer,Server上设定好地址变量,不同的Client读写这些变量值实现通信。

示意图如下

配置Server和Client

OPC和DCOM配置:通信不成功都是配置的问题。。。

配置OPCserver
一般一个电脑(win10)同时安装Server(比如KEPServer)和Client(Java编写的),就配置这个电脑就行
如果是在两个电脑上,那就都需要配置。

3.通信实现

Utgard

Github上的

博客参考

4.实现过程

1.补充学习了一下OPC的概念:

2.使用MatrikonOPC,了解OPCserver是怎么用的

3.关于OPC UA

  • 支持的OPC UA的西门子PLC至少是s7-1500
  • 我的s7-300是没法用的,所以就不需要搜集OPC UA的资料了

4.关于用Java实现

  • C#和C++都不用配置DCOM,直接调用函数
  • 既然是非要用Java,那就别想太方便,需要配置DCOM。

5.关于Utgard

  • utgard是一个开源的项目,基于j-interop做的,用于和OPC SERVER通讯。
  • j-interop是纯java封装的用于COM/DCOM通讯的开源项目,这样就不必使用JNI

6.关于JeasyOPC

  • JeasyOPC源码下载
  • 借助一个dll库来实现的和OPCServer的通信,但是JCustomOpc.dll,,太老了,而且支持只32位系统

7.最终实现

  • 当然选Utgard
  • 过程就是把需要的jar包找到,
  • 然后复制编程指导里的读写代码,读就是启动线程一直对相应地址变量读取数值,写就是对相应地址变量写入数值

8.测试

  • 参考OPC_Client里的例子
  • 关于配置文件的代码直接复制用了
  • 例子实际也用不到,试了试,,因为实际只需要对地址变量读写数值就可以了

9.问题:

  • 在虚拟机里用localhost一直报错,要写固定IP才行
  • 配置里的IP是安装OPCServer软件的电脑的IP,如果使用无线连接,请查看无线的IP地址
  • 能不能循环对一个组(group)监控?好像不可以,官方Demo里有两种数据读取方式:1.循环监控item;2.item添加到group,只读取一次
  • 如果Java写的client和安装OPCServer软件是两台电脑:那两个电脑都要配置相同DCOM,包括账号密码都要一样
  • win10家庭版是否可以?可以,有些麻烦,主要是用户管理部分配置,有人已经验证过可以,我就不试了。
  • 关于组态王,作为OPCSerever,我怎么尝试都没连接上,,有人能连上,我就不试了。
  • 关于异步:我使用的同步读取数据,,异步读取没试过,别问我异步的问题。

10.maven依赖

        <!--utgard -->
        <dependency>
            <groupId>org.openscada.external</groupId>
            <artifactId>org.openscada.external.jcifs</artifactId>
            <version>1.2.25</version>
        </dependency>
        <dependency>
            <groupId>org.openscada.jinterop</groupId>
            <artifactId>org.openscada.jinterop.core</artifactId>
            <version>2.1.8</version>
        </dependency>
        <dependency>
            <groupId>org.openscada.jinterop</groupId>
            <artifactId>org.openscada.jinterop.deps</artifactId>
            <version>1.5.0</version>
        </dependency>
        <dependency>
            <groupId>org.openscada.utgard</groupId>
            <artifactId>org.openscada.opc.dcom</artifactId>
            <version>1.5.0</version>
        </dependency>
        <dependency>
            <groupId>org.openscada.utgard</groupId>
            <artifactId>org.openscada.opc.lib</artifactId>
            <version>1.5.0</version>
        </dependency>
        <dependency>
            <groupId>org.bouncycastle</groupId>
            <artifactId>bcprov-jdk15on</artifactId>
            <version>1.61</version>
        </dependency>
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-core</artifactId>
            <version>1.3.0-alpha4</version>
        </dependency>
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
            <version>1.3.0-alpha4</version>
            <scope>test</scope>
        </dependency>

5.代码

下载代码:

截图:

说明

地址变量进行读取数值和写入数值操作,一般分循环和批量两种方式,(同步和异步就不讨论了):

  • 循环读取:Utgard提供了一个AccessBase类来循环读取数值
  • 循环写入:启动一个线程来循环写入数值
  • 批量读取:通过组(Group),增加项(Item)到组,然后对Item使用read()
  • 批量写入:通过组(Group),增加项(Item)到组,然后对Item使用write()

根据实际使用,对例子加了注释,方便理解

读取数值

import java.util.concurrent.Executors;

import org.jinterop.dcom.common.JIException;
import org.jinterop.dcom.core.JIString;
import org.jinterop.dcom.core.JIVariant;
import org.openscada.opc.lib.common.ConnectionInformation;
import org.openscada.opc.lib.da.AccessBase;
import org.openscada.opc.lib.da.DataCallback;
import org.openscada.opc.lib.da.Item;
import org.openscada.opc.lib.da.ItemState;
import org.openscada.opc.lib.da.Server;
import org.openscada.opc.lib.da.SyncAccess;

public class UtgardTutorial1 {

<span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title">main</span><span class="hljs-params">(String[] args)</span> <span class="hljs-keyword">throws</span> Exception </span>{
    <span class="hljs-comment">// 连接信息</span>
    <span class="hljs-keyword">final</span> ConnectionInformation ci = <span class="hljs-keyword">new</span> ConnectionInformation(); 
    ci.setHost(<span class="hljs-string">"192.168.0.1"</span>);         <span class="hljs-comment">// 电脑IP</span>
    ci.setDomain(<span class="hljs-string">""</span>);                  <span class="hljs-comment">// 域,为空就行</span>
    ci.setUser(<span class="hljs-string">"OPCUser"</span>);             <span class="hljs-comment">// 电脑上自己建好的用户名</span>
    ci.setPassword(<span class="hljs-string">"123456"</span>);          <span class="hljs-comment">// 密码</span>

    <span class="hljs-comment">// 使用MatrikonOPC Server的配置</span>
    <span class="hljs-comment">// ci.setClsid("F8582CF2-88FB-11D0-B850-00C0F0104305"); // MatrikonOPC的注册表ID,可以在“组件服务”里看到</span>
    <span class="hljs-comment">// final String itemId = "u.u";    // MatrikonOPC Server上配置的项的名字按实际</span>

    <span class="hljs-comment">// 使用KEPServer的配置</span>
    ci.setClsid(<span class="hljs-string">"7BC0CC8E-482C-47CA-ABDC-0FE7F9C6E729"</span>); <span class="hljs-comment">// KEPServer的注册表ID,可以在“组件服务”里看到</span>
    <span class="hljs-keyword">final</span> String itemId = <span class="hljs-string">"u.u.u"</span>;    <span class="hljs-comment">// KEPServer上配置的项的名字,没有实际PLC,用的模拟器:simulator</span>
    <span class="hljs-comment">// final String itemId = "通道 1.设备 1.标记 1";</span>

    <span class="hljs-comment">// 启动服务</span>
    <span class="hljs-keyword">final</span> Server server = <span class="hljs-keyword">new</span> Server(ci, Executors.newSingleThreadScheduledExecutor());

    <span class="hljs-keyword">try</span> {
        <span class="hljs-comment">// 连接到服务</span>
        server.connect();
        <span class="hljs-comment">// add sync access, poll every 500 ms,启动一个同步的access用来读取地址上的值,线程池每500ms读值一次</span>
        <span class="hljs-comment">// 这个是用来循环读值的,只读一次值不用这样</span>
        <span class="hljs-keyword">final</span> AccessBase access = <span class="hljs-keyword">new</span> SyncAccess(server, <span class="hljs-number">500</span>);
        <span class="hljs-comment">// 这是个回调函数,就是读到值后执行这个打印,是用匿名类写的,当然也可以写到外面去</span>
        access.addItem(itemId, <span class="hljs-keyword">new</span> DataCallback() {
            <span class="hljs-meta">@Override</span>
            <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">changed</span><span class="hljs-params">(Item item, ItemState itemState)</span> </span>{
                <span class="hljs-keyword">int</span> type = <span class="hljs-number">0</span>;
                <span class="hljs-keyword">try</span> {
                    type = itemState.getValue().getType(); <span class="hljs-comment">// 类型实际是数字,用常量定义的</span>
                } <span class="hljs-keyword">catch</span> (JIException e) {
                    e.printStackTrace();
                }
                System.out.println(<span class="hljs-string">"监控项的数据类型是:-----"</span> + type);
                System.out.println(<span class="hljs-string">"监控项的时间戳是:-----"</span> + itemState.getTimestamp().getTime());
                System.out.println(<span class="hljs-string">"监控项的详细信息是:-----"</span> + itemState);

                <span class="hljs-comment">// 如果读到是short类型的值</span>
                <span class="hljs-keyword">if</span> (type == JIVariant.VT_I2) {
                    <span class="hljs-keyword">short</span> n = <span class="hljs-number">0</span>;
                    <span class="hljs-keyword">try</span> {
                        n = itemState.getValue().getObjectAsShort();
                    } <span class="hljs-keyword">catch</span> (JIException e) {
                        e.printStackTrace();
                    }
                    System.out.println(<span class="hljs-string">"-----short类型值: "</span> + n); 
                }

                <span class="hljs-comment">// 如果读到是字符串类型的值</span>
                <span class="hljs-keyword">if</span>(type == JIVariant.VT_BSTR) {  <span class="hljs-comment">// 字符串的类型是8</span>
                    JIString value = <span class="hljs-keyword">null</span>;
                    <span class="hljs-keyword">try</span> {
                        value = itemState.getValue().getObjectAsString();
                    } <span class="hljs-keyword">catch</span> (JIException e) {
                        e.printStackTrace();
                    } <span class="hljs-comment">// 按字符串读取</span>
                    String str = value.getString(); <span class="hljs-comment">// 得到字符串</span>
                    System.out.println(<span class="hljs-string">"-----String类型值: "</span> + str); 
                }
            }
        });
        <span class="hljs-comment">// start reading,开始读值</span>
        access.bind();
        <span class="hljs-comment">// wait a little bit,有个10秒延时</span>
        Thread.sleep(<span class="hljs-number">10</span> * <span class="hljs-number">1000</span>);
        <span class="hljs-comment">// stop reading,停止读取</span>
        access.unbind();
    } <span class="hljs-keyword">catch</span> (<span class="hljs-keyword">final</span> JIException e) {
        System.out.println(String.format(<span class="hljs-string">"%08X: %s"</span>, e.getErrorCode(), server.getErrorMessage(e.getErrorCode())));
    }
}

}

读取数值与写入数值

import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

import org.jinterop.dcom.common.JIException;
import org.jinterop.dcom.core.JIVariant;
import org.openscada.opc.lib.common.ConnectionInformation;
import org.openscada.opc.lib.da.AccessBase;
import org.openscada.opc.lib.da.DataCallback;
import org.openscada.opc.lib.da.Group;
import org.openscada.opc.lib.da.Item;
import org.openscada.opc.lib.da.ItemState;
import org.openscada.opc.lib.da.Server;
import org.openscada.opc.lib.da.SyncAccess;

public class UtgardTutorial2 {

<span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title">main</span><span class="hljs-params">(String[] args)</span> <span class="hljs-keyword">throws</span> Exception </span>{

    <span class="hljs-comment">// 连接信息 </span>
    <span class="hljs-keyword">final</span> ConnectionInformation ci = <span class="hljs-keyword">new</span> ConnectionInformation();
    
    ci.setHost(<span class="hljs-string">"192.168.0.1"</span>);          <span class="hljs-comment">// 电脑IP</span>
    ci.setDomain(<span class="hljs-string">""</span>);                   <span class="hljs-comment">// 域,为空就行</span>
    ci.setUser(<span class="hljs-string">"OPCUser"</span>);              <span class="hljs-comment">// 用户名,配置DCOM时配置的</span>
    ci.setPassword(<span class="hljs-string">"123456"</span>);           <span class="hljs-comment">// 密码</span>
    
    <span class="hljs-comment">// 使用MatrikonOPC Server的配置</span>
    <span class="hljs-comment">// ci.setClsid("F8582CF2-88FB-11D0-B850-00C0F0104305"); // MatrikonOPC的注册表ID,可以在“组件服务”里看到</span>
    <span class="hljs-comment">// final String itemId = "u.u";    // 项的名字按实际</span>

    <span class="hljs-comment">// 使用KEPServer的配置</span>
    ci.setClsid(<span class="hljs-string">"7BC0CC8E-482C-47CA-ABDC-0FE7F9C6E729"</span>); <span class="hljs-comment">// KEPServer的注册表ID,可以在“组件服务”里看到</span>
    <span class="hljs-keyword">final</span> String itemId = <span class="hljs-string">"u.u.u"</span>;    <span class="hljs-comment">// 项的名字按实际,没有实际PLC,用的模拟器:simulator</span>
    <span class="hljs-comment">// final String itemId = "通道 1.设备 1.标记 1";</span>
    
    <span class="hljs-comment">// create a new server,启动服务</span>
    <span class="hljs-keyword">final</span> Server server = <span class="hljs-keyword">new</span> Server(ci, Executors.newSingleThreadScheduledExecutor());
    <span class="hljs-keyword">try</span> {
        <span class="hljs-comment">// connect to server,连接到服务</span>
        server.connect();

        <span class="hljs-comment">// add sync access, poll every 500 ms,启动一个同步的access用来读取地址上的值,线程池每500ms读值一次</span>
        <span class="hljs-comment">// 这个是用来循环读值的,只读一次值不用这样</span>
        <span class="hljs-keyword">final</span> AccessBase access = <span class="hljs-keyword">new</span> SyncAccess(server, <span class="hljs-number">500</span>);
        <span class="hljs-comment">// 这是个回调函数,就是读到值后执行再执行下面的代码,是用匿名类写的,当然也可以写到外面去</span>
        access.addItem(itemId, <span class="hljs-keyword">new</span> DataCallback() {
            <span class="hljs-meta">@Override</span>
            <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">changed</span><span class="hljs-params">(Item item, ItemState state)</span> </span>{
                <span class="hljs-comment">// also dump value</span>
                <span class="hljs-keyword">try</span> {
                    <span class="hljs-keyword">if</span> (state.getValue().getType() == JIVariant.VT_UI4) { <span class="hljs-comment">// 如果读到的值类型时UnsignedInteger,即无符号整形数值</span>
                        System.out.println(<span class="hljs-string">"&lt;&lt;&lt; "</span> + state + <span class="hljs-string">" / value = "</span> + state.getValue().getObjectAsUnsigned().getValue());
                    } <span class="hljs-keyword">else</span> {
                        System.out.println(<span class="hljs-string">"&lt;&lt;&lt; "</span> + state + <span class="hljs-string">" / value = "</span> + state.getValue().getObject());
                    }
                } <span class="hljs-keyword">catch</span> (JIException e) {
                    e.printStackTrace();
                }
            }
        });

        <span class="hljs-comment">// Add a new group,添加一个组,这个用来就读值或者写值一次,而不是循环读取或者写入</span>
        <span class="hljs-comment">// 组的名字随意,给组起名字是因为,server可以addGroup也可以removeGroup,读一次值,就先添加组,然后移除组,再读一次就再添加然后删除</span>
        <span class="hljs-keyword">final</span> Group group = server.addGroup(<span class="hljs-string">"test"</span>); 
        <span class="hljs-comment">// Add a new item to the group,</span>
        <span class="hljs-comment">// 将一个item加入到组,item名字就是MatrikonOPC Server或者KEPServer上面建的项的名字比如:u.u.TAG1,PLC.S7-300.TAG1</span>
        <span class="hljs-keyword">final</span> Item item = group.addItem(itemId);

        <span class="hljs-comment">// start reading,开始循环读值</span>
        access.bind();

        <span class="hljs-comment">// add a thread for writing a value every 3 seconds</span>
        <span class="hljs-comment">// 写入一次就是item.write(value),循环写入就起个线程一直执行item.write(value)</span>
        ScheduledExecutorService writeThread = Executors.newSingleThreadScheduledExecutor();
        writeThread.scheduleWithFixedDelay(<span class="hljs-keyword">new</span> Runnable() {
            <span class="hljs-meta">@Override</span>
            <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">run</span><span class="hljs-params">()</span> </span>{
                <span class="hljs-keyword">final</span> JIVariant value = <span class="hljs-keyword">new</span> JIVariant(<span class="hljs-string">"24"</span>);  <span class="hljs-comment">// 写入24</span>
                <span class="hljs-keyword">try</span> {
                    System.out.println(<span class="hljs-string">"&gt;&gt;&gt; "</span> + <span class="hljs-string">"写入值:  "</span> + <span class="hljs-string">"24"</span>);
                    item.write(value);
                } <span class="hljs-keyword">catch</span> (JIException e) {
                    e.printStackTrace();
                }
            }
        }, <span class="hljs-number">5</span>, <span class="hljs-number">3</span>, TimeUnit.SECONDS); <span class="hljs-comment">// 启动后5秒第一次执行代码,以后每3秒执行一次代码</span>

        <span class="hljs-comment">// wait a little bit ,延时20秒</span>
        Thread.sleep(<span class="hljs-number">20</span> * <span class="hljs-number">1000</span>);
        writeThread.shutdownNow();  <span class="hljs-comment">// 关掉一直写入的线程</span>
        <span class="hljs-comment">// stop reading,停止循环读取数值</span>
        access.unbind();
    } <span class="hljs-keyword">catch</span> (<span class="hljs-keyword">final</span> JIException e) {
        System.out.println(String.format(<span class="hljs-string">"%08X: %s"</span>, e.getErrorCode(), server.getErrorMessage(e.getErrorCode())));
    }
}

}

数组类型

如果地址变量的数据类型是数组类型呢?

// 读取Float类型的数组
if (type == 8196) { // 8196是打印state.getValue().getType()得到的
    JIArray jarr = state.getValue().getObjectAsArray(); // 按数组读取
    Float[] arr = (Float[]) jarr.getArrayInstance();  // 得到数组
    String value = "";
    for (Float f : arr) {
        value = value + f + ",";
    }
    System.out.println(value.substring(0, value.length() - 1); // 遍历打印数组的值,中间用逗号分隔,去掉最后逗号
}

// 写入3位Long类型的数组
Long[] array = {(long) 1,(long) 2,(long) 3};
final JIVariant value = new JIVariant(new JIArray(array));
item.write(value);

读取和写入数值需要按数据类型来操作

这是常用的数据类型

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值