QtDBus作为Qt的进程通信

http://www.cuteqt.com/blog Qt技术分享博客 Wed, 24 Mar 2010 12:21:46 +0000 http://wordpress.org/?v=2.9.1 en hourly 1 http://www.cuteqt.com/blog/?p=178 http://www.cuteqt.com/blog/?p=178#comments Wed, 20 May 2009 06:34:10 +0000 臭虫 http://www.cuteqt.com/blog/?p=178 使用QtDBus作为Qt的进程通信中我们已经讨论了怎样利用QtDBus设计一个能够进行进程通信的例子。本文将引入一个新的工具qdbusxml2cpp,通过它可以帮助自动生成继承于QDBusAbstractAdaptor和QDBusAbstractInterface这两个类的实现代码。这两个类分别作用于进程通信服务端和客户端,简化了开发者的代码设计。

我将以http://doc.trolltech.com/4.5/dbus-remotecontrolledcar.html的例子作为分析。
同时本例子也在你的 $QTDIR/examples/dbus/remotecontrolledcar目录下

在开始我们的例子之前,我们讲一下这个car.xml文件。这个文件描述了约定好的一些进程调用函数。我们看一下内容

<!DOCTYPE node PUBLIC "-//freedesktop//DTD D-BUS Object Introspection 1.0//EN"
"http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd">
<node name="/com/trollech/examples/car">
<interface name="com.trolltech.Examples.CarInterface">
<method name="accelerate"/>
<method name="decelerate"/>
<method name="turnLeft"/>
<method name="turnRight"/>
<signal name="crashed"/>
</interface>
</node>

1)qdbusxml2cpp生成QDBusAbstractAdaptor子类的用法为,在shell下输入命令

$ qdbusxml2cpp -c CarAdaptor -a car_adaptor_p.h:car_adaptor.cpp car.xml

QDBusAbstractAdaptor是你要生成的这个子类名,你可以胡编乱造,没人管着你。
car_adaptor_p.h:car_adaptor.cpp是生成的文件名,还是可以随便写没人管得到你。
最后的那个car.xml是你的输入文件。

类文件已经生成,下面就是我们进程通信服务器端.cpp里的内容

new CarAdaptor(car);
QDBusConnection connection = QDBusConnection::sessionBus();
connection.registerObject("/Car", car);
connection.registerService("com.trolltech.CarExample");

这四行代码的后面三行我们在使用QtDBus作为Qt的进程通信里已经讲过了,而且如果我们把第三行改为

connection.registerObject("/Car", car,QDBusConnection::ExportAllSlots);

并把第一行删掉,例子照样能工作。因为QDBusConnection::ExportAllSlots标记把对象car的所有slots都开放给外部进程调用了,这并不是一种很安全的方式,所以connection.registerObject的最后一个参数默认标记是QDBusConnection::ExportAdaptors,也就是Adaptor里指明的那些函数,也就是我们.xml文件里声明的那些函数。

2)qdbusxml2cpp生成QDBusAbstractInterface子类的用法为,在shell下输入命令

$ qdbusxml2cpp -c CarInterface -p car_interface_p.h:car_interface.cpp car.xml

选项”-p”是生成interface(或者说proxy,由它代理发送进程调用的请求)。后面的参数如果我还接着解释就是对大家智商的否定:)

好了,客户端的类文件也生成了,怎么用呢。比服务器端更简单
先创建一个接口的实例(随便在代码的什么地方)

car = new CarInterface("com.trolltech.CarExample", "/Car",QDBusConnection::sessionBus(), this);

接着调用那些约定好的进程调用函数就可以了。比如

car->decelerate();
car->turnLeft();
car->accelerate();

好了,总得来说就是这么简单。细心的朋友可能会发现你这个qdus的.xml的函数都很简单,但是我的函数调用有参数,有返回值怎么写。我找了找我们万能的assitant,也没有发现介绍这个qdus的.xml语法格式介绍的。不过不用担心,有一个工具叫qdbuscpp2xml,你随便找个.h,然后用命令

$ qdbuscpp2xml -A mydbus.h -o hello.xml

把你不想export的函数去掉就行了,依照葫芦画瓢就行了。

]]>
http://www.cuteqt.com/blog/?feed=rss2&p=178 0
http://www.cuteqt.com/blog/?p=167 http://www.cuteqt.com/blog/?p=167#comments Mon, 18 May 2009 02:36:33 +0000 臭虫 http://www.cuteqt.com/blog/?p=167 自从Qt-4.2版本引入D-Bus作为进程通信的一个方法之后,它就越来越有取代qcop的趋势。它不仅在应用层之间可以互相通信,还可以和内核做通信。比如某些硬件事件的发生,就可以通过D-Bus来交互。Linux系统中的蓝牙bluez正是使用了这种D-Bus.

那我们就来看看在我们的Qt应用中怎么使用QtDBus功能。

作为服务器端,我们需要做的工作有:
1. 申请一个总线连接
2. 在总线连接上挂载服务,这样其他进程才能请求该服务
3. 在挂载的服务上注册一个执行服务的对象

作为服务请求端,我们需要做的工作有:
a. 申请一个总线连接
b. 创建一个接口,连接到要请求的服务上
c. 正式发送请求

我们分别来看看对应的步骤,我们应该做什么操作
1.和a.在代码上是完全一样的。

if (!QDBusConnection::sessionBus().isConnected()) {
    fprintf(stderr, "Cannot connect to the D-Bus session bus.\n"
    "To start it, run:\n"
    "\teval `dbus-launch --auto-syntax`\n");
    return 1;
}

sessionBus()是一个静态函数,返回当前的一个QDBusConnection连接,如果原来没有则创建一个.

2. 使用registerService函数去注册,SERVICE_NAME可以随便写,我这里写的是com.cuteqt.blogexample

if (!QDBusConnection::sessionBus().registerService(SERVICE_NAME)) {
    fprintf(stderr, "%s\n",
    qPrintable(QDBusConnection::sessionBus().lastError().message()));
    exit(1);
}

执行完这一步后,你可以使用Qt带的工具,qdbusviewer就能看到com.cuteqt.blogexample已经注册上了

3. qtqt是我的QtQt类的几个对象,将其注册到总线上,这个提供服务的就是QtQt的类函数

QtQt qtqt;
QDBusConnection::sessionBus().registerObject("/", &qtqt, QDBusConnection::ExportAllSlots);

b. 这一步注意SERVICE_NAME的名字和服务器端的代码一样即可,可以判断iface.isValid()

QDBusInterface iface(SERVICE_NAME, "/", "", QDBusConnection::sessionBus());

c. 真正发出具体的总线服务内容请求

QDBusReply<QString> reply = iface.call("blogurl","bug");
if (reply.isValid()) {
printf("Reply was: %s\n", qPrintable(reply.value()));
return 0;
}
else{
fprintf(stderr, "Call failed: %s\n", qPrintable(reply.error().message()));
return 1;
}

我们注意这个blogurl()请求名字就是我们在服务器端注册对象qtqt的类函数,bug是我们传送过去的参数

完整源码下载请点击
服务器端和请求端在一个应用里。运行的时候将一份可执行档拷贝名字为client的就可以。先运行服务端,再运行client。
代码就50几行,很好理解,能演示功能。但不鲁棒:)

]]>
http://www.cuteqt.com/blog/?feed=rss2&p=167 3
原文地址:https://www.cnblogs.com/chenxuelian/p/1694623.html