Linux下学习用C语言实现MQTT(一)(同步函数)

本文介绍MQTT协议的基础知识,包括其在物联网领域的应用,并详细讲解如何在Ubuntu14.04环境下使用C语言实现MQTT的发布与订阅功能。文中提供了安装Paho C库的步骤,以及具体的发布端和订阅端代码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

使用系统版本Ubuntu14.04(该文章代码严谨性并不高,主要用于了解MQTT)
先介绍一下MQTT:
MQTT(Message Queuing Telemetry Transport,消息队列遥测传输)是IBM开发的一个即时通讯协议,有可能成为物联网的重要组成部分。该协议支持所有平台,几乎可以把所有联网物品和外部连接起来,被用来当做传感器和制动器(比如通过Twitter让房屋联网)的通信协议。
MQTT协议是为大量计算能力有限,且工作在低带宽、不可靠的网络的远程传感器和控制设备通讯而设计的协议,它具有以下主要的几项特性:

1、使用发布/订阅消息模式,提供一对多的消息发布,解除应用程序耦合;

2、对负载内容屏蔽的消息传输;

3、使用 TCP/IP 提供网络连接;

4、有三种消息发布服务质量:

“至多一次”,消息发布完全依赖底层 TCP/IP 网络。会发生消息丢失或重复。这一级别可用于如下情况,环境传感器数据,丢失一次读记录无所谓,因为不久后还会有第二次发送。

“至少一次”,确保消息到达,但消息重复可能会发生。

“只有一次”,确保消息到达一次。这一级别可用于如下情况,在计费系统中,消息重复或丢失会导致不正确的结果。

5、小型传输,开销很小(固定长度的头部是 2 字节),协议交换最小化,以降低网络流量;

6、使用 Last Will 和 Testament 特性通知有关各方客户端异常中断的机制;

在这里插入图片描述

使用客户端库的应用程序通常使用类似的结构:
  1.创建一个客户端对象;
  2.设置连接MQTT服务器的选项;
  3.如果多线程(异步模式)操作被使用则设置回调函数(详见 Asynchronous vs synchronous client applications);
  4.订阅客户端需要接收的任意话题;
  5.重复以下操作直到结束:
    a.发布客户端需要的任意信息;
    b.处理所有接收到的信息;
  6.断开客户端连接;
  7.释放客户端使用的所有内存。

接下来开始谈LinuxC编程简单实现MQTT
在linux下用C语言实现MQTT通信,要用到一系列MQTT函数,这些函数在Linux自带库中是没有的。
所以第一步:安装Paho C库
在git下下载paho C库git clone https://github.com/eclipse/paho.mqtt.c.git

cd paho.mqtt.c
make//编译
sudo make install//安装

在编译时可能出现某些头文件的缺失,这里我就不一一列举了,问百度把哈哈哈

在执行make之后,在build/output下能看到这些动态库,这些库中有我们将要用到的函数的定义

zhanghang@Ubuntu-14:~/MQTT/paho.mqtt.c/build/output$ ls
libpaho-mqtt3a.so      libpaho-mqtt3as.so      libpaho-mqtt3c.so      libpaho-mqtt3cs.so      paho_c_version
libpaho-mqtt3a.so.1    libpaho-mqtt3as.so.1    libpaho-mqtt3c.so.1    libpaho-mqtt3cs.so.1    samples
libpaho-mqtt3a.so.1.0  libpaho-mqtt3as.so.1.0  libpaho-mqtt3c.so.1.0  libpaho-mqtt3cs.so.1.0  test

在这里说一下这里面的各个动态库的作用:
paho-mqtt3a : 一般实际开发中就是使用这个,a表示的是异步消息推送(asynchronous)。
paho-mqtt3as : as表示的是 异步+加密(asynchronous+OpenSSL)。
paho-mqtt3c : c 表示的应该是同步(Synchronize),一般性能较差,是发送+等待模式。
paho-mqtt3cs : 同上,增加了一个OpenSSL而已。
在samples中还会有一些示例代码;

zhanghang@Ubuntu-14:~/MQTT/paho.mqtt.c/build/output/samples$ ls
MQTTAsync_publish    MQTTClient_publish        MQTTClient_subscribe  paho_cs_pub  paho_c_sub
MQTTAsync_subscribe  MQTTClient_publish_async  paho_c_pub            paho_cs_sub

现在给发布端的C代码:

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<errno.h>
#include<unistd.h>
#include"MQTTClient.h"

int main(int argc,char **argv)
{
    char *address="tcp://localhost:1883";
    char *client_id="publish_client";
    char *topic="mqtt_examples";
    char buf[1024];
    const int time_out=10000;
    int  rv;
    int  QOS=1;
    MQTTClient   client;
    MQTTClient_connectOptions conn_opts=MQTTClient_connectOptions_initializer;
    MQTTClient_message publish_msg=MQTTClient_message_initializer;
    MQTTClient_deliveryToken token;

    conn_opts.keepAliveInterval=60;
    conn_opts.cleansession=1;

    MQTTClient_create(&client,address,client_id,MQTTCLIENT_PERSISTENCE_NONE,NULL);
    if((rv=MQTTClient_connect(client,&conn_opts))!=MQTTCLIENT_SUCCESS)
    {
    printf("MQTTClient_connect failure:%s\n",strerror(errno));
        return 0;
    }
    publish_msg.qos=QOS;
    publish_msg.retained=0;
    while(1)
    {
        printf("enter the message you want to send\n");
        fgets(buf,sizeof(buf),stdin);
        publish_msg.payload=(void *)buf;
        publish_msg.payloadlen=strlen(buf);
        MQTTClient_publishMessage(client,topic,&publish_msg,&token);
        printf("waiting for %d seconds for publication of %s on topic %s for client with CLIENTID :%s\n",time_out/1000,buf,topic,client_id);
        rv=MQTTClient_waitForCompletion(client,token,time_out);
        printf("Message with delivery token %d delivered\n",rv);
        printf("%s\n",buf);//用于测试
        sleep(3);
    }
}

注意编译时,如果没将相应的头文件和动态库放到编译器默认寻找的位置需要加上相应的链接选项,如下:

zhanghang@Ubuntu-14:~$ gcc mqtt_publish.c -o mqtt_publish -lpaho-mqtt3c -L ./MQTT/paho.mqtt.c/ -I ./MQTT/paho.mqtt.c/src/

运行如下

zhanghang@Ubuntu-14:~$ ./mqtt_publish 
enter the message you want to send

采用交互式传递类似网络soket的传递信息,便于理解

下面给出一个简单的订阅端程序:

#include<stdio.h>
#include<string.h>
#include<unistd.h>
#include<stdlib.h>
#include<errno.h>
#include"MQTTClient.h"

int main()
{
    char *address="tcp://localhost:1883";
    char *client_id="client_sub";
    char *payload="mqtt_examples";
    int    rv,i;
    char *ptr=NULL;
    char *topic=NULL;
    int topic_len;
    MQTTClient client;
    MQTTClient_connectOptions conn_opts=MQTTClient_connectOptions_initializer;
    MQTTClient_deliveryToken token;
    MQTTClient_message *receive_msg=NULL;
    conn_opts.keepAliveInterval=60;
    conn_opts.cleansession=1;

    if((rv=MQTTClient_create(&client,address,client_id,MQTTCLIENT_PERSISTENCE_NONE,NULL))<0)
    {
        printf("MQTTClient_create failure:%s\n",strerror(errno));
        return 0;
    }
    printf("MQTTClient_create successfully\n");
    if((rv=MQTTClient_connect(client,&conn_opts))!=MQTTCLIENT_SUCCESS)
    {
        printf("MQTTClient_connect failure:%s\n",strerror(errno));
        return 0;
    }
    printf("MQTTClient_connect successfuly\n");
    MQTTClient_subscribe(client,payload,1);
    /* if((rv=MQTTClient_receive(client,&topic,&topic_len,&receive_msg,5000))!=MQTTCLIENT_SUCCESS)
    { 
        printf("MQTTClient_receive failure:%s\n",strerror(errno));
        return 0;
    }  
    printf("MQTTClient_receive successfully\n");*/
    //receive 函数放在外面传递信息不会改变
    while(1)
    {
        if((rv=MQTTClient_receive(client,&topic,&topic_len,&receive_msg,100000))!=MQTTCLIENT_SUCCESS)//最后一个参数是超时时间,单位是毫秒
        {
            printf("MQTTClient_receive failure:%s\n",strerror(errno));
            break;
        }
        printf("MQTTClient_receive successfully\n");
        ptr=receive_msg->payload;
        printf("Topic:%s\nTopic_len:%d\nmsg:",topic,topic_len);
        for(i=0;i<receive_msg->payloadlen;i++)
        {
            putchar(*ptr++);
        }
        printf("\nmsg_len:%d\nmsg_id:%d\n",receive_msg->payloadlen,receive_msg->msgid);
        sleep(3);
    }
    printf("end\n");
    MQTTClient_disconnect(client, 10000);
    MQTTClient_destroy(&client);
    return 0;
}

订阅端订阅了话题mqtt_examples,同样采用循环不断接受消息;
编译:

zhanghang@Ubuntu-14:~$ gcc mqtt_subscribe.c -o mqtt_subscribe -lpaho-mqtt3c -L ./MQTT/paho.mqtt.c/build/output/

**注意:在运行时也有可能出现找不到动态库,因为编译是默认动态编译,在运行时才会加载动态库
解决办法:假设需要的库在/test/lib下则可以:

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/test/lib      //该方法重启会失效

如果有sudo去权限也可以将这句话加到~/.bashrc中,重启生效**

此时我们分别在两个终端运行publish和subsceibe:

zhanghang@Ubuntu-14:~$ ./mqtt_publish 
enter the message you want to send
zhanghang@Ubuntu-14:~$ ./mqtt_subscribe 
MQTTClient_create successfully
MQTTClient_connect successfuly

在publish端发送信息,可在subscrib端收到信息:

zhanghang@Ubuntu-14:~$ ./mqtt_publish 
enter the message you want to send
hello
waiting for 10 seconds for publication of hello
 on topic mqtt_examples for client with CLIENTID :publish_client
Message with delivery token 0 delivered
hello

hienter the message you want to send

waiting for 10 seconds for publication of hi
 on topic mqtt_examples for client with CLIENTID :publish_client
Message with delivery token 0 delivered
hi

enter the message you want to send
it is me
waiting for 10 seconds for publication of it is me
 on topic mqtt_examples for client with CLIENTID :publish_client
Message with delivery token 0 delivered
it is me

enter the message you want to send
zhanghang@Ubuntu-14:~$ ./mqtt_subscribe 
MQTTClient_create successfully
MQTTClient_connect successfuly
MQTTClient_receive successfully
Topic:mqtt_examples
Topic_len:13
msg:hello

msg_len:6
msg_id:1
MQTTClient_receive successfully
Topic:mqtt_examples
Topic_len:13
msg:hi

msg_len:3
msg_id:2
MQTTClient_receive successfully
Topic:mqtt_examples
Topic_len:13
msg:it is me

msg_len:9
msg_id:3

使用mosquitto和mqtt.fx软件测试无论是publish还是subscribe都能通过
下一节详细讲解里面用到的一些函数。
(在此处学习过程中遇到过一些问题:在安装paho-mqtt.c后,编译时可能会出现头文件缺失等问题,这种问题百度上有一堆解答,但是介于MQTT系列函数等的特殊性,在网上有关mqttC编程的案例比较少,同时发生错误解答也会比较少,不像网络socket等的一些函数,网上有一堆,所以还是要在以往的学习中积累一些经验,遇到问题解决的办法也会多一些,在成功编译mqtt-paho.c后的make install 主要是将相应的库和头文件移到相应的编译器默认寻找的地方,方便编译和运行。但是由于这个命令需要sudo 权限,如果没有权限或者不愿意用权限,可以像我一样用链接选项和export…
在建立publish和subscribe通信时,publish端发送的消息在subscribe端可能会出现多余字符串“Packet.c”或者其一部分(在源字符串后面)(在直接打印receive_msg->payload时),使用mqtt.fx作为publish也会出现,解决办法(既然知道有效字符穿的长度->payloadlen,可以用循环字符打印,如上程序所示。)

<think>好的,用户现在在Linux环境下想使用Mosquitto的动态库,需要示例代码或者方法。我需要结合之前的引用内容来给出合适的回答。首先,用户提到了动态库的使用,所以可能需要编译、链接以及代码示例的步骤。根据引用1,用户可能已经安装了Mosquitto,但需要确认是否从源码编译,因为动态库通常是在编译过程中生成的。引用2提供了如何在CMake项目中集成Mosquitto的动态库,包括头文件路径和链接库的设置。这可能对用户有用,尤其是使用CMake的情况。引用3提到交叉编译的方法,但用户的问题是在Linux上,可能不需要交叉编译,但可以忽略这部分。引用4展示了C例程的结构,包括头文件和库的位置,这可能对编写代码有帮助。 用户需要动态库的使用示例,所以应该分步骤说明:安装、项目配置、代码示例。安装部分可以参考引用1的下载和编译步骤,但可能需要强调动态库的生成,比如makeinstall是否安装到系统路径。项目配置部分,引用2的CMake配置是关键,需要详细说明如何设置头文件和链接库。代码示例部分,引用4提到了同步函数的C例程,可能需要提供个简单的发布/订阅代码,包含初始化和清理步骤。另外,需要注意用户可能遇到的常见问题,比如动态库路径问题,运行时找不到库的情况,可以通过设置LD_LIBRARY_PATH或者修改rpath来解决,如引用2中的CMAKE_INSTALL_RPATH设置。此外,确保代码中正确包含头文件,并链接到正确的动态库文件,比如libmosquitto.so.1。最后,生成相关问题需要考虑用户可能的后续问题,比如交叉编译、优化连接、异步函数使用等,结合引用中的内容来提问。</think>以下是在Linux系统使用Mosquitto动态库的完整开发指南: ### 、环境准备 1. 安装Mosquitto开发包 ```bash sudo apt-get install libmosquitto-dev # 自动安装头文件和动态库 ``` 或从源码编译安装(参考[^1]): ```bash wget http://mosquitto.org/files/source/mosquitto-2.0.15.tar.gz tar zxvf mosquitto-2.0.15.tar.gz cd mosquitto-2.0.15 make && sudo make install ``` ### 二、CMake项目配置(参考[^2]) ```cmake cmake_minimum_required(VERSION 3.10) project(mqtt_demo) # 设置动态库路径 set(MOSQUITTO_LIB "/usr/local/lib/libmosquitto.so") # 添加可执行文件 add_executable(mqtt_demo main.c) # 链接动态库 target_link_libraries(mqtt_demo ${MOSQUITTO_LIB}) ``` ### 三、C语言示例代码(同步函数参考[^4]) ```c #include <mosquitto.h> #include <stdio.h> int main() { struct mosquitto *mosq = NULL; // 初始化库 mosquitto_lib_init(); // 创建客户端实例 mosq = mosquitto_new("demo-client", true, NULL); if(!mosq){ fprintf(stderr, "Error: Out of memory.\n"); return 1; } // 连接服务器 if(mosquitto_connect(mosq, "localhost", 1883, 60) != MOSQ_ERR_SUCCESS){ fprintf(stderr, "Unable to connect.\n"); return 1; } // 发布消息 mosquitto_publish(mosq, NULL, "test/topic", 6, "hello", 0, false); // 清理资源 mosquitto_destroy(mosq); mosquitto_lib_cleanup(); return 0; } ``` ### 四、编译运行注意事项 1. 运行时若提示找不到动态库,需配置库路径: ```bash export LD_LIBRARY_PATH=/usr/local/lib:$LD_LIBRARY_PATH ``` 2. 验证动态库依赖关系: ```bash ldd mqtt_demo | grep libmosquitto # 应显示类似 libmosquitto.so.1 => /usr/local/lib/libmosquitto.so.1 ```
评论 27
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值