• 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
  • 相关阅读:
    ubuntu之路——day14 只用python的numpy在底层实现多层神经网络
    2019春 软件工程 助教总结
    ubuntu之路——day13 只用python的numpy在较为底层的阶段实现单隐含层神经网络
    ubuntu之路——day12.1 不用tf和torch 只用python的numpy在较为底层的阶段实现简单神经网络
    ubuntu之路——day11.7 end-to-end deep learning
    ubuntu之路——day11.6 多任务学习
    ubuntu之路——day11.5 迁移学习
    mysql 主从复制 (1)
    Windows下Nginx的启动、停止、重启等命令
    mysql-5.7.17-winx64压缩版的安装包下载和安装配置
  • 原文地址:https://www.cnblogs.com/chenxuelian/p/1694623.html
Copyright © 2020-2023  润新知