智能指针和前置声明之间的小问题

对比Go等其他语言的工程,C++工程让人痛苦的一件事情就是当工程稍微庞大一点,编译时间就蹭蹭蹭往上爬。一般来说看过Effective C++这本书或者其他类似书籍的人都知道要解决编译时长的问题,就要解决好和头文件之间的依赖关系。所以在任何必要的时候要首先考虑使用前置声明而不是之间include头文件。也就是说,在定义类的时候成员变量如果是自定义类型,可以考虑将其声明为指针类型或者是配合智能指针。函数传参时也是一样,使用指针或者引用。

对于一个C工程来说,因为没有智能指针和引用的概念,所以都是直接使用指针配合前置声明。用起来得心应手。

但是C++工程里,有时候为了方便和省心,更多时候指针类型的成员变量会使用智能指针包一下。这个时候有可能会出现编译通不过的情况:

MSVC:

error C2338: can‘t delete an incomplete type
warning C4150: deletion of pointer to incomplete type ‘base‘;

Clang:

error : invalid application of ‘sizeof‘ to an incomplete type ‘base‘
                    static_assert(0 < sizeof (_Ty),
                                      ^~~~~~~~~~~~
note: in instantiation of member function ‘std::default_delete<base>::operator()‘ requested here
                            this->get_deleter()(get());
                            ^
    ./main.h(6,8) :  note: in instantiation of member function ‘std::unique_ptr<base, std::default_delete<base> >::~unique_ptr‘ requested here
    struct test
           ^
./main.h(5,8) :  note: forward declaration of ‘base‘
    struct base;

看到这里,还是要感谢下clang的输出,比较清楚地把问题的本质原因找出来了。但是等等,我哪里调用了智能指针的析构函数?

稍微有点警觉的情况下,你应该反应过来是默认的析构函数在做析构智能指针这事情。

我们先来做一个尝试,把默认的析构函数显示写出来,然后按习惯把析构函数的定义放到cpp文件里。这时你会发现,编译通过并且能正常运行。

问题来了,为什么显示声明析构函数并将其定义挪到cpp里,这个问题就解决了呢?

还是来一段标准里的话吧:

12.4/4
If a class has no user-declared destructor, a destructor is implicitly declared as defaulted(8.4). An implicitly declared destructor is an inline public member of its class.

所以这个隐式的inline析构函数在调用智能指针的析构函数析构管理的指针对象时,需要知道该对象的大小。而此时只能看到前置声明而无法看到定义也就无从知道大小,只能GG了。

1 #pragma once
2
3 struct base
4 {
5   int x;
6 };

base.h

 1 #pragma once
 2
 3 #include <memory>
 4
 5 struct base;
 6 struct test
 7 {
 8   std::unique_ptr<base> base_;
 9
10   void print_base() const;
11 };

test.h

 1 #include "main.h"
 2 #include "base.h"
 3
 4 #include <cstdio>
 5
 6 void
 7 test::print_base() const
 8 {
 9   std::printf("%d\n", base_->x);
10 }

test.cpp

1 #include "test.h"
2
3 int main()
4 {
5   test t;
6   t.print_base();
7
8   return 0;
9 }

main.cpp

时间: 2024-10-03 23:02:15

智能指针和前置声明之间的小问题的相关文章

(转)Delphi2009初体验 - 语言篇 - 智能指针(Smart Pointer)的实现

快速导航 一. 回顾历史二. 智能指针简介三. Delphi中的interface四. Delphi中智能指针的实现五. interface + 泛型 = 强类型的智能指针!六. 智能指针与集合七. 注意事项八. 总结 本随笔所有源代码打包下载 一.回顾历史 在c++中,对象可以创建在栈里,也可以创建在堆里.如: class CTestClass{public: CTestClass() { printf("Create"); } void DoPrint() {} ~CTestCla

C++:浅谈c++资源管理以及对[STL]智能指针auto_ptr源码分析,左值与右值

C++:浅谈c++资源管理以及对[STL]智能指针auto_ptr源码分析 by 小威威 1. 知识引入 在C++编程中,动态分配的内存在使用完毕之后一般都要delete(释放),否则就会造成内存泄漏,导致不必要的后果.虽然大多数初学者都会有这样的意识,但是有些却不以为意.我曾问我的同学关于动态内存的分配与释放,他的回答是:"只要保证new和delete成对出现就行了.如果在构造函数中new(动态分配内存),那么在析构函数中delete(释放)就可以避免内存泄漏了!" 事实果真如此么?

c/c++ 数组的智能指针 使用

数组的智能指针 使用 数组的智能指针的限制: 1,unique_ptr的数组智能指针,没有*和->操作,但支持下标操作[] 2,shared_ptr的数组智能指针,有*和->操作,但不支持下标操作[],只能通过get()去访问数组的元素. 3,shared_ptr的数组智能指针,必须要自定义deleter 小例子 #include <iostream> #include <memory> #include <vector> using namespace s

c/c++ 继承与多态 文本查询的小例子(非智能指针版本)

问题:在上一篇继承与多态 文本查询的小例子(智能指针版本)在Query类里使用的是智能指针,只把智能指针换成普通的指针,并不添加拷贝构造方法,会发生什么呢? 执行时,代码崩掉. 分析下面一行代码: Query qb = ~Query("Alice"); 1,首先调用Query(string)的构造函数,把Query的成员q指向了new WordQuery(s) Query::Query(const std::string& s) : q(new WordQuery(s)){ s

c/c++ vector,map,set,智能指针,综合运用的小例子

标准库,智能指针,综合运用的小例子 功能说明:查询单词在文件中出现的次数,如果在同一行出现多次,只算一次. 比如查询单词:你好 输出的结果: 你好 出现了:2次 (行号 2)xxxxxxx 你好 (行号 3)bbb ccc 你好 xxxxx 注意点:代码的46行,必须使用引用. //非常重要,必须用引用,要不然就会拷贝一个新的set给lines,不是map里的set auto &lines = wm[word];//lines是shared_ptr 代码: #include <iostrea

C++几种智能指针之间的比较

这些智能指针在设计的时候,一个关键的问题就是所有权的控制.如果把指针所指向的对象比作电视机的话,那么指针就是观众.第一个人需要看电视的时候需要打开它,没人看的时候就要保证把电视关掉. 对于std::auto_ptr,boost::shared_ptr和scoped_ptr,情况如下: 1. std::auto_ptr:auto_ptr这个种族很有趣,每次只让一个人看电视,大家可以换着看,由最后那个不想看的人关掉.当把一个auto_ptr赋值给另一个时,就好比换一个人看电视似的.总之,电视只能一个

Smart pointer 智能指针小总结

Smart pointer line 58之后smart pointer里的计数已经是0,所以会真正释放它引用的对象,调用被引用对象的析构函数.如果继续用指针访问,会出现如下图的内存访问异常.所以说如果选择了用智能指针,就不要再试图用其他方式再去访问对象了. 1 // sharedTest.cpp : Defines the entry point for the console application. 2 // 3 4 #include "stdafx.h" 5 #include

C++ 基础知识回顾(string基础、智能指针、迭代器、容器类)

[1] string基础 [1.1] string 的构造 1 #include <iostream> 2 #include <string> 3 4 int main() 5 { 6 using namespace std; 7 8 cout << "1 --- string(const char* s):将string对象初始化为s指向的C风格字符串" << endl; 9 string one("benxintuzi_1&

技术分享会之——智能指针

由于之前也只是了解智能指针,要我说估计只能说个它是干什么的,用不了几分钟. 昨天花了一天时间各种百度,算是对智能指针有了一点了解,这篇文章基本就是这次分享会的PPT的copy,没有底层的东西,多是概念. 我觉得理解智能指针需要了解它发展的三个过程:起因,经过,结果.这篇文章主要讲述的是起因,经过和结果等以后工作了,实际接触了再说吧. 起因: 1.为什么需要智能指针 我们先看两个例子 一:内存泄露 <pre name="code" class="cpp">