c++ - 自定义类/对象的构造函数和初始化

标签 c++ constructor initialization

我可以想象这个问题已经被问过了,但我实际上找不到任何合适的解决方案,所以如果这是一个多余的问题,请原谅。

我有一个自定义类

class myClass_A
{
public:
    myClass_A();          // Constructor
    myFunction_A();       // Some function from Class A
};

现在我有另一个自定义类,它有一个类型为 myClass_A 的成员

class myClass_B
{
public:
    myFunction_B();       // Some function from Class B

private:
    myClass_A m_instance; // Instance of Class A
}

现在 myFunction_B() 想从 m_instance 调用方法 myFunction_A() 有点像这样:

myClass_B::myFunction_B()
{
    m_instance.myFunction_A();
}

现在,如果我编译我的代码(这基本上就像我上面发布的示例),它将成功而不会出现任何警告或错误。所以我的问题是:

A.这个例子中会调用构造函数吗?

B.我真的可以从未初始化的对象调用方法吗?

C.假设未调用构造函数,但我仍然可以从该对象调用方法 -> 这是否仍然意味着我的类的成员未初始化?

如果这些问题有点愚蠢,我很抱歉,但我觉得我现在的理解速度很慢。

最佳答案

这些都是非常好的和重要的问题。

关于 A:

在执行构造函数的主体之前,C++ 会生成自动调用类的所有聚合(即成员)对象的默认构造函数 的代码。基本上,它所做的是转换以下代码:

class myClass_B {
public:
    myClass_B()
    {
        m_instance.foo();
        m_pInstance->foo();
    }
private:
    myClass_A m_instance;
    myClass_A* m_pInstance;
};

进入以下代码:

class myClass_B {
public:
    myClass_B()
        : m_instance()
        , m_pInstance()
    {
        m_instance.foo();
        m_pInstance->foo();
    }
private:
    myClass_A m_instance;
    myClass_A* m_pInstance;
};

编译器自动插入的两行称为initializer list ,它在执行构造函数的主体之前调用每个聚合对象的默认构造函数。请注意,第二个 m_pInstance() 调用了“指针的默认构造函数”,它创建了一个未初始化的指针;这几乎总是不是你想要的。请参阅下文了解如何解决该问题。

现在让我们假设 myClass_A 的构造函数具有签名 myClass_A(int someNumber),即它接受一个参数。然后,C++ 不能自动为 myClass_B 生成初始化列表,因为它不知道要将哪个数字传递给 myClass_A 的构造函数。它会向您抛出一个编译器错误,可能是提示缺少 myClass_A 的默认构造函数。您必须自己编写初始化列表,例如:

class myClass_B {
public:
    myClass_B()
        : m_instance(21)
        , m_pInstance(new myClass_A(21))
    {
        m_instance.foo();
        m_pInstance->foo();
    }
private:
    myClass_A m_instance;
    myClass_A* m_pInstance;
};

这是正确的代码,它调用 myClass_A 构造函数,参数 someNumber 的值为 21。这也显示了如何正确初始化指针:使其指向一些新分配的对象。

关于 B:

不像其他人说的那样,你可以! (试一试)

但它会导致意想不到的行为,这不是您想要的。 (包括只有当行星正确对齐时它才可能做你想做的事。)它很可能会崩溃但不保证会崩溃。这可能会让您度过漫长的调试之夜。如果您的编译器很聪明,它可能会识别出这一点并警告您,但它不会给您错误。

另请注意,对于具有默认构造函数的非指针聚合对象,默认构造函数 被调用,您将一切顺利。当您使用内置类型或指针时,问题就来了。这是使用未初始化的变量,是导致错误的最常见原因之一。如果您的代码做了一些非常奇怪的事情,请始终检查您是否初始化了所有变量。将任何成员变量的条目放入初始化列表中应该成为一种反射,即使它正在调用默认构造函数。把事情说清楚。

关于 C:

是的。详见 B。有趣的是,如果你调用的方法不使用“this”指针(这包括不使用任何属性变量和不调用任何使用属性变量的方法),你的方法保证工作。当您在未初始化的对象上调用方法时会发生什么,方法中的“this”对象(即所有属性变量也是如此)是随机内存。该方法的代码将执行但使用随机内存,这就是失败的原因。

我希望这能让事情变得更清楚一些。

关于c++ - 自定义类/对象的构造函数和初始化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9802341/

相关文章:

c++ - 简单便携的乒乓通讯

c++ - 检查类型的所有字段是否为 "primitives"

c++ - 静态非模板成员函数与静态模板成员函数

c++ - 为什么在运算符重载中允许返回构造函数?

ios - AppDelegate.swift 和 RESideMenu - 故障 InitContentViewController

c++ - C++ 的 Sublime Text 自动完成插件?

c++ - 使用声明指定的构造函数

java - java记录上的构造函数注释

java - 如何在初始化时定义 map 内容?

c - 如何使用 posix_memalign 在 c 中初始化结构