文中一系列思考和内容引发自以下问题:
我需要在一个类的构造函数中调用另一个对象的构造函数,并使用this初始化其中的一个引用成员。

主要遇到的问题:

1. 构造函数的初始化列表中能访问this吗?

很明显c++创建一个对象分为两部分,创建内存和调用构造函数。
显然在初始化列表中,当前对象占用的内存已经创建好了,ok,this是可以访问的,只是其中的某些成员是没有初始化的(因为没有构造函数还没执行完,只能说对象是部分有效的)。

那也间接说明一个问题,在构造函数的函数体中使用this是完全可靠的,所有编译器可以自动初始化的成员都完成了初始化(比如基类的成员、带有默认构造函数的成员,注意具体初始化顺序是由类成员定义顺序确定的),但当前类的部分未在初始化列表中初始化的且没有默认构造函数类成员变量(比如常见的c++内置类型,int、float、指针等)的值是未定义的。

2. 构造函数参数与成员同名

构造函数的形式如下:带有一个和类成员同名的参数。打印输出只是为了验证成员变量是否初始化。

class A
{
public:
A(int a): a(a){cout<<a<<endl;}
private:
int a;
};

由于初始化列表中不能直接出现this,所以编译器会处理这种重名的情况。也就说,你不能在构造函数的初始化列表中显式用this做限定符,比如下面代码是无法通过编译的:

A:this->a(a){}

3. 如何在一个类的构造函数汇中调用另一个构造函数

构造函数是不允许嵌套调用的,但可以调用不同的重载形式。比如下面代码:(注意这是一道面试题目)

struct CLS
{
int m_i;
CLS( int i ) : m_i(i){}
CLS(){CLS();}
};
int main()
{
CLS obj;
cout << obj.m_i << endl;
return ;
}

输出是多少?

---------------------------------------------------------------------------

答案是未知,因为m_i是未初始化的变量,是个野值。

"CLS(0);"的语句表示创建一个临时的CLS对象,并把该对象的成员m_i初始化为0。当前对象的值并没有初始化。

如果需要实现构造函数类调用另一个构造函数,需要借助于placement new运算符。代码如下:

struct CLS
{
int m_i;
CLS( int i ) : m_i(i){}
CLS()
{
new (this)CLS();
}
};

如果你对placement new不了解,建议看看c++ primer或者TCPL。

在c++11中可以直接通过委托或继承构造函数的形式实现上面功能。

struct CLS
{
int m_i;
CLS( int i ) : m_i(i){}
CLS():CLS(){}
};

4. 解决方案

写到这里。我对于构造函数的初始化列表中引用this的情况基本了解,可以用下面代码解决本文开始提出的问题。

class Context;
class Ref
{
public:
Ref(Context&context):context(context){}
private:
Context &context;
};
class Context
{
public:
Context():ref(*this){}
private:
Ref ref;
};

复述下开始的问题:我需要在一个类的构造函数中调用另一个对象的构造函数,并使用this初始化其中的一个引用成员。

类Context会在其构造函数的初始化列表中通过this调用Ref的构造函数。

5. 参考资料

[1] 从一道题谈C++中构造函数调用构造函数    http://www.cnblogs.com/chio/archive/2007/10/20/931043.html

[2] c++ 一个构造函数 调用 另一个 构造函数    http://www.cnblogs.com/ayanmw/archive/2012/08/20/2647808.html

关于构造函数和this调用的思考的更多相关文章

  1. 尝试解决在构造函数中同步调用Dns.GetHostAddressesAsync()引起的线程死锁

    (最终采用的是方法4) 问题详情见:.NET Core中遇到奇怪的线程死锁问题:内存与线程数不停地增长 看看在 Linux 与 Windows 上发生线程死锁的后果. Linux: Microsoft ...

  2. C++构造函数的自动调用(调用一个父类的构造函数,有显性调用最好,否则就默认调用无参数的构造函数)——哲学思想:不调用怎么初始化父类的成员数据和VMT?

    我总是记不住构造函数的特点,关键还是没有领会那个哲学思想:父类的构造函数一方面要初始化它自己的成员数据,另一方面也要建立它自己的VMT呀!心里默念一百遍:一定调用父类构造函数,一定调用父类构造函数,一 ...

  3. C++类继承中的构造函数和析构函数 调用顺序

    思想: 在C++的类继承中,构造函数不能被继承(C11中可以被继承,但仅仅是写起来方便,不是真正的继承) 建立对象时,首先调用基类的构造函数,然后在调用下一个派生类的构造函数,依次类推: 析构对象时, ...

  4. JavaScript中以构造函数的方式调用函数

    转自:http://www.cnblogs.com/Saints/p/6012188.html 构造器函数(Constructor functions)的定义和任何其它函数一样,我们可以使用函数声明. ...

  5. C++构造函数中不能调用虚函数

    在构造函数中调用虚函数,并不会产生多态的效果,就跟普通函数一样. c++ primer 第四版中497页15.4.5构造函数和析构中的虚函数讲到,如果在构造函数或析构函数中调用虚函数,则运行的是为构造 ...

  6. java中父类与子类, 不同的两个类中的因为构造函数由于递归调用导致栈溢出问题

    /* 对于类中对成员变量的初始化和代码块中的代码全部都挪到了构造函数中, 并且是按照java源文件的初始化顺序依次对成员变量进行初始化的,而原构造函数中的代码则移到了构造函数的最后执行 */ impo ...

  7. 关于预测io调用的思考

    什么是预测io 预测io是linux2.6版本内核调用默认的调用程序,对应用程序进行跟踪,统计应用程序使用io情况,在读操作返回之前先停顿6ms时间(linux默认时间),如果这期间有读操作过来,可以 ...

  8. 转:C++何时调用构造函数,何时调用析构函数

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明.本文链接:https://blog.csdn.net/wjf1997/article/detai ...

  9. 关于父类私有属性在子类构造函数中super调用的解释

    package test; public class Car { private int carMoney; //汽车租金 private String carName; //汽车名字 private ...

随机推荐

  1. TestNg的IReporter接口的使用

    IReporter接口是干嘛的?就是让用户自定义报告的,很多人想要自定义报告,于是乎找各种插件,比如什么testng-xslt啊,reportng啊,各种配置,最后出来的结果,还不能定制化,但为什么不 ...

  2. magento 为用户注册增加一个字段(转)

    步骤 I. 加一个occupation/title字段到用户注册页,差不多在register.html的54行,在email下方加一个Occupation显示代码 代码: <li>< ...

  3. 定制jQuery File Upload为微博式单文件上传

    日志未经声明,均为AlloVince原创.版权采用『 知识共享署名-非商业性使用 2.5 许可协议』进行许可. jQuery File Upload是一个非常优秀的上传组件,主要使用了XHR作为上传方 ...

  4. mysql单机多实例

    在数据库服务器上,可以架构多个Mysql服务器,进行单机多实例的读写分离: 可以通过mysqld_multi来进行多实例的管理,mysqld_multi是用perl写的脚本,原理是通过mysql_ad ...

  5. Struts2之数据标签(二)

    Struts2之数据标签(一):http://blog.csdn.net/u012561176/article/details/46848817 1.action标签:使用此标签能够同意在JSP页面中 ...

  6. Facade 设计模式

    目的 在一个子系统的一组接口上提供一个统一的接口.Facade 设计模式定义了一个更高级别的接口,使子系统更容易使用. 通过一个更加简洁的接口来包装一个复杂的子系统. 解决的问题 客户端需要一个简化的 ...

  7. TCP网络编程杂谈

    作为一名IT工程师,网络通信编程相信都会接触到,比如Web开发的HTTP库,Java中的Netty,或者C/C++中的Libevent,Libev等第三方通信库,甚至是直接使用Socket API,但 ...

  8. systemd的程序自启动脚本编写

    以FreeSWITCH的自启动脚本为例. 一. 编写freeswitch.service文件 [Unit] Description=FreeSWITCH After=syslog.target net ...

  9. C#中委托、事件和回调函数的理解

    在C#中我们经常会碰到事件,尤其是在WPF或者WinForm中,窗体加载.或者点击一个按钮,都会触发事件.实际上,事件是对委托的封装.如果不进行封装,让委托暴露给调用者,调用者就可以把委托变量重新引用 ...

  10. 什么是Web缓存控制(基于HTTP头域)

    这是一篇转载的知识性的文档,主要目的是为了让Web缓存相关概念更容易被开发者理解并应用于实际的应用环境中.为了简要起见,某些实现方面的细节被简化或省略了.如果你更关心细节实现则完全不必耐心看完本文,后 ...