qt 获取windows 的消息(qt get message from win32 API )

本文介绍了一种Qt应用程序与Windows系统进行消息交互的方法。通过创建MFC DLL并使用透明窗体接收Win32 API消息,实现了Qt应用与Windows平台的双向通信。此方案虽非最优,但在特定场景下提供了可行的解决方案。

qt 给win32 发送消息很简单,但是要获取windows 消息却十分复杂,最后想了一个不是很完美 但是也是以现在本人能力所能实现的唯一途径了,基本原理是 利用vc编写一个mfc 的dll,这个dll 中创建一个透明窗体,利用这个dll 获取win32 API 消息。

源码 已经在vs2010 vs6.0 qt 4.7 下试验通过

下面贴出 重要的实现源码:

VC dll- ReceiveMessage.cpp

#include "stdafx.h"

#include "resource.h"

typedef int (*CALLBACKFUNC)(DWORD Type, DWORD position);


CALLBACKFUNC pfnCallback_tmp;

HINSTANCE hInstance;

unsigned longWM_UNITOKEN_NOTIFY = RegisterWindowMessage("notify_HuFeng");

BOOL APIENTRY DllMain( HANDLE hModule,
DWORD ul_reason_for_call,
LPVOID lpReserved)

{
switch(ul_reason_for_call)
{
case DLL_PROCESS_ATTACH:
hInstance = (HINSTANCE)hModule;
break;
}
return TRUE;
}

HWND m_hWnd = 0;
unsigned long nStep = 0;
unsigned long TotalStep = 0;

//INT_PTR __stdcall

BOOL CALLBACK DialogProcedure(HWND hWnd, UINT Message, WPARAM wParam, LPARAM lParam)

{

switch(Message)

case WM_CREATE:{
break;};
case WM_COMMAND:
{
SendMessage(hWnd, WM_DESTROY, 0, 0);
break;
};
case WM_DESTROY:{PostQuitMessage(0);break;};
case WM_INITDIALOG : { break;};
default :{
if(Message==WM_UNITOKEN_NOTIFY)
{pfnCallback_tmp(wParam,lParam);};break;}
}

return 0;

}

LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)

{

if (message==WM_DESTROY)
{
PostQuitMessage (0) ;
return 0 ;
}
return DefWindowProc (hwnd, message, wParam, lParam) ;

}

DWORD __stdcall ThreadFuction(LPVOID pParam)

{

MSG Msg;

HWND hWnd_HuFeng= CreateDialog(hInstance, MAKEINTRESOURCE(IDD_DIALOGH), NULL, DialogProcedure);

int erro = GetLastError();

ShowWindow(hWnd_HuFeng, SW_HIDE);

while(GetMessage(&Msg, NULL, 0, 0))
{
TranslateMessage(&Msg);
DispatchMessage(&Msg);
}
return 0;

}

extern "C" void __stdcall ReceiveMessageFun(CALLBACKFUNC pfnCallback)

{

DWORDThreadID;

HANDLE m_hThread;

pfnCallback_tmp = pfnCallback;
m_hThread=CreateThread(NULL, 0, ThreadFuction, NULL, 0, &ThreadID);
CloseHandle(m_hThread);

}

QT 接收部分

头文件.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include "windows.h"
#include "qthread.h"
namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow {
Q_OBJECT
public:
MainWindow(QWidget *parent = 0);
~MainWindow();

protected:
void changeEvent(QEvent *e);

private:
Ui::MainWindow *ui;
};

class MyThread : public QThread
{
public:
void run();
void foo();
};


#endif // MAINWINDOW_H

QT 接收部分

receive.cpp

#include <QtGui/QApplication>
#include "mainwindow.h"

#include <QtCore/qobject.h>

#include <limits.h>
#include "QLibrary"
typedef int (* CALLBACKFUNC)(unsigned long Type, unsigned long position);
//extern "C" void __stdcall ReceiveMessageFun(CALLBACKFUNC pfnCallback);
CALLBACKFUNC pfnCallback;

int outputMsg(unsigned long Type, unsigned long position)
{
qDebug("Received Message");
return 0;
}

int main(int argc, char *argv[])
{
int m;
QApplication a(argc, argv);
//MainWindow w;
// outputMsg(12,34);
MyThread aa;
// aa.start();
aa.run();
// w.show();
return a.exec();
}

void MyThread::run()
{
QLibrary myLib("../ReceiveMessage");
if(myLib.load())
{
qDebug("link lib success");}

else
{
qDebug("link lib failed");
}
pfnCallback = outputMsg;
typedef void(*ReceiveMessageFun_HF)(CALLBACKFUNC pfnCallback);//库版本
ReceiveMessageFun_HF pGetLibraryVerion =(ReceiveMessageFun_HF)myLib.resolve("ReceiveMessageFun");
pGetLibraryVerion(pfnCallback);
// sleep(1);
}

QT 发送部分 头文件

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include "windows.h"
#include "stdlib.h"
namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow {
Q_OBJECT
public:
MainWindow(QWidget *parent = 0);
~MainWindow();

protected:
void changeEvent(QEvent *e);
void _SendNotify(DWORD fuccode,DWORD retcode);
private:
Ui::MainWindow *ui;
DWORD a,b;
private slots:
void on_pushButton_clicked();
};

#endif // MAINWINDOW_H

QT 发送部分

receive.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "QFileDialog"
#include "QDebug"
#include "QTextCodec"
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
ui->setupUi(this);
}

MainWindow::~MainWindow()
{
delete ui;
}

void MainWindow::changeEvent(QEvent *e)
{
QMainWindow::changeEvent(e);
switch (e->type()) {
case QEvent::LanguageChange:
ui->retranslateUi(this);
break;
default:
break;
}
}

void MainWindow::on_pushButton_clicked()
{
a=200;
b=50;
MainWindow::_SendNotify(a, b);
}

void MainWindow::_SendNotify(DWORD fuccode, DWORD retcode)
{
unsigned long WM_UNITOKEN_NOTIFY=::RegisterWindowMessageW(L"notify_HuFeng");
PostMessageW(HWND_BROADCAST, WM_UNITOKEN_NOTIFY, fuccode, retcode);
}

代码不尽完善 只是提供一个思路

要通过Qt发送GET请求来获取OneNET云平台设备的最新属性消息,可以按照以下步骤进行: ### 1. 准备请求参数 - **API域名** (`api domain`): `https://iot-api.heclouds.com` - **命名空间** (`namespace`): `thingmodel` - **URL路径** (`url and parameters`): `/query-device-property` ### 2. 构建请求URL 假设你的产品ID为 `S4Zo40WUkk`,设备名为 `MQTT-device-test`,则请求URL为: ``` https://iot-api.heclouds.com/thingmodel/query-device-property?product_id=S4Zo40WUkk&device_name=MQTT-device-test ``` ### 3. 设置请求头 为了进行安全鉴权,你需要在请求头中添加 `Authorization` 字段。具体的授权信息需要从OneNET平台获取。 ### 4. 发送GET请求 以下是使用Qt发送GET请求的示例代码: ```cpp #include <QNetworkAccessManager> #include <QNetworkRequest> #include <QNetworkReply> #include <QUrlQuery> #include <QJsonObject> #include <QJsonDocument> void sendGetRequest() { // 创建网络访问管理器 QNetworkAccessManager *manager = new QNetworkAccessManager(this); // 构建请求URL QUrl url("https://iot-api.heclouds.com/thingmodel/query-device-property"); QUrlQuery query; query.addQueryItem("product_id", "S4Zo40WUkk"); query.addQueryItem("device_name", "MQTT-device-test"); url.setQuery(query.query()); // 创建网络请求 QNetworkRequest request(url); request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json"); // 添加安全鉴权信息 QString authorization = "your_authorization_string"; // 替换为实际的授权信息 request.setRawHeader("Authorization", authorization.toUtf8()); // 发送GET请求 QNetworkReply *reply = manager->get(request); // 连接信号槽处理响应 connect(reply, &QNetworkReply::finished, this, [reply]() { if (reply->error() == QNetworkReply::NoError) { QByteArray response = reply->readAll(); QJsonDocument jsonDoc = QJsonDocument::fromJson(response); QJsonObject jsonObject = jsonDoc.object(); int code = jsonObject.value("code").toInt(); QString msg = jsonObject.value("msg").toString(); QString requestId = jsonObject.value("request_id").toString(); QJsonValue data = jsonObject.value("data"); qDebug() << "Code:" << code; qDebug() << "Message:" << msg; qDebug() << "Request ID:" << requestId; qDebug() << "Data:" << data; // 处理data部分的具体内容 if (code == 0) { QJsonArray dataArray = data.toArray(); for (const QJsonValue &value : dataArray) { QJsonObject item = value.toObject(); QString identifier = item.value("identifier").toString(); QString name = item.value("name").toString(); QString value = item.value("value").toString(); qDebug() << "Identifier:" << identifier << "Name:" << name << "Value:" << value; } } else { qDebug() << "Error occurred: " << msg; } } else { qDebug() << "Network error: " << reply->errorString(); } reply->deleteLater(); }); } ``` ### 5. 处理解析返回的数据 - **code**: 表示API调用的结果,0表示成功。 - **msg**: 表示API调用的字符串信息。 - **request_id**: 当前API的请求ID。 - **data**: 包含详细的返回内容,通常是一个JSON数组,每个元素代表一个设备属性。 ### 6. 错误处理 如果请求失败,可以通过 `reply->error()` 和 `reply->errorString()` 获取错误信息,并进行相应的处理。 以上代码展示了如何使用Qt发送GET请求并解析OneNET云平台返回的设备属性最新消息。希望对你有所帮助!
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值