C++实用技巧分享,控制对象创建

简介: C++实用技巧分享,控制对象创建

Hello,大家好,今天给大家分享两个C++的小技巧。

在通常情况下,我们定义了一个普通的类,然后就可以创建该类的实例对象,但是在某些情况下,允许用户任意的创建对象并不是一个好的方法,这时我们就需要限制用户创建对象。

一般的,对象的创建分两种方式,一种是在栈上创建对象,另一种是在堆上创建对象,下面我们就从这两个方面来讨论如何限制用户操作。

1 只允许用户在堆上创建对象

所谓只允许用户在堆上创建对象,就是只允许用户使用new操作符来创建对象,换句话说要禁止用户在栈上创建对象。

也许你会想,根据类的构造函数和访问限制特点,可以将构造函数声明为私有,但是这样一来,new操作符也不能访问构造函数了。

class X
{
    //...
};
X gObject;//line1
void test()
{
    X lObject;//line2
    X* ptr= new X;//line3
}

针对以上代码,我们希望"line1"和"line2"编译错误,"line3"编译正确,那么我们就需要对类的特殊函数做一些处理。

上面已经说到了,构造函数是不能声明为私有的,否则也会阻止在堆上创建对象,既然这样,构造函数就只能声明为公有了,还有什么方法可以阻止在栈上创建呢。

我们再仔细想一想,在C++中,有这样一条规则:当用户试图在栈上创建对象时,编译器会查找匹配且可以访问的构造函数和析构函数,如果其中一个无法访问,编译就会报错。

这似乎给了我们一点提示,将析构函数声明为私有,这样的话不就可以禁止用户在栈上创建对象了吗。

class X
{
public:
    X(){}
    void Delete()
    {
        delete this;
    }
private:
    ~X(){}
};
X gObject;//line1
void test()
{
    X lObject;//line2
    X* ptr= new X;//line3
}

我们编译一下,发现符合预期。

但是,马上下一个问题来了,如果你使用delete来删除创建出来的对象时,编译器还是报错了。

为此,我们可以再编写一个释放函数,在函数中调用delete操作符,这样所有的问题就都解决了。

下面是示例代码:

class X
{
public:
    X(){}
    void Delete()
    {
        delete this;
    }
private:
    ~X(){}
};
X gObject;//line1
void test()
{
    X lObject;//line2
    X* ptr= new X;//line3
    //...
    ptr->Delete();
}

好了,大功告成,但是,有一点需要注意,我们在成员函数Delete()中,删掉了this指针所指对象,这有点像对象的自杀行为,如果该对象再被引用将会报错,所以请确定不再使用该对象时,再调用删除函数,这个地方一定要注意。

那么我们可以总结:使析构函数私有化,可以禁止在栈上创建对象。

2 只允许在栈上创建对象

这跟我们上一小节刚好相反,我们需要禁止new操作符创建对象,聪明的同学可能已经想到了,那就是将new和delete操作符声明为私有的即可。

class Y
{
public:
    Y(){}
private:
    void* operator new(size_t size);
    void operator delete(void* addr);
};
void test()
{
    Y obj;//line1
    Y* ptr= new Y;//line2
}

编译一下,正如我们所预想的,"line2"将不能通过编译,这就阻止了用户在堆上创建对象,是不是很神奇。

好了,关于今天的分享就到这里了,尽管这样的技巧在平时用的不多,但是在某些场景下的确是很好的处理办法。

例如,如果你正在编写一个同步锁或文件对象等,为了让锁自动打开和释放或者是让文件自动打开和关闭,你就使该类实例只能在栈上创建,在构造和析构函数中编写初始化与释放代码,这样用户创建的类对象就会自动管理资源,从而你也不必担心用户因为忘记释放资源而造成的种种坑了。

再比如有些工厂函数或类方法,不要求用户自行new对象,所以可以加限制,这些技巧并不多用,但在某些场景下还是很好用的,可以让用户更安全的使用类。等遇到实际问题了,可以往这方面想想,有个思路,平时有个印象即可


最后不要忘记点赞哦,您的支持就是对我最大的鼓励^_^

欢迎关注微信公众号-小豆君Qt分享


相关文章
|
1月前
|
编译器 C++
C++之类与对象(完结撒花篇)(上)
C++之类与对象(完结撒花篇)(上)
35 0
|
8天前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
35 4
|
9天前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
32 4
|
1月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
27 4
|
1月前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
23 4
|
1月前
|
存储 编译器 C++
【C++类和对象(下)】——我与C++的不解之缘(五)
【C++类和对象(下)】——我与C++的不解之缘(五)
|
1月前
|
编译器 C++
【C++类和对象(中)】—— 我与C++的不解之缘(四)
【C++类和对象(中)】—— 我与C++的不解之缘(四)
|
1月前
|
C++
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
53 1
|
1月前
|
编译器 C语言 C++
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
19 1
|
1月前
|
存储 编译器 C语言
【C++打怪之路Lv3】-- 类和对象(上)
【C++打怪之路Lv3】-- 类和对象(上)
16 0