unique函数_std::unique_ptr使用incomplete type的报错分析和解决

在将原始指针升级到C++11的std::unique_ptr时,如果使用incomplete type,会导致编译错误。错误源于std::unique_ptr需要知道类型大小以进行删除操作。解决方法包括:改用std::shared_ptr,自定义删除器或将析构函数等特殊成员函数的实现移到源文件中。推荐方法是仅在头文件中声明析构函数,避免初始化,以保持代码简洁。
摘要由CSDN通过智能技术生成

Pimpl(Pointer to implementation)很多同学都不陌生,但是从原始指针升级到C++11的独占指针std::unique_ptr时,会遇到一个incomplete type的报错,本文来分析一下报错的原因以及分享几种解决方法~

问题现象

首先举一个传统C++中的Pimpl的例子

// widget.h

// 预先声明
class Impl;

class Widget
{
    
    Impl * pImpl;
};

很简单,没什么问题,但是使用的是原始指针,现在我们升级到std::unique_ptr

// widget.h

// 预先声明
class Impl;

class Widget
{
    
    std::unique_ptr<Impl> pImpl;
};

很简单的一次升级,而且也能通过编译,看似也没问题,但当你创建一个Widget的实例

// pimpl.cpp

#include "widget.h"

Widget w;

这时候,问题来了

$ g++ pimpl.cpp
In file included from /usr/include/c++/9/memory:80,
    from widget.h:1,
    from pimpl.cpp:1:
/usr/include/c++/9/bits/unique_ptr.h:
    In instantiation of ‘void std::default_delete<_Tp>::operator()(_Tp*) const [with _Tp = Impl]’:
/usr/include/c++/9/bits/unique_ptr.h:292:17:
    required from ‘std::unique_ptr<_Tp, _Dp>::~unique_ptr() [with _Tp = Impl; _Dp = std::default_delete<Impl>]’
widget.h:5:7:   required from here
/usr/include/c++/9/bits/unique_ptr.h:79:16: error: invalid application of ‘sizeof’ to incomplete type ‘Impl’
79 |  static_assert(sizeof(_Tp)>0,
   |                ^~~~~~~~~~~

原因分析

从报错我们可以看出,std::unique_ptr中需要静态检测类型的大小static_assert(sizeof(Impl)>0,但是我们的Impl是一个预先声明的类型,是incomplete type,也就没法计算,所以导致报错。

想要知道怎么解决,首先需要知道std::unique_ptr为啥需要计算这个,我们来看一下STL中相关的源码,从报错中得知是unique_ptr.h的292行,调用了79行,我们把前后相关源码都粘出来(来自g++ 9.3.0中的实现)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值