java opc da 连接 kepserver 报错 踩坑 解决办法

注:da有很大的局限性,并且已经过时,推荐使用ua,可以不用配置dcom并且兼容da,我的另一篇文章里有写 java opc-ua 读写、订阅操作_weixin_45411740的博客-CSDN博客

第一步你需要下载一个kepserver,去官网下载就好,或者找我下载,我这边提供一个无限时长的,

第二步,配置dcom,如果你的kepserver和你的java客户端在一台电脑上,只需要配置这台电脑,如果不在一台电脑上,那么就需要两台电脑都配置,配置Dcom网上教程很多,我这里就不挂了

第三步,java代码编写

读取

package tcb;

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("127.0.0.1");         // 电脑IP
        ci.setDomain("");                  // 域,为空就行
        ci.setUser("jltcopc");             // 电脑上自己建好的用户名
        ci.setPassword("jltc8606");          // 用户名的密码

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

        // 使用KEPServer的配置
        ci.setClsid("7BC0CC8E-482C-47CA-ABDC-0FE7F9C6E729"); // KEPServer的注册表ID,可以在“组件服务”里看到
        final String itemId = "通道 9.设备 1.tag";    // 项的名字按实际,没有实际PLC,用的模拟器:simulator
        // 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);
            // 这是个回调函数,就是读到值后执行这个打印,是用匿名类写的,当然也可以写到外面去
            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())));
        }
    }
}

写
package tcb;

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("127.0.0.1");          // 电脑IP
        ci.setDomain("");                   // 域,为空就行
        ci.setUser("jltcopc");              // 用户名,配置DCOM时配置的
        ci.setPassword("jltc8606");           // 密码

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

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

        // create a new server,启动服务
        final Server server = new Server(ci, Executors.newSingleThreadScheduledExecutor());
        try {
            // connect to server,连接到服务
            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())));
        }
    }
}
这里通常会出现两个错误点

1.00000005,这个错误是由于账户密码错误导致

 解决办法如下,

1.检查用户民密码,

2.检查有没有配置到Dcom组

3.配置注册表访问权限,具体步骤如下

右键点击权限

点击添加,高级,查找,找到你dcom配置的账户,应用,确定。

不能保存时,点击高级,点击更改,修改所有者为当前登录的用户

4.点击本地安全策略,把网络访问修改为经典

至此,0000005的问题应该可以全部解决

2.80010111问题

这个问题其实是由于电脑版本的问题,windows 2004之后的协议都是5.7,utgard的协议是5.6的,把windows降到1909就不会报这个错了

 

在这里查看电脑版本,不能回退版本的只能重装了,我这边是在公司的云服务器上搭建了一个虚拟机,然后重新装了一个win10之后测试成功

  • 15
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 在Java连接Kepserver OPC UA加密,首先需要确保你的Java项目中已经导入了支持OPC UA的相关库文件。然后,可以按照以下步骤进行连接: 1. 创建一个OPC UA TCP连接的URL地址,地址格式为:opc.tcp://{服务器IP}:{端口号}。在这个URL中,需要将服务器的IP地址和端口号替换为实际的Kepserver服务器地址和端口号。 2. 使用创建的URL地址创建一个OPC UA的连接对象,使用`new UaTcpClient(url)`方法来创建,其中`url`为上一步创建的URL地址。 3. 在连接对象上调用`setSecurityPolicy`方法来设置安全策略,常用的策略有`SecurityPolicy.SuchAsBasic128Rsa15`和`SecurityPolicy.SuchAsBasic256`。根据实际情况选择合适的安全策略。 4. 设置用户凭证,如果Kepserver服务器设置了用户名和密码,可以通过调用连接对象的`setUserIdentity`方法来设置用户凭证,示例代码如下: ```java UsernameProviderImpl usernameProvider = new UsernameProviderImpl(username, password); client.setUserIdentityProvider(usernameProvider); ``` 其中`UsernameProviderImpl`是自定义的类,实现了`UserIdentityProvider`接口,用于提供用户名和密码。 5. 连接Kepserver服务器,调用连接对象的`connect`方法来建立连接,示例代码如下: ```java client.connect().get(); ``` 其中`get`方法是为了等待连接结果,如果连接成功,则会返回`CompletableFuture`对象,可以通过调用`get`方法来判断连接是否成功。 至此,你已经成功使用Java连接到了Kepserver OPC UA加密服务器。接下来,你可以使用OPC UA客户端API来读取、写入和订阅服务器上的数据了。 请注意,以上步骤仅为示例,实际使用时可能需要根据具体情况进行适当调整。 ### 回答2: Java可以通过使用OPC-UA库来连接Kepserver OPC-UA加密。首先,需要在项目中添加OPC-UA库的依赖。然后,在Java代码中创建一个OPC-UA连接对象,配置连接参数,并使用正确的地址和端口连接Kepserver OPC-UA服务器。 连接Kepserver OPC-UA服务器时,需要注意以下几点: 1. 确保Kepserver OPC-UA服务器已启用加密功能,并配置了正确的安全策略。 2. 在Java代码中,需要使用正确的加密算法和密钥长度来配置安全策略。 3. 客户端和服务器之间的证书验证也是关键。需要正确配置服务器的证书,并在Java代码中导入和配置客户端的证书,以便双方能够互相验证身份。 4. 在Java代码中,使用正确的用户名和密码进行身份验证,以及授权访问服务器的操作。 完成上述步骤后,您可以使用Java代码来读取和写入Kepserver OPC-UA服务器上的变量和标签。通过从服务器读取加密配置信息,并在Java代码中正确设置加密参数,您可以确保通过加密通信连接Kepserver OPC-UA服务器,保护数据的安全性和机密性。 ### 回答3: Java 连接 KepServer OPC UA 加密的步骤如下。 1. 首先,需要使用 JavaOPC UA SDK 来创建一个 OPC UA 客户端。你可以选择一些开源的 SDK,比如 Eclipse Milo 或者 Prosys OPC UA Java SDK。 2. 在 Java 代码中,你需要导入相应的 SDK,以及其他必要的类库和依赖项。 3. 配置客户端连接参数,包括 KepServer 的 IP 地址、端口号和加密设置。通常情况下,KepServer 在默认情况下是不启用加密的,所以你需要先在 KepServer 的配置文件中启用加密功能。启用加密功能后,你需要提供相应的证书和密钥,以及声明要使用的加密算法。 4. 在 Java 代码中,使用 OPC UA SDK 提供的 API 创建一个 OPC UA 客户端对象,并使用配置好的连接参数初始化该对象。 5. 连接KepServer,使用客户端对象的连接方法进行连接。在连接过程中,客户端会将之前配置的加密信息发送给 KepServer,以便建立一个安全的加密连接。 6. 连接成功后,你可以使用客户端对象的方法来读取和写入 KepServer 上的 OPC UA 变量值。你可以使用节点 ID 或者节点名称来指定要读取和写入的变量。 7. 当你完成所有的操作后,使用客户端对象的断开连接方法来断开连接。 总结:要在Java代码中连接KepServer的加密的OPC UA服务,首先需要使用相应的SDK创建一个OPC UA客户端对象,并设置连接参数以启用加密功能。然后使用该客户端对象连接KepServer,并使用其提供的方法进行数据读取和写入等操作。最后,在完成所有操作后断开连接
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值