moc 元对象编译器, 全称是 Meta-Object Compiler,也就是“元对象编译器”。是QT翻译扩展语法到C++语言的工具,目前见扩展了信号与槽机制。
信号/槽方式编程上更方便(不容易出错)
回调需要自行处理麻烦的回调管理,稍微不注意就出错。
而且信号/槽方式更利于mvc分离实现。
信号和槽机制的优点:
类型安全, 关联的信号和槽的参数必须是等同。
降低Qt对象间的耦合度,只需要emit,对象无需知道哪个对象来接收该信号,
信号槽的效率和回调函数相比,变低10倍, 原因如下:
1.信号和槽实现
1.Q_OBJECT
Q_OBJECT展开后,会有一个QMetaObject元对象静态类、还有一些元对象操作函数:
signals和slots:
我们以这为例:
预处理之前会调用moc程序,对文件预处理之后生成一个moc_xxx.cpp文件.
如下图所示:
moc会将signals和slots下的函数名转换为字符数组.并生成一个名称idx索引号.
然后生成一个qt_meta_data_Widget(由于类名是Widget,所以后缀是Widget)数组:
其中4, 14,表示有4个方法,然后14表示unit偏移位置,即qt_meta_data_Widget[14]就是第一个方法.
然后并创建一个qt_static_metacall回调函数,实现调用目标类指针的槽函数:
当我调用emit信号时,其实就是调用moc实现的一个信号函数, 信号函数内部调用了QMetaObject::activate()函数:
而activate函数就会去QObjectConnectionListVector连接链表容器里面查找信号对应的索引号所在的值,里面存放了每个接受对象指针和槽函数id的链表:
然后遍历该信号关联的链表里的所有目标对象指针和槽函数,并调用qt_static_metacall回调函数,实现调用槽函数.
connect:
connect会将信号和槽函数字符串化, 然后执行connect的时候会判断信号槽参数是否一致.并遍历”信号槽”字符串的索引号.如果索引号都定义了,则在发送方的连接链表容器的信号索引处,添加一个目标对象指针和槽函数索引号的类到链表中.