物联网实战--入门篇之(十)安卓QT--后端开发

news2025/1/19 19:27:24

目录

一、项目配置

二、MQTT连接

三、数据解析

四、数据更新

五、数据发送

六、指令下发


一、项目配置

        按常规新建一个Quick空项目后,我们需要对项目内容稍微改造、规划下。

        首先根据我们的需要在.pro文件内添加必要的模块,其中quick就是qml了,core那是核心模块,必须的,network是网络模块,MQTT需要使用网络。

        然后就是为QML文件和图片文件各自建立一个资源文件,这样编译的时候会把这些资源带上,否则的话打包发布的时候你需要把QML文件和图片文件放到指定文件夹内,这在安卓里就不方便了。

        最后就是如何加载前端QML文件的问题了,如下图所示,后端通过QML加载引擎QQmlApplicationEngine把QML主文件加载进来就能显示界面了,下一行是前端到后端的设置接口名称,这样在QML文件里就可以用theMainInterface这个名称引用后端的函数了,完成开关、调速等动作。

        在新工程默认的文件里,加载前端文件这一步是在main.c文件里完成的,我们这里为了整体前后端的交互,特意建了一个MainInterface的类,在主函数中直接定义这个类的变量即可,这样整个工程结构比较清晰。

二、MQTT连接

        QT标准库里没有mqtt,需要自己单独下载GitCode - 开发者的代码家园,我的项目里已经集成了,只要右键添加进来即可,这里主要是要写个自己的MQTT管理类,把状态保活、话题订阅等任务内部处理掉,就是BaseMqtt类了,里面还有个内容是域名解析需要处理。

#include "BaseMqtt.h"


BaseMqtt::BaseMqtt(QObject *parent) : QObject(parent)
{
    isConnected=false;
    checkTimer = new QTimer(this);
    checkTimer->setInterval(1*100);//心跳检测
    checkTimer->start();
    m_heartTime=0;
    connect(checkTimer, SIGNAL(timeout()),this,SLOT(slotCheckTimeout()));

    m_mqttClient=nullptr;
    m_hostAddress="";
}
BaseMqtt::~BaseMqtt(void)
{
    qDebug()<<"~BaseMqtt, hostName="<<m_connectParam.hostName;
}

void BaseMqtt::slotLookUpHost(QHostInfo info)
{
   if(info.error() == QHostInfo::NoError)
   {
       foreach (QHostAddress address, info.addresses())
       {
            m_hostAddress=address.toString();
            qDebug()<<m_connectParam.hostName<<" mqtt found ip= "<<m_hostAddress;
            break;
       }
   }
}

void BaseMqtt::slotCheckTimeout(void)
{
    m_heartTime++;
    if(m_hostAddress.isEmpty())
    {
        if(m_heartTime%10==0)
        {
            qDebug("mqtt start dns!");
            QHostInfo::lookupHost(m_connectParam.hostName, this, SLOT(slotLookUpHost(QHostInfo)));
        }
        return;
    }
    if(m_mqttClient==nullptr)
    {
        if(!m_connectParam.certPath.isEmpty())//使用SSL连接
        {
            QFile file;
            QByteArray client_key_text, client_crt_text, root_crt_text;
            QString certPath=m_connectParam.certPath;
            QSslSocket ssl_socket;
            
            file.setFileName(certPath+"/client.key");
            file.open(QIODevice::ReadOnly | QIODevice::Text);
            client_key_text = file.readAll();
            file.close();
            
            file.setFileName(certPath+"/client.crt");
            file.open(QIODevice::ReadOnly | QIODevice::Text);
            client_crt_text = file.readAll();
            file.close();

            file.setFileName(certPath+"/rootCA.crt");
            file.open(QIODevice::ReadOnly | QIODevice::Text);
            root_crt_text = file.readAll();
            file.close();
            
            QSslKey client_key(client_key_text, QSsl::Rsa);
            QSslCertificate client_crt(client_crt_text);            

            ssl_socket.setPrivateKey(client_key);
            ssl_socket.setLocalCertificate(client_crt);
            
            QSslConfiguration ssl_config=QSslConfiguration::defaultConfiguration();
            ssl_config.setCaCertificates(QSslCertificate::fromData(root_crt_text)); //QSslCertificate::fromPath(certPath+"/rootCA.crt"); 
            ssl_config.setPrivateKey(ssl_socket.privateKey());
            ssl_config.setLocalCertificate(ssl_socket.localCertificate());
            ssl_config.setPeerVerifyMode(QSslSocket::QueryPeer);
            ssl_config.setPeerVerifyDepth(10);
            ssl_config.setProtocol(QSsl::TlsV1_2);

            m_mqttClient = new QMQTT::Client(m_hostAddress, m_connectParam.hostPort, ssl_config, true, this);
//            qDebug()<<"\n###SSL PrivateKey="<<ssl_config.privateKey();
//            qDebug()<<"###SSL Certificate="<<ssl_config.localCertificate();
//            qDebug()<<"###SSL rootCA="<<ssl_config.caCertificates();
//            qDebug()<<"hostName="<<m_hostAddress<<", hostPort="<<m_connectParam.hostPort;
//            qDebug()<<"userName="<<m_connectParam.userName<<", passWord="<<m_connectParam.passWord<<", clientID="<<m_connectParam.clientID;
        }
        else//普通连接
        {
            QHostAddress host(m_hostAddress);
            m_mqttClient = new QMQTT::Client(host, m_connectParam.hostPort, this);
        }
        signalSlotInit();
        m_mqttClient->setUsername(m_connectParam.userName);
        m_mqttClient->setPassword(m_connectParam.passWord);
        m_mqttClient->setClientId(m_connectParam.clientID);
        m_mqttClient->setAutoReconnect(true);
        m_mqttClient->setCleanSession(true);
        m_mqttClient->setKeepAlive(30);
        m_mqttClient->connectToHost();
    }
    else if(this->mqttIsConnected())
    {
        for(auto iter : m_subTopicList)//订阅话题
        {
            if(iter.isSubed==false)
            {
                mqttSubscribeMessage(iter.subTopic, iter.qos);
                break;
            }
        }
        if(m_heartTime%200==0)//保持连接
        {
//            qDebug()<<"mqtt send keep";
            mqttPublishMessage("dev/sub/data1", QByteArray("heart"));
        }
    }
    else
    {
//        qDebug()<<"BaseMqtt no connected!";
    }
}

void BaseMqtt::mqttConnect(QString hostName, u16 hostPort, QString userName, QByteArray passWord, QString clientID, QString certPath)
{
    clientID=clientID+QString("_")+takeHostMac().remove(":");
    m_connectParam.hostName=hostName;
    m_connectParam.hostPort=hostPort;
    m_connectParam.userName=userName;
    m_connectParam.passWord=passWord;
    m_connectParam.clientID=clientID;
    m_connectParam.certPath=certPath;

    u8 *pData=(u8*)hostName.toUtf8().data();
    if(pData[0]>='0' && pData[0]<='9')//判断是否为域名,使用域名时 域名的第一个字符不能是数字
    {
        m_hostAddress=hostName;
    }
}

bool BaseMqtt::mqttIsConnected(void)
{
//    if(m_mqttClient!=nullptr)
//        return m_mqttClient->isConnectedToHost();
    return isConnected;
}

void BaseMqtt::mqttPublishMessage(QString topicFilter, QByteArray msgBa)
{
    if(m_mqttClient==nullptr || m_mqttClient->isConnectedToHost()==false)
       return;
    QMQTT::Message message;
    message.setTopic(topicFilter);
    message.setPayload(msgBa);
    m_mqttClient->publish(message);
}

void BaseMqtt::mqttPingresp(void)
{
//    m_mqttClient->pingresp();
}

void BaseMqtt::mqttSubscribeMessage(QString topicFilter, quint8 qos)
{
    if(m_mqttClient==nullptr)
       return;
    m_mqttClient->subscribe(topicFilter, qos);
}

void BaseMqtt::mqttUnsubscribeMessage(QString topicFilter)
{
    if(m_mqttClient==nullptr)
       return;
    m_mqttClient->unsubscribe(topicFilter);
}

void BaseMqtt::signalSlotInit(void)
{
    connect(m_mqttClient, SIGNAL(connected()), this, SLOT(slotMqttConnected()));
    connect(m_mqttClient, SIGNAL(disconnected()), this, SLOT(slotMqttDisconnected()));
    connect(m_mqttClient, SIGNAL(error(QMQTT::ClientError)), this, SLOT(slotMqttError(QMQTT::ClientError)));
    connect(m_mqttClient, SIGNAL(pingresp()), this, SLOT(slotMqttPingresp()));
    connect(m_mqttClient, SIGNAL(published(QMQTT::Message,quint16)), this, SLOT(slotMqttPuslished(QMQTT::Message,quint16)));
    connect(m_mqttClient, SIGNAL(received(QMQTT::Message)), this, SLOT(slotMqttReceived(QMQTT::Message)));
    connect(m_mqttClient, SIGNAL(subscribed(QString,quint8)), this, SLOT(slotMqttSubscribed(QString,quint8)));
    connect(m_mqttClient, SIGNAL(unsubscribed(QString)), this, SLOT(slotMqttUnsubscribed(QString)));
}

void BaseMqtt::mqttAddTopic(QString topic, u8 qos)
{
    for(auto iter : m_subTopicList)
    {
        if(iter.subTopic==topic)
        {
            qDebug()<<"have the same topic="<<topic;
            return;
        }
    }
    SubTopicStruct tag_subTopic;
    tag_subTopic.subTopic=topic;
    tag_subTopic.isSubed=false;
    tag_subTopic.qos=qos;
    m_subTopicList.append(tag_subTopic);
    qDebug()<<"mqttAddTopic="<<topic;
}

void BaseMqtt::mqttDelTopic(QString topic)
{
    int i=0;
    for(auto iter : m_subTopicList)
    {
        if(topic==iter.subTopic)
        {
            if(iter.isSubed==true)
            {
                this->mqttUnsubscribeMessage(topic);
            }
            m_subTopicList.removeAt(i);
            qDebug()<<"remove topic="<<topic;
            return;
        }
        i++;
    }
}

void BaseMqtt::slotMqttConnected(void)
{
    isConnected=true;
    emit sigMqttConnected();
    qDebug()<<"clientId="<<m_mqttClient->clientId()<<"connected";
}

void BaseMqtt::slotMqttDisconnected(void)
{
    isConnected=false;
    int nSize=m_subTopicList.size();
    for(int i=0; i<nSize; i++)
    {
        m_subTopicList[i].isSubed=false;
    }
    qDebug()<<"clientId="<<m_mqttClient->clientId()<<"disconnected";
    emit sigMqttDisconnected();
}

void BaseMqtt::slotMqttError(const QMQTT::ClientError error)
{
    qDebug()<<"clientId="<<m_mqttClient->clientId()<<"slotMqttError="<<error;
}

void BaseMqtt::slotMqttPingresp(void)
{
//    qDebug("BaseMqtt::slotMqttPingresp");
}

void BaseMqtt::slotMqttPuslished(const QMQTT::Message &message, quint16 msgid)
{
    msgid=message.id();
    msgid=msgid;
//    qDebug("BaseMqtt::slotMqttPuslished, msgid=%d ", msgid);
//    qDebug()<<"msg="<<message.payload().toHex();
}

void BaseMqtt::slotMqttReceived(const QMQTT::Message &message)
{
    emit sigtMqttReceived(message);

}

void BaseMqtt::slotMqttSubscribed(const QString &topic, const quint8 qos)
{
    int i=0;
//    qDebug()<<"slotMqttSubscribed, topic="<<topic<<", qos="<<qos;
    for(auto iter : m_subTopicList)
    {
        if(iter.subTopic==topic)
        {
            m_subTopicList[i].isSubed=true;
            break;
        }
        i++;
    }
    emit sigMqttSubscribed(topic, qos);
}

void BaseMqtt::slotMqttUnsubscribed(const QString &topic)
{
    int i=0;
    for(auto iter : m_subTopicList)
    {
        if(iter.subTopic==topic)
        {
            m_subTopicList[i].isSubed=false;
            break;
        }
        i++;
    }
    emit sigMqttUnsubscribed(topic);
}

QString BaseMqtt::takeHostMac(void)
{
    DrvCommon drv_com;    
    return drv_com.takeRandMac();
}


        对于应用层就很简单了,就是创建对象、连接和添加订阅话题即可。其中有个槽函数slotMqttReceived就是用来接收设备发来的数据的。

三、数据解析

        数据解析跟嵌入式端是差不多的,下面是代码,像剥洋葱一样,查找帧头、校验、根据命令类型执行解析。


void MainInterface::slotMqttReceived(const QMQTT::Message &message)
{
    QByteArray msg_ba=message.payload();
    u8 *pData=(u8*)msg_ba.data();
//    qDebug()<<"mqtt recv="<<msg_ba.toHex(':');
    u8 head[2]={0xAA, 0x55};
   pData=drv_com.memstr(pData, msg_ba.size(), head, 2);
    if(pData!=nullptr)
    {
      u16 total_len=pData[2]<<8 | pData[3];
      u16 crcValue=pData[total_len]<<8 | pData[total_len+1];
      if(crcValue==drv_com.crc16(pData, total_len))
      {
        pData+=4;
        u32 device_sn=pData[0]<<24|pData[1]<<16|pData[2]<<8|pData[3];
        pData+=4;
        m_currDevSn=device_sn;
        u8 cmd_type=pData[0];
        pData+=1;
        qDebug("recv device_sn=%08X, cmd_type=%d", device_sn, cmd_type);
        m_keepTime=m_secCounts;
        switch(cmd_type)
        {
            case AIR_CMD_HEART:
            {
    
                break;
            }
            case AIR_CMD_DATA:
            {
                int temp=pData[0]<<8|pData[1];//温度 原始数据
                float temp_f=(temp-1000)/10.f;//温度浮点数据
                pData+=2;
                int humi=pData[0]<<8|pData[1];
                float humi_f=humi/10.f;
                pData+=2;
                int pm25=pData[0]<<8|pData[1];
                pData+=2;
                u8 speed=pData[0];
                pData+=1;
                u8 state=pData[0];
                pData+=1;
                qDebug("temp_f=%.1f C, humi_f=%.1f%%, pm25=%d ug/m3, speed=%d, state=%d", temp_f,  humi_f, pm25, speed,state);
                QString dev_sn_str=QString::asprintf("%08X", device_sn);
                QString temp_str=QString::asprintf("%.0f", temp_f);
                QString humi_str=QString::asprintf("%.0f", humi_f);
                QString pm25_str=QString::asprintf("%03d", pm25);
                
                int alarm_level=0;
                if(pm25<20)alarm_level=0;
                else if(pm25<30)alarm_level=1;
                else alarm_level=2;
                
                emit siqUpdateSensorValues(dev_sn_str, temp_str, humi_str, pm25_str);
                emit siqUpdateAlarmLevel(alarm_level);
                emit siqUpdateSwitchState(state);
                break;
            }
            case AIR_CMD_SET_SPEED:
            {
    
                break;
            }
        }
      }
    }
}

        这里我们需要把数据送到前端去显示,所以定义了几个信号内容,如下图所示,从上到下依次是状态数据,污染等级和开关状态,这些数据都是设备端发送上来的,通过后端处理加工后发到前端显示。这里对于污染等级的数值可以自定义,我这边为了方便测试是20、30两个分界线,小米的净化器应该是30和80两条线。

 四、数据更新

        对于前端显示,这里先提一下如何接收后端发来的数据的,如下图所示。以Connections对象为基础,设置它的属性target为theMainInterface,这个其实就是我们加载QML文件时候设置的前后端交互名称,这里用上了,相当于是信号发射者;信号接收器就是C++文件里定义的信号函数前面加个on,然后首字母改成大写就可以了,这里是s改为S,这样这里就能接收到后端发送过来的传感器数据了,很简单吧。至于如何显示,放到前端部分再讲解。

五、数据发送

        数据发送底层就是跟嵌入式端一样,组合后通过mqtt发送出去就行了,有点区别就是这时候要带上目标的序列号dev_sn,这样带有序列号的话题设备端才能收到数据。


void MainInterface::airSendLevel(u32 dev_sn, int cmd_type, u8 *cmd_buff, u16 cmd_len)
{
    u8 make_buff[500]={0};
    u16 make_len=0;
    
    make_buff[make_len++]=0xAA;
    make_buff[make_len++]=0x55;
    make_buff[make_len++]=0;
    make_buff[make_len++]=0;
    make_buff[make_len++]=dev_sn>>24;
    make_buff[make_len++]=dev_sn>>16;
    make_buff[make_len++]=dev_sn>>8;
    make_buff[make_len++]=dev_sn;
    make_buff[make_len++]=cmd_type;
    memcpy(&make_buff[make_len], cmd_buff, cmd_len);
    make_len+=cmd_len;
    make_buff[2]=make_len>>8;
    make_buff[3]=make_len;
    u16 crcValue=drv_com.crc16(make_buff, make_len);
    make_buff[make_len++]=crcValue>>8;
    make_buff[make_len++]=crcValue;
    QByteArray msg_ba((char*)make_buff, make_len);
    QString topic=QString::asprintf("air/dev/sub/%08X", dev_sn);
    if(m_mqttClient)
    {
        m_mqttClient->mqttPublishMessage(topic, msg_ba);
    }
}
六、指令下发

        在应用层,主要就是开关和调速两个功能,这里要看下这两个函数的定义,比较特别,在头文件定义的函数名称前多了Q_INVOKABLE,添加了这个关键字后,这个函数就可以在QML文件里直接调用了,是不是很方便。置于函数的内容应该比较简单了,就是组合下报文给底层函数发送出去就行了,这里的命令定义跟嵌入式端是一样的,两边有改动的话一定要及时同步,不然就乱了。对于调速设置,函数的输入是0~1的浮点数,当小于0.1的时候会强制等于0.1,这样调速最低时才不会停掉,只是慢速转动。

        对于这里的后端,总的来说没什么难点,主要还是做好前后端数据交互的准备。这里有个细节提一下,细心的也会发现,有个定时器槽函数slotCheckTimeout(),虽然这里没什么用,但是在其他有复杂任务或者多线程的时候很有用,它可以定时执行任务,相当于局部的main函数了,以后有机会再慢慢学习。

本项目的交流QQ群:701889554

   写于2024-4-2

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1571964.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

阿里云服务器资费:一年或1个月费用价格,2024年更新

阿里云服务器资费多少钱&#xff1f;一年或1个月费用价格&#xff1a;2核2G3M轻量服务器61元一年、ECS云服务器2核2G3M 99元一年&#xff0c;2核4G轻量165元一年&#xff0c;2核4G ECS 199元一年&#xff0c;阿里云服务器网aliyunfuwuqi.com整理如下&#xff1a; 1、ECS经济型e…

vue 实现的h5 页面,如何设置页面中的 title

修改页面中的title 公共修改方式在App.vue 中&#xff1a; created() {document.title "测试标题"; },单个页面修改&#xff0c;就在单个页面编写就ok

(ISPRS,2023)深度语义-视觉对齐用于zero-shot遥感图像场景分类

文章目录 相关论文摘要引言类别嵌入局限性——问题1普通ZSL模型局限性——问题2自动属性注释过程——对应问题1深度语义-视觉对齐&#xff08;DSVA&#xff09;模型——对应问题2 基于遥感多模态相似性的自动属性标注属性词汇表构造使用CLIP模型自动标注属性对CLIP模型进行训练…

移动Web学习04-移动端订单结算页PC端个人中心页面

5、电商结算页面案例 css body{background-color: #F2F2F2; } * {box-sizing: border-box;margin: 0;padding: 0; }.main{padding: 12px 11px 80px; }.pay{display: flex;height: 80px;background-color: #fff;bottom: 0;width: 100%;border-top: 1px solid #ededed;position:…

腾讯云轻量应用服务器是什么?和普通云服务器有区别吗?

腾讯云轻量应用服务器是什么&#xff1f;轻量应用服务器是开箱即用、面向轻量应用场景的云服务器产品&#xff0c;与之对应的是云服务器CVM&#xff0c;CVM是专业级的云服务器&#xff0c;中小企业、个人开发者可以使用轻量应用服务器在云端构建网站、Web应用、小程序/小游戏、…

计算机网络-TCP重传、滑动窗口、流量控制、拥塞控制

重传机制 超时重传&#xff1a;超时重传时间&#xff08;RTO&#xff09;设定为略大于RTT&#xff08;动态&#xff09;。触发场景包括自己发送的数据包丢失和别人给自己的回应数据包丢失。启动重传机制后如果还没有收到数据包&#xff0c;则RTO设置为上次的两倍&#xff0c;直…

基于顺序表实现通讯管理系统!(有完整源码!)

​​​​​​​ 个人主页&#xff1a;秋风起&#xff0c;再归来~ 文章专栏&#xff1a;C语言实战项目 个人格言&#xff1a;悟已往之不谏&#xff0c;知来者犹可追 克心守己&#xff0c;律己则安&#xff01;​​​​​​​ 目录 1、实现思路 ​…

从B2B转向B2B2C模式:工业品牌史丹利百得的转型历程

图片来源&#xff1a;Twitter 在当今数据驱动的营销环境中&#xff0c;企业努力更好了解客户&#xff0c;并在整个客户旅程中提供个性化体验。史丹利百得&#xff08;Stanley Black & Decker&#xff09;是一家领先的工具和工业设备供应商&#xff0c;近年来开始重大转型。…

Oracle的物理结构解析

这些图是我自己画的&#xff0c;我也会在我的公众号【会用数据库】解析。理解起来非常简单&#xff0c;而且非常好记。不用死记硬背&#xff0c;有兴趣可以来公众号看呀。

深度学习【向量化(array)】

为什么要向量化 在深度学习安全领域、深度学习练习中&#xff0c;你经常发现在训练大量数据时&#xff0c;深度学习算法表现才更加优越&#xff0c;所以你的代码运行的非常快至关重要&#xff0c;否则&#xff0c;你将要等待非常长的时间去得到结果。所以在深度学习领域向量化…

Apache Pulsar源码解析之Lookup机制

引言 在学习Pulsar一段时间后&#xff0c;相信大家也或多或少听说Lookup这个词&#xff0c;今天就一起来深入剖析下Pulsar是怎么设计的它吧 Lookup是什么 在客户端跟服务端建立TCP连接前有些信息需要提前获取&#xff0c;这个获取方式就是Lookup机制。所获取的信息有以下几种…

机器学习实战18-机器学习中XGBClassifier分类器模型的应用实战,以及XGBClassifier分类器的调优策略

大家好&#xff0c;我是微学AI&#xff0c;今天给大家介绍一下机器学习实战18-机器学习中XGBClassifier分类器模型的应用实战&#xff0c;以及XGBClassifier分类器的调优策略。XGBClassifier是基于eXtreme Gradient Boosting (XGBoost)算法的分类器模型&#xff0c;在机器学习领…

成为图像SoC大牛有多难?

IPC&#xff08;Internet Protocol Camera&#xff0c;即网络摄像机&#xff09;芯片架构师需要具备一系列跨学科的知识和技能。IPC芯片架构师的工作涉及到先进工艺、低功耗、SoC架构、处理器架构、图像处理、视频压缩、网络通信以及嵌入式系统设计等多个领域。以下是一些关键的…

Ubuntu22.04中基于Qt开发Android App

文章目录 前言在Ubuntu22.04中配置开发环境案例测试参考 前言 使用Qt开发手机应用程序是一种高效且灵活的选择。Qt作为一个跨平台的开发框架&#xff0c;为开发者提供了统一的开发体验和丰富的功能库。首先&#xff0c;Qt的跨平台性让开发者可以使用相同的代码库在不同的操作系…

收藏|深入浅出分析光刻机

光刻技术是指在光照作用下&#xff0c;借助光致抗蚀剂&#xff08;又名光刻胶&#xff09;将掩膜版上的图形转移到基片上的技术。 光刻机是半导体生产制造的主要生产设备之一&#xff0c;也是决定整个半导体生产工艺水平高低的核心技术机台。半导体技术发展都是以光刻机的光刻线…

【测试开发学习历程】python流程控制

前言&#xff1a;写到这里也许自己真的有些疲惫了&#xff0c;但是人生不就是像西西弗斯推石上山一样的枯燥乏味吗&#xff1f; 在python当中的流程控制主要包含以下两部分的内容&#xff1a; 条件判断 循环 1 条件判断 条件判断用if语句实现&#xff0c;if语句的几种格式…

微软detours代码借鉴点备注

comeasy 借鉴点1 Loadlibray的时间选择 注入库wrotei.dll&#xff0c;为了获取istream的接口&#xff0c;需要loadlibrary&#xff0c;但是在dllmain中是不建议这样做的。因此&#xff0c;动态库在dllmain的时候直接挂载了comeasy.exe的入口 //获取入口 TrueEntryPoint (i…

太阳能光伏储能系统:全周期一站式解决方案

随着全球能源结构的不断变革&#xff0c;清洁能源的重要性日益凸显。太阳能光伏储能系统作为一种高效、环保的能源解决方案&#xff0c;正逐渐成为推动能源转型的关键力量。本文将详细介绍太阳能光伏储能系统的全周期一站式解决方案&#xff0c;以期为读者提供全面、深入的了解…

动态多目标优化:动态约束多目标优化测试集DCP1-DCP9的TruePF(提供MATLAB代码)

一、进化动态约束多目标优化测试集DCP1-DCP9 参考文献&#xff1a; [1]G. Chen, Y. Guo, Y. Wang, J. Liang, D. Gong and S. Yang, “Evolutionary Dynamic Constrained Multiobjective Optimization: Test Suite and Algorithm,” in IEEE Transactions on Evolutionary Com…

聚能共创下一代智能终端操作系统 软通动力荣膺“OpenHarmony优秀贡献单位”

近日&#xff0c;由开放原子开源基金会指导&#xff0c;以“开源共享未来”为主题的OpenHarmony社区年会在北京成功举办。本次活动汇集OpenHarmony项目群共建单位及生态伙伴等多方力量&#xff0c;旨在对2023年度OpenHarmony年度开源事业全面总结的同时&#xff0c;吸引更多伙伴…