2

我刚从 C++ 开始,在理解类中私有成员变量的范围如何工作时遇到了一些问题。请看下面的代码

class Foo{
    private:
        std::vector<int> container;
    public:
        // other methods
};

int main(int argc, char* argv[])
{
    Foo* foo = new Foo;
    // other method calls to which foo is passed
    delete foo;
    return 0;
}

在上面的代码中,变量“容器”是一个私有成员变量。我正在调用“Foo”实例并将其传递给其他几个方法和类。以下是我的疑惑

  1. 变量“容器”的范围是什么?在我删除实例 foo 之前,该变量是否存在?
  2. 我需要将“容器”作为指向向量的指针吗?

谢谢您的帮助

4

2 回答 2

7
  1. 是的,只要包含它的对象存在,容器成员的生命周期delete就会持续,直到您调用指向它的指针(foo在您的情况下)。
  2. 不,没有理由这样做。使其成为指针需要您创建一个动态对象vector<int>,您需要管理其生命周期(包括在容器指针上调用 delete)。这里没有必要。假设您希望容器与 Foo 对象一样长,您可以直接包含它,而无需使用指针。

传递foo指针只会传递指针。它指向的对象不会被复制,如果需要,只会复制指向它的指针。如果您了解 Java,那么如果我告诉您传递指针与仅传递对 Java 中的对象的引用相同,那么它将对您有所帮助,例如:

Foo f = new Foo();
// just passes the reference (pointer in C++) to doIt. 
// the actual object is not copied
doIt(f);
于 2009-01-01T04:23:40.950 回答
0

我正在调用“Foo”实例

实际上,您正在创建类 Foo的实例。

具体来说,您正在通过new()从堆中分配一块内存。这块内存足够大,可以包含Foo::container和Foo所需的任何其他开销类。

(在这个例子中,没有。对于其他类,可能有额外的属性或虚拟指针表。)

自然地,new()调用(也许是默认的?)Foo::Foo()构造函数,而后者又通过std::vector构造函数初始化Foo::container

变量“容器”的范围是什么?

container是实例foo的属性 [component] 。只要实例foo存在,它就存在。

在范围方面,我们可以说Foo::container。但是如果没有类Foo的实例,您将无法访问Foo::constainer。(例如对象foo。) Foo::constainer没有类Foo的实例就不存在。

(有些类变量的工作方式有所不同,其中一个值在所有实例之间共享。但这里不是这种情况。)

此范围与您的公共/受保护/私人/朋友成员访问控制无关。

例如,在某些Foo::myPublicMethod()中,您可以参考Foo::container。尽管您也可以在这种情况下放弃显式范围,并将其称为container

请注意,作为私有,您不能在 Foo 类方法之外访问Foo::container 。

在我删除实例 foo 之前,该变量是否存在?

是的。

我需要将“容器”作为指向向量的指针吗?

不,你可以,但你当然不必这样做。

一般来说,我建议不要将类实例成员作为指针加上构造函数中的 new 和析构函数中的 delete。这是低效和麻烦的。(默认的复制构造函数可以复制指针值,然后析构函数可以删除相同的指针值两次。)


根据您的需要,您可以考虑:

int main(int argc, char* argv[])
{
  Foo foo;
  // other method calls to which foo is passed
  return 0;
}

返回 0后foo将超出范围;,并被自动删除。此外,foo将从堆栈而不是堆中分配。


您可能会发现The Annotated C++ Reference Manual的二手副本很有用。它很旧,但它具有很高的信噪比。

于 2009-01-01T04:59:56.197 回答