我可以使用 std::unique_ptr<BaseClass>(&DerivedClass)吗?

Can I use std::unique_ptr<BaseClass>(&DerivedClass)?

本文关键字:gt BaseClass DerivedClass lt std unique 我可以 ptr 可以使      更新时间:2023-10-16

我有一个段故障,但不知道为什么。使用std::unique_ptr<BaseClass>(&DerivedClassObj)有问题吗?谢谢。

下面是代码

# test.cc
#include <iostream>
#include <memory>
#include <vector>
using namespace std;
struct a {
  virtual void x(){cerr<<"a.x"<<endl;}
  virtual void y(){cerr<<"a.y"<<endl;}
  virtual void z(){cerr<<"a.z"<<endl;x();y();}
};
struct b: public a {
  virtual void y() {cerr<<"b.y"<<endl;}
};

int main(){
  cerr<<0<<endl;
  {
    b bb;
    vector<unique_ptr<a> > pb;
    cerr<<1<<endl;
    bb.z();
    pb.push_back(unique_ptr<a>(&bb));
    pb[0]->z();
    cerr<<2<<endl;
  }
  cerr<<3<<endl;
  return 0;
}

输出

0
1
a.z
a.x
b.y
a.z
a.x
b.y
2
Segmentation fault (core dumped)
下面是编译命令
$ g++ -std=c++0x    test.cc   -o test

g++版本信息。

$ g++ -v
Using built-in specs.
COLLECT_GCC=g++
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/4.6/lto-wrapper
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.6.3-1ubuntu5' --with-bugurl=file:///usr/share/doc/gcc-4.6/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.6 --enable-shared --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.6 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --enable-plugin --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.6.3 (Ubuntu/Linaro 4.6.3-1ubuntu5) 

尝试将智能指针包装在指向具有自动存储持续时间("在堆栈上",对于newbs)的对象的指针周围是未定义的;你不能做这件事。

除此之外,std::unique_ptr获取其底层指针的所有权,并在完成后删除该指针;这是一个问题,因为你给了它一个指向"堆栈上"对象的指针,这个指针已经被自动删除了。因此,你将删除你的对象两次。

我建议你动态地分配你的对象,使用以下命令:

unique_ptr<a> ptr(new b);
ptr->z();
pb.push_back(std::move(ptr));

& help;然后不要再使用ptr

您还需要在a中使用虚析构函数。

你应该找到一本关于c++的好书,因为这些都是相当基本的概念,应该包含在你的阅读材料中。