[QT入门篇]3 QObject的拷贝构造函数与赋值运算符

本文主要是针对QObject的拷贝构造函数和赋值运算符进行说明。先来看一下拷贝构造函数定义:拷贝构造函数,又称复制构造函数,是一种特殊的构造函数,它由编译器调用来完成一些基于同一类的其他对象的构建及初始化。其唯一的形参必须是引用,但并不限制为const,一般普遍的会加上const限制。此函数经常用在函数调用时用户定义类型的值传递及返回。拷贝构造函数要调用基类的拷贝构造函数和成员函数。如果可以的话,它将用常量方式调用,另外,也可以用非常量方式调用。

还记得《[QT入门篇]1 QT中的对象模型》中的说明么?QObject被当做标识对待,不是值,所以不能复制,这和拷贝构造函数的定义是冲突的,所以OQbject没有拷贝构造函数,或者说QObject的拷贝构造函数不能调用。来,看看代码,加深理解。

拷贝构造函数的一般形式为:T(const T& t)

代码1:

#include <QCoreApplication>
#include <QObject>
 
int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);
 
    QObject x;
    x.setObjectName("x");
 
    QObject y(x);
 
    return a.exec();
}

编译时,提示“QObject y(x);”有错误:

=========================================

E:\Qt\Qt5.6.0\5.6\mingw49_32\include\QtCore\qobject.h:461: error: ‘QObject::QObject(const QObject&)‘ is private

Q_DISABLE_COPY(QObject)

^

========================================

从错误提示中,我们能够看出QObject::QObject(const QObject&)是私有的,不能调用。跟踪到OQbject.h中能看到这样的定义:

……

private:

Q_DISABLE_COPY(QObject)

……

毫无疑问,QT把QObject的拷贝构造函数干掉了。同理,QObject的“=”运算符也是一样被干掉了,看下面的代码:

代码2:

#include <QCoreApplication>
#include <QObject>
 
int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);
 
    QObject x;
    x.setObjectName("x");
 
    QObject y;
    y = x;
 
    return a.exec();
}

编译时,“y=x”报错:

========================

E:\Qt\Qt5.6.0\5.6\mingw49_ ‘QObject& QObject::operator=(const QObject&)‘ is private

Class &operator=(const Class &) Q_DECL_EQ_DELETE;

======================

“=”运算符也被置为私有,在qglobal.h中有这样的定义:

……

#define Q_DISABLE_COPY(Class) \

Class(const Class &) Q_DECL_EQ_DELETE;\

Class &operator=(const Class &) Q_DECL_EQ_DELETE;

……

总结一下:QObject既没有拷贝构造函数也没有赋值运算符。QT的设计即是如此。实际上,这两者都被声明了,只不过它们使用了Q_DISABLE_COPY宏并在类的私有段声明的。QObject所有的直接子类和间接子类都没有拷贝构造函数和赋值运算符。

这样做的结果是,开发者在某些场景下需要使用OQbject作为“值”时,必须使用QObject指针传递,而不能使用值传递。

QT为了防止开发人员出错也是煞费苦心啊~

时间: 2024-11-09 17:06:20

[QT入门篇]3 QObject的拷贝构造函数与赋值运算符的相关文章

C++ 拷贝构造函数和赋值运算符

这篇文章主要介绍拷贝构造函数和赋值运算符的区别,以及在什么时候调用拷贝构造函数,什么情况下调用赋值运算符. 拷贝构造函数和赋值运算符 在默认情况下(用户没有定义,但是也没有显示的删除),编译器会自动隐式生成一个拷贝构造函数和赋值运算符,但用户可以使用delete来指定不生成拷贝构造函数和赋值运算符,这样的对象就不能通过值传递,也不能进行赋值运算 1 #include <iostream> 2 3 using namespace std; 4 class Person { 5 public: 6

[Qt入门篇]5 Qt的属性系统——声明属性

[Qt入门篇]5 Qt的属性系统--声明属性 Qt提供了灵活的属性系统,它基于Qt的元对象系统,不依赖于编译器,这保证了Qt独立于编译其和平台的特点.这篇文章主要看看如何声明属性. 属性系统比较复杂,先看一个简单的例子.在QWidget中,有很多属性的声明,找一个简单学习: Q_PROPERTY(boolmodalREADisModal) 这里出现了5个元素:Q_PROPERTY.bool.modal.READ.isModal.这五个元素都是啥作用呢? Q_PROPERTY:用于声明属性的宏:

第五篇:明确拒绝不想编译器自动生成的拷贝构造函数和赋值运算符重载函数

前言 如果你不想要编译器帮你自动生成的拷贝机制 (参考前文),那么你应当明确的拒绝. 如何拒绝?这便是本文要解决的主要问题. 问题描述 当你定义了一个类,而这个类中各对象之间也是封装的 - 禁止同类对象之间的相互赋值以及复制,那么你需要屏蔽掉编译器帮你生成的拷贝构造函数以及赋值运算符. 在许多代码中,会看到通过将拷贝构造函数和赋值运算符重载函数声明为私有且不予实现来实现这个功能.然而,这不是最科学的做法. 因为这没有做到真正的屏蔽:你在自己的成员函数中,或者友元函数中仍然可以调用这两个私有函数,

明确拒绝不想编译器自动生成的拷贝构造函数和赋值运算符重载函数

前言 如果你不想要编译器帮你自动生成的拷贝机制 (参考前文),那么你应当明确的拒绝. 如何拒绝?这便是本文要解决的主要问题. 问题描述 当你定义了一个类,而这个类中各对象之间也是封装的 - 禁止同类对象之间的相互赋值以及复制,那么你需要屏蔽掉编译器帮你生成的拷贝构造函数以及赋值运算符. 在许多代码中,会看到通过将拷贝构造函数和赋值运算符重载函数声明为私有且不予实现来实现这个功能.然而,这不是最科学的做法. 因为这没有做到真正的屏蔽:你在自己的成员函数中,或者友元函数中仍然可以调用这两个私有函数,

不使用编译器自动生成的拷贝构造函数和赋值运算符的方法

方法1:声明私有的拷贝构造函数和赋值运算符,这样不但阻止了编译器生成默认版本,并且使得用户无法调用他们,但是这时成员函数和友元函数还是可以调用他们,为了阻止他们的调用可以不定义这些私有的拷贝构造函数和赋值运算符.(标准库中也是如此阻止拷贝的) 代码段1.1:HomeForSale.h文件 #ifndef HOMEFORSALE_H #define HOMEFORSALE_H class CHomeForSale { public: CHomeForSale(){} private: CHomeF

关注C++细节——含有本类对象指针的类的构造函数、析构函数、拷贝构造函数、赋值运算符的例子

本例只是对含有本类对象指针的类的构造函数.析构函数.拷贝构造函数.复制运算符使用方法的一个简单示例,以加深对构造函数和拷贝控制成员的理解. 读C++ primer 5th 第13章后加上自己的理解,完整的写了下课后习题的代码. 第一版: #include <string> #include <iostream> using namespace std; class TreeNode{ private: string value; TreeNode *left; TreeNode *

拷贝构造函数和赋值运算符重载的区别

拷贝构造函数是用一个已存在的对象去构造一个不存在的对象(拷贝构造函数毕竟还是构造函数嘛),也就是初始化一个对象.而赋值运算符重载函数是用一个存在的对象去给另一个已存在并初始化过(即已经过构造函数的初始化了)的对象进行赋值. 它们定义上的区别,楼上的已经说过了. 比如:String s1("hello"),s2=s1;//拷贝构造函数Sring s1("hello"),s2;s1=s2;//赋值运算符重载以下情况都会调用拷贝构造函数:1.一个对象以值传递的方式传入函数

拷贝构造函数与赋值运算符重载函数要点

拷贝构造函数 一个小例子 最近在<剑指Offer>上看到了一道题(程序如下),要求我们分析编译运行的结果,并提供3个选项: A. 编译错误: B. 编译成功,运行时程序崩溃:C. 编译运行正常,输出10. 1 #include <iostream> 2 using namespace std; 3 4 class A 5 { 6 private: 7 int value; 8 9 public: 10 A(int n) { value = n; } 11 A(A other) {

拷贝构造函数和赋值运算符的调用情况

很多情况下,对于 如下的说明,很难接受 int a(10); 其实完全等价于int a=10; 所以下面的情况更加难以接受: class A;//定义class A 应用: A a; A b(a);//其实完全等价于A b=a,调用的都是拷贝构造函数,在所有人的眼中如果不是,非常熟悉这样赋值的方式,还真是看做是赋值运算. 例子如下: void Test(A a) { } 在这个函数中,通过值的传递,其实调用的是拷贝构造函数,而没有调用赋值运算符,或者默认的构造函数, class T { publ