我将以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的函数去掉就行了,依照葫芦画瓢就行了。
]]>
那我们就来看看在我们的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几行,很好理解,能演示功能。但不鲁棒:)