【OpcUA开发笔记 2】open62541在Linux下编译及Qt开发

前言

       在上一篇中,我们记录了在windows下通过CMake编译mingw版本的open62541,事实上是为了这一篇做铺垫,我们本次就在ubuntu下编译open62541,并通过Qt来调用它。

一、编译

1. 建立工程文件夹

   我在主目录中建立文件夹project,并将其设置为具有可读写权限,即cd到该文件夹,然后执行以下命令,这很关键,否则编译好的东西在关掉控制台的时候就消失白干了。

sudo chmod 777 ./

2. 下载源码

git clone https://github.com/open62541/open62541.git

3. 配置

       确保自己安装了g++,cmake之后,在open62541文件夹下建立build文件夹,cd到里面之后,执行以下代码,该选项用于生成open62541.h和open62541.c文件,当然还会生成open62541.a静态库文件,在这三个文件中,可以将.h和.c文件配合使用,也可以将.h和.a文件配合使用,前者是源码,后者是封装,但以下选项不打开为ON,这三者都不会生成。

cmake .. -D UA_ENABLE_AMALGAMATION=ON

 4. 编译

       在该目录下继续执行make:

make

        完成后,build\bin中只有open62541.a文件,如果想要继续编译例子,则需要先关闭UA_ENABLE_AMALGAMATION,再打开UA_BUILD_EXAMPLES,这两个编译时是互斥的。

5. 编译例子

        进入build文件夹,分别执行以下命令:

cmake .. -D UA_ENABLE_AMALGAMATION=OFF
cmake .. -D UA_BUILD_EXAMPLES=ON

然后执行 make,就会发现build\bin文件夹下多了个examples文件夹,你可以用他们试着玩一玩了。

二、用Qt调用open62541

        在windows下就用CMake生成VS版本比较好,一些windows依赖库也就自己加到工程里去了,如果要用Qt,注意以下几点:

1. Qt5建立Non-Qt-Project / C++ Plain Application工程,Qt6目前测试有问题;

2. 要用qMake,这样不用写cMake代码配置工程;

3. 我们将上一节生成的.h和.a文件放到open62541文件夹中并拷贝到工程目录下,在工程中添加库.a和.h文件(或者.c搭配.h文件,这种方式可以改一些配置。另外,在windows下基于mingw编译的.a静态库也能执行,不过要跨平台,在Linux下仍然需要重新编译.a静态库);

4. 如果有using namespace std,要放到 #include "open62541/open62541.h" 上面;

5. 需要在.pro中加入LIBS += -lpthread libwsock32 libws2_32

        本座在ubuntu下用Qt5.14.2调用open62541,跟以上类似,只是不需要第五步。以下给一个Server的例子,该例子来自于源码根目录下的examples\tutorial_server_variable.c,只是我们已经生成了open62541.h就可以干掉源码里的以下两句:

#include <open62541/plugin/log_stdout.h>
#include <open62541/server.h>

用一句 #include "open62541/open62541.h"代替就好了,免得有其他引用问题

//#include <open62541/plugin/log_stdout.h>
//#include <open62541/server.h>

#include "open62541/open62541.h"
#include <stdio.h>

static void
addVariable(UA_Server *server) {
    /* Define the attribute of the myInteger variable node */
    UA_VariableAttributes attr = UA_VariableAttributes_default;
    UA_Int32 myInteger = 42;
    UA_Variant_setScalar(&attr.value, &myInteger, &UA_TYPES[UA_TYPES_INT32]);
    attr.description = UA_LOCALIZEDTEXT("en-US","the answer");
    attr.displayName = UA_LOCALIZEDTEXT("en-US","the answer");
    attr.dataType = UA_TYPES[UA_TYPES_INT32].typeId;
    attr.accessLevel = UA_ACCESSLEVELMASK_READ | UA_ACCESSLEVELMASK_WRITE;

    /* Add the variable node to the information model */
    UA_NodeId myIntegerNodeId = UA_NODEID_STRING(1, "the.answer");
    UA_QualifiedName myIntegerName = UA_QUALIFIEDNAME(1, "the answer");
    UA_NodeId parentNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER);
    UA_NodeId parentReferenceNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES);
    UA_Server_addVariableNode(server, myIntegerNodeId, parentNodeId,
                              parentReferenceNodeId, myIntegerName,
                              UA_NODEID_NUMERIC(0, UA_NS0ID_BASEDATAVARIABLETYPE), attr, NULL, NULL);
}

static void
addMatrixVariable(UA_Server *server) {
    UA_VariableAttributes attr = UA_VariableAttributes_default;
    attr.displayName = UA_LOCALIZEDTEXT("en-US", "Double Matrix");
    attr.accessLevel = UA_ACCESSLEVELMASK_READ | UA_ACCESSLEVELMASK_WRITE;

    /* Set the variable value constraints */
    attr.dataType = UA_TYPES[UA_TYPES_DOUBLE].typeId;
    attr.valueRank = UA_VALUERANK_TWO_DIMENSIONS;
    UA_UInt32 arrayDims[2] = {2,2};
    attr.arrayDimensions = arrayDims;
    attr.arrayDimensionsSize = 2;

    /* Set the value. The array dimensions need to be the same for the value. */
    UA_Double zero[4] = {0.0, 0.0, 0.0, 0.0};
    UA_Variant_setArray(&attr.value, zero, 4, &UA_TYPES[UA_TYPES_DOUBLE]);
    attr.value.arrayDimensions = arrayDims;
    attr.value.arrayDimensionsSize = 2;

    UA_NodeId myIntegerNodeId = UA_NODEID_STRING(1, "double.matrix");
    UA_QualifiedName myIntegerName = UA_QUALIFIEDNAME(1, "double matrix");
    UA_NodeId parentNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER);
    UA_NodeId parentReferenceNodeId = UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES);
    UA_Server_addVariableNode(server, myIntegerNodeId, parentNodeId,
                              parentReferenceNodeId, myIntegerName,
                              UA_NODEID_NUMERIC(0, UA_NS0ID_BASEDATAVARIABLETYPE),
                              attr, NULL, NULL);
}

/**
 * Now we change the value with the write service. This uses the same service
 * implementation that can also be reached over the network by an OPC UA client.
 */

static void
writeVariable(UA_Server *server) {
    UA_NodeId myIntegerNodeId = UA_NODEID_STRING(1, "the.answer");

    /* Write a different integer value */
    UA_Int32 myInteger = 43;
    UA_Variant myVar;
    UA_Variant_init(&myVar);
    UA_Variant_setScalar(&myVar, &myInteger, &UA_TYPES[UA_TYPES_INT32]);
    UA_Server_writeValue(server, myIntegerNodeId, myVar);

    /* Set the status code of the value to an error code. The function
     * UA_Server_write provides access to the raw service. The above
     * UA_Server_writeValue is syntactic sugar for writing a specific node
     * attribute with the write service. */
    UA_WriteValue wv;
    UA_WriteValue_init(&wv);
    wv.nodeId = myIntegerNodeId;
    wv.attributeId = UA_ATTRIBUTEID_VALUE;
    wv.value.status = UA_STATUSCODE_BADNOTCONNECTED;
    wv.value.hasStatus = true;
    UA_Server_write(server, &wv);

    /* Reset the variable to a good statuscode with a value */
    wv.value.hasStatus = false;
    wv.value.value = myVar;
    wv.value.hasValue = true;
    UA_Server_write(server, &wv);
}

/**
 * Note how we initially set the DataType attribute of the variable node to the
 * NodeId of the Int32 data type. This forbids writing values that are not an
 * Int32. The following code shows how this consistency check is performed for
 * every write.
 */

static void
writeWrongVariable(UA_Server *server) {
    UA_NodeId myIntegerNodeId = UA_NODEID_STRING(1, "the.answer");

    /* Write a string */
    UA_String myString = UA_STRING("test");
    UA_Variant myVar;
    UA_Variant_init(&myVar);
    UA_Variant_setScalar(&myVar, &myString, &UA_TYPES[UA_TYPES_STRING]);
    UA_StatusCode retval = UA_Server_writeValue(server, myIntegerNodeId, myVar);
    printf("Writing a string returned statuscode %s\n", UA_StatusCode_name(retval));
}

/** It follows the main server code, making use of the above definitions. */

int main(void) {
    UA_Server *server = UA_Server_new();

    addVariable(server);
    addMatrixVariable(server);
    writeVariable(server);
    writeWrongVariable(server);

    UA_Server_runUntilInterrupt(server);
    UA_Server_delete(server);
    return 0;
}

至此,你发现已经可以运行了,下载一个UaExpert当做客户端来测一下吧。

  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: 基于open62541Qt中编写OPC UA特定的客户端程序,需要按照以下步骤进行: 首先,确保已经安装了Qt开发环境,并在项目中集成了open62541库。可以在Qt的官方网站上下载并安装Qt,然后将open62541库添加到项目的.pro文件中。 其次,需要在Qt中创建一个新的OPC UA客户端程序文件。可以使用Qt的设计器来创建窗口,添加控件和布局。 在程序中,首先需要进行open62541库的初始化。可以调用open62541的API函数来实现,例如ua_client_init()。 接下来,需要连接到OPC UA服务器。可以使用UA_Client_connect()函数并提供服务器的URL地址来连接。 连接成功后,可以浏览服务器上的对象节点。使用UA_Client_getObjectTypes()函数来获取服务器上的对象类型,并使用UA_Client_browse()函数来浏览对象节点。 获取到所需的节点后,可以读取节点的值。通过调用UA_Client_readValueAttribute()函数来读取节点的值。 还可以订阅节点的值变化,以便实时获取数据。可以使用UA_Client_createSubscription()函数来创建一个订阅,然后使用UA_Client_Subscriptions_newNotifications()函数来获取订阅的通知。 在程序中,还可以实现写入节点值的功能。可以使用UA_Client_writeValueAttribute()函数来写入节点的值。 最后,记得在程序结束时关闭客户端连接,并释放open62541库的资源。可以调用UA_Client_disconnect()函数来关闭连接,并调用ua_client_cleanup()函数来释放资源。 通过以上步骤,基于open62541Qt中编写OPC UA特定的客户端程序就可以实现了。 ### 回答2: 基于open62541库在Qt中编写OPC UA特定的客户端程序可以实现与OPC UA服务器的通信和数据交互。以下是一个简要的示例: 首先,需要在Qt中添加open62541库。可以通过下载源代码并手动添加到Qt项目中,或者使用包管理工具安装已经编译好的库。 接下来,创建一个Qt项目并在源代码中包含open62541的头文件。可以使用以下语句包含open62541库的头文件: ``` #include <open62541/client_config_default.h> ``` 然后,通过使用open62541库提供的函数,连接到OPC UA服务器。需要提供服务器的地址和端口号,以及其他必要的配置信息。例如: ``` UA_Client* client = UA_Client_new(); UA_ClientConfig_setDefault(UA_Client_getConfig(client)); UA_StatusCode status = UA_Client_connect(client, "opc.tcp://localhost:4840"); if(status != UA_STATUSCODE_GOOD) { // 连接服务器失败的错误处理 UA_Client_delete(client); return; } ``` 之后,可以使用open62541库提供的函数与OPC UA服务器进行不同的操作,如读取节点值、写入节点值等。例如,读取节点的值可以使用以下代码: ``` UA_NodeId nodeId = UA_NODEID_STRING(1, "nodeId"); UA_Variant value; UA_StatusCode status = UA_Client_readValueAttribute(client, nodeId, &value); if(status == UA_STATUSCODE_GOOD) { // 读取节点值成功 qDebug() << "NodeId: " << UA_NodeId_print(&nodeId); qDebug() << "Value: " << UA_Variant_toScalar(&value)->data; } else { // 读取节点值失败的错误处理 } ``` 最后,记得在不需要连接时关闭客户端并释放资源: ``` UA_Client_disconnect(client); UA_Client_delete(client); ``` 以上就是基于open62541Qt编写OPC UA特定的客户端程序的简要说明。可以通过进一步学习和研究open62541库的文档和示例来深入了解如何在Qt中完整实现OPC UA的客户端功能。 ### 回答3: 基于open62541Qt中编写OPCUA客户端程序需要以下步骤: 1. 首先,我们需要在Qt项目中集成open62541。可以通过CMake或手动将open62541源代码添加到项目中。添加open62541的路径,并将其编译为静态库或动态库。 2. 在Qt项目中创建一个新的类,用于处理OPCUA客户端逻辑。可以命名为"OPCUAClient"。在该类中,包含open62541的头文件,并定义OPCUA客户端的相关功能。 3. 在OPCUAClient类中,首先需要创建一个OPCUA客户端实例。可以使用open62541库提供的函数ua_client_new创建一个新的客户端实例。 4. 在OPCUAClient类中,实现连接到OPCUA服务器的方法。使用open62541库提供的函数ua_client_connect,传入OPCUA服务器的URL和连接的回调函数。在回调函数中,可以实现连接成功或连接失败后的相应操作。 5. 在OPCUAClient类中,实现浏览OPCUA服务器的方法。使用open62541库提供的函数ua_client_browse,传入OPCUA节点的路径和浏览的回调函数。在回调函数中,可以获取浏览结果,如节点的名称、类型等。 6. 在OPCUAClient类中,实现读取OPCUA节点数据的方法。使用open62541库提供的函数ua_client_read,传入要读取的节点和读取的回调函数。在回调函数中,可以获取读取的结果,如节点的值、数据类型等。 7. 在OPCUAClient类中,实现订阅OPCUA节点数据变化的方法。使用open62541库提供的函数ua_client_subscribe,传入要订阅的节点和数据变化的回调函数。在回调函数中,可以获取数据变化的通知,如节点的新值、时间戳等。 8. 在Qt项目的主窗口类中,创建OPCUAClient类的实例,并调用相应的方法进行OPCUA客户端操作。 通过以上步骤,我们可以在Qt中使用open62541库编写OPCUA特定的客户端程序。在OPCUAClient类中实现连接、浏览、读取和订阅OPCUA节点数据的方法,并在主窗口类中进行调用和展示。这样可以实现与OPCUA服务器的通信和数据交互。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Jamie.T

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值