0

假设我有一个带有私有构造函数的类,该类将用于表示单个对象。假设我有一些非静态成员,我想在不使用范围解析运算符的情况下访问它们。我注意到我可以通过创建一个指向类类型的指针来实现这一点。我想知道为什么即使默认构造函数是私有的,我也可以声明指向该类的指针?这是一个示例程序。


// Example program
#include <iostream>
#include <string>


class OnlyOne{
    public:
    void Location(){
        std::cout<<10<<std::endl;
        
    }
    private:
    OnlyOne();
};
int main()
{
    //does not work Location() is not a static member
    //OnlyOne::Location();
     
      // doesn't work because default constructor is private.
     //OnlyOne one;
    //one.Location();
    
    
    OnlyOne* two=nullptr;
    two->Location();
}

我一直在网上寻找是否可以找到答案,但无法得到我想要的东西。

4

2 回答 2

4

当您声明某种类型的指针时,该类型不必是可构造的。事实上,这种类型甚至不必是完整的。所以,这一行:

OnlyOne* two = nullptr;

很好。

请注意,这一行:

two->Location();

调用未定义的行为,因为没有 指向的对象two,因此没有可以调用Location成员函数的对象。

事实上,由于这种类型是不可构造的,two所以永远不能指向一个有效的对象。您必须提供一种构造这样一个对象的方法,或者通过提供一个公共构造函数,或者通过一个静态成员函数来构造并返回这样一个对象。

于 2020-07-19T16:16:02.347 回答
1

您可以使用单例 pttern 来实现具有私有构造函数的类:


// Example program
#include <iostream>
#include <string>


class OnlyOne{
    public:

    static OnlyOne* instance() {
        static OnlyOne obj;
        return &obj;
    }

    void Location(){
        std::cout<<10<<std::endl;
        
    }
    private:
    OnlyOne() { }
};
int main()
{
    //does not work Location() is not a static member
    //OnlyOne::Location();
     
      // doesn't work because default constructor is private.
     //OnlyOne one;
    //one.Location();
    
    
    OnlyOne* two=OnlyOne::instance();
    two->Location();
}
于 2020-07-19T16:16:24.927 回答