DBUS属性原理

DBus 属性的设置和读取原理主要围绕两个概念:属性的访问和方法调用。以下是详细的解释:

1. DBus 属性的定义和访问

在 DBus 中,属性是通过 property 元素定义的,用于表示对象的状态。属性可以是只读、只写或读写的。定义属性时,您可以指定其名称、类型和访问权限:

cpp 复制代码
<property name="propertyName" type="type" access="access"/>

    name:属性的名称。
    type:属性的数据类型(例如 b 为布尔型,i 为整型)。
    access:属性的访问权限,可以是 read、write 或 readwrite。

2. 读取属性

当客户端请求读取属性时,DBus 框架会调用对象的 getter 方法(通常是 propertyName())。getter 方法从对象中检索属性的当前值,并将其返回给客户端。例子:

cpp 复制代码
bool transparencyState() const {
    return m_pManager->getTransparencyState(); // 返回属性的当前值
}

3. 设置属性

当客户端请求设置属性时,DBus 框架会调用对象的 setter 方法(通常是 setPropertyName(value)),将新值传递给它。setter 方法负责更新对象的内部状态,并可能触发相应的信号。例子:

cpp 复制代码
void setTransparencyState(bool state) {
    m_pManager->setTransparencyState(state); // 更新属性的值
    emit transparencyStateChanged();          // 发射信号通知变化
}

4. 属性与方法的区别

  • 属性:属性提供了一种简单的机制来访问和修改对象的状态。使用属性时,客户端可以像访问普通对象属性一样直接读写这些值。
  • 方法:方法用于执行动作或操作,通常需要传入参数,并可能返回结果。方法调用涉及到调用特定的函数执行任务。

5. 信号

属性的变化可以通过信号通知客户端。使用 NOTIFY 关键字定义的信号会在属性值发生变化时发射,帮助客户端及时更新状态。

示例:

假设有一个名为 transparencyState 的属性,它是一个布尔值:

  • 定义属性:

<property name="transparencyState" type="b" access="readwrite"/>

  • 读取属性:

客户端请求属性值时,DBus 框架调用 transparencyState() 方法。

  • 设置属性:

客户端发送新值时,DBus 框架调用 setTransparencyState(value) 方法,并更新对象内部状态。

  • 发射信号:

如果 transparencyState 属性的值发生变化,transparencyStateChanged 信号会被发射,通知客户端属性已经更新。

6. 函数中发送了信号,是否还需要NOTIFY,不会重复发送信号吗?

实际上,信号不会被重复发送。如果你在 setTransparencyState 中手动发射信号,并且声明了 NOTIFY,信号只会在你手动调用 emit transparencyStateChanged() 时发送。NOTIFY 本身并不会自动发射信号,它只是将信号与属性的变化关联起来,以便 Qt 的元对象系统知道应该监听哪个信号。

NOTIFY 声明不仅用于自动处理属性变化的信号,还用于确保属性变化能够被 Qt 的元对象系统正确处理和通知。

主要原因:

  1. Qt 元对象系统:NOTIFY 声明帮助 Qt 的元对象系统识别哪些信号与属性相关联。这使得其他对象(例如 UI 元素)可以自动连接到这些信号,并在属性变化时做出响应。
  2. 信号与槽机制:通过 NOTIFY,您可以确保在属性值变化时,相关的槽函数能够被自动调用。即使您手动发射信号,NOTIFY 声明可以确保这些变化被正确地传播和处理。
    关键点:
  • 手动发射信号:NOTIFY 声明不会自动触发信号发射。信号的发射仍然完全依赖于你在代码中手动调用 emit。因此,手动发射信号的次数完全由你控制,不会因为有 NOTIFY 就重复发送。
  • NOTIFY作用:NOTIFY 只是告诉 Qt 元对象系统,这个信号是用于通知该属性变化的。它不会自己发射信号,而是让其他组件知道哪个信号与属性的变化相关联。

示例说明:

cpp 复制代码
class MyObject : public QObject {
    Q_OBJECT
    Q_PROPERTY(bool transparencyState READ transparencyState WRITE setTransparencyState NOTIFY transparencyStateChanged)

public:
    bool transparencyState() const {
        return m_transparencyState;
    }

    void setTransparencyState(bool state) {
        if (m_transparencyState != state) {  // 确保只有值发生变化时才发射信号
            m_transparencyState = state;
            emit transparencyStateChanged(); // 手动发射信号
        }
    }

signals:
    void transparencyStateChanged();

private:
    bool m_transparencyState;
};

结论:

  • NOTIFY 声明不会自动发射信号,它仅仅用于标识和关联信号与属性的变化。
  • 信号的实际发射依赖于你手动调用 emit。
  • 因此,手动发射信号时,不会因为 NOTIFY 而产生重复信号的情况。
相关推荐
woshilys25 分钟前
sql server 查询对象的修改时间
运维·数据库·sqlserver
Hacker_LaoYi26 分钟前
SQL注入的那些面试题总结
数据库·sql
疯狂飙车的蜗牛1 小时前
从零玩转CanMV-K230(4)-小核Linux驱动开发参考
linux·运维·驱动开发
建投数据1 小时前
建投数据与腾讯云数据库TDSQL完成产品兼容性互认证
数据库·腾讯云
迷雾漫步者1 小时前
Flutter组件————FloatingActionButton
前端·flutter·dart
向前看-2 小时前
验证码机制
前端·后端
Hacker_LaoYi2 小时前
【渗透技术总结】SQL手工注入总结
数据库·sql
岁月变迁呀2 小时前
Redis梳理
数据库·redis·缓存
独行soc2 小时前
#渗透测试#漏洞挖掘#红蓝攻防#护网#sql注入介绍06-基于子查询的SQL注入(Subquery-Based SQL Injection)
数据库·sql·安全·web安全·漏洞挖掘·hw
燃先生._.3 小时前
Day-03 Vue(生命周期、生命周期钩子八个函数、工程化开发和脚手架、组件化开发、根组件、局部注册和全局注册的步骤)
前端·javascript·vue.js