【转载】JAVA连接KEPServerEX OPC连接配置说明

本文详细介绍了如何使用JAVA连接到KEPServerEX OPC服务器,包括软件安装、DCOM配置、Maven依赖设置、读写OPC数据的步骤,并提供了常见问题解答,特别是针对Windows 10 2004版以上遇到的问题。
摘要由CSDN通过智能技术生成

 

=====  欢迎加入KEPServerEX QQ讨论群:905485143   ======

 

————————————————
版权声明:本文为CSDN博主「qq_36079837」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_36079837/article/details/115722521

 

=====  欢迎加入KEPServerEX QQ讨论群:905485143   ======

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

 

  1. 安装软件
  2. 配置DCOM-Maven依赖
  3. 获取KepServer内的值-读取OPC数据
  4. 获取KepServer内的值-写入
  5. 常见问题:FAQ

1. 安装软件

  1. KEPServerEX 6
  2. 可以使用虚拟机进行测试
  3. 虚拟机系统可以使用1607版本
  4. 双击打开KEPServerEX 6.4.321.0.exe

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

 

2. 配置DCOM-Maven依赖

参考链接:https://blog.csdn.net/u011051912/article/details/109282856

配置防火墙如觉得麻烦 直接关闭防火墙即可

Maven依赖

    <!--utgard -->
        <dependency>
            <groupId>org.openscada.external</groupId>
            <artifactId>org.openscada.external.jcifs</artifactId>
            <version>1.2.25</version>
            <exclusions>
                <exclusion>
                    <groupId>org.bouncycastle</groupId>
                    <artifactId>bcprov-jdk15on</artifactId>
                </exclusion>
            </exclusions>
        </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>
            <exclusions>
                <exclusion>
                    <groupId>org.bouncycastle</groupId>
                    <artifactId>bcprov-jdk15on</artifactId>
                </exclusion>
            </exclusions>
        </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>
        </dependency>
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
        </dependency>

3. 获取KepServer内的值-读取OPC数据

可参考:https://www.cnblogs.com/ioufev/p/9366877.html
视频教程创建字段:https://www.bilibili.com/video/BV13V411f7Ch/

读取OPC数据

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 {

	public static void main(String[] args) throws Exception {
		// 连接信息
		final ConnectionInformation ci = new ConnectionInformation();
		ci.setHost("192.168.44.128"); // 安装opc电脑IP
		ci.setDomain(""); // 域,为空就行
		ci.setUser("OPCServer"); // 电脑上自己建好的用户名
		ci.setPassword("123456"); // 用户名的密码
		ci.setClsid("7BC0CC8E-482C-47CA-ABDC-0FE7F9C6E729"); // KEPServer的注册表ID,可以在“组件服务”里看到
		// ci.setProgId("");
		// 要读取的标记
		String itemId = "通道 1.设备 2.tag1";
		
		// final String itemId = "通道 1.设备 1.标记 1";

		// 启动服务
		final Server server = new Server(ci, Executors.newSingleThreadScheduledExecutor());

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

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

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

![在这里插入图片描述](https://img-blog.csdnimg.cn/20210415152259669.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzM2MDc5ODM3,size_16,color_FFFFFF,t_70

 

4. 获取KepServer内的值-写入值如下

 



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 {
    
    public static void main(String[] args) throws Exception {
 
        // 连接信息 
        final ConnectionInformation ci = new ConnectionInformation();
        ci.setHost("192.168.44.128");  // 安装opc电脑IP
		ci.setDomain(""); // 域,为空就行
		ci.setUser("OPCServer"); // 电脑上自己建好的用户名
		ci.setPassword("123456"); // 用户名的密码
		ci.setClsid("7BC0CC8E-482C-47CA-ABDC-0FE7F9C6E729"); // KEPServer的注册表ID,可以在“组件服务”里看到
		// ci.setProgId("");
		// 要读取的标记
		String itemId = "通道 2.设备 1.TAG5";
        // create a new server,启动服务
        final Server server = new Server(ci, Executors.newSingleThreadScheduledExecutor());
        try {
            // connect to server,连接到服务
        	System.err.println("启动成功~");
            server.connect();
  
            // add sync access, poll every 500 ms,启动一个同步的access用来读取地址上的值,线程池每500ms读值一次
            // 这个是用来循环读值的,只读一次值不用这样
            final AccessBase access = new SyncAccess(server, 500);
            // 这是个回调函数,就是读到值后执行再执行下面的代码,是用匿名类写的,当然也可以写到外面去
            access.addItem(itemId, new DataCallback() {
                @Override
                public void changed(Item item, ItemState state) {
                    // also dump value
                    try {
                        if (state.getValue().getType() == JIVariant.VT_UI4) { // 如果读到的值类型时UnsignedInteger,即无符号整形数值
                            System.out.println("<<< " + state + " / value = " + state.getValue().getObjectAsUnsigned().getValue());
                        } else {
                            System.out.println("<<< " + state + " / value = " + state.getValue().getObject());
                        }
                    } catch (JIException e) {
                        e.printStackTrace();
                    }
                }
            });
 
            // Add a new group,添加一个组,这个用来就读值或者写值一次,而不是循环读取或者写入
            // 组的名字随意,给组起名字是因为,server可以addGroup也可以removeGroup,读一次值,就先添加组,然后移除组,再读一次就再添加然后删除
            final Group group = server.addGroup("test"); 
            // Add a new item to the group,
            // 将一个item加入到组,item名字就是MatrikonOPC Server或者KEPServer上面建的项的名字比如:u.u.TAG1,PLC.S7-300.TAG1
            final Item item = group.addItem(itemId);
 
            // start reading,开始循环读值
            access.bind();
 
            // add a thread for writing a value every 3 seconds
            // 写入一次就是item.write(value),循环写入就起个线程一直执行item.write(value)
            ScheduledExecutorService writeThread = Executors.newSingleThreadScheduledExecutor();
            writeThread.scheduleWithFixedDelay(new Runnable() {
                @Override
                public void run() {
                    final JIVariant value = new JIVariant("24");  // 写入24
                    try {
                        System.out.println(">>> " + "写入值:  " + "24");
                        item.write(value);
                    } catch (JIException e) {
                        e.printStackTrace();
                    }
                }
            }, 5, 3, TimeUnit.SECONDS); // 启动后5秒第一次执行代码,以后每3秒执行一次代码
 
            // wait a little bit ,延时20秒
            Thread.sleep(20 * 1000);
            writeThread.shutdownNow();  // 关掉一直写入的线程
            // stop reading,停止循环读取数值
            access.unbind();
        } catch (final JIException e) {
            System.out.println(String.format("%08X: %s", e.getErrorCode(), server.getErrorMessage(e.getErrorCode())));
        }
    }
}

 

把helloopc莫这个值修改成24

在这里插入图片描述

=====  欢迎加入KEPServerEX QQ讨论群:905485143   ======

5. 常见问题:FAQ

Message not found for errorCode: 0x80010111 这个错误
出现这个问题是windows问题windows10版本 2004之后的协议都是5.7,utgard的协议是5.6的,把win10降到1607就不会报这个错了

————————————————
版权声明:本文为CSDN博主「qq_36079837」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_36079837/article/details/115722521


 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值