компилятор не скомпилирует умный указатель, используя -std = c ++ 11 - PullRequest
0 голосов
/ 13 октября 2018

Я пытаюсь скомпилировать простое объявление общего указателя, но использую g++ -std=c++11 main.cpp -o main с использованием cmd, но по некоторым причинам он выдает кучу ошибок.Я пытался найти похожие вопросы о переполнении стека, но ни один из них не соответствует моей проблеме.

код:

std::shared_ptr<int[]>array(new int[100]);

Заголовочные файлы:

#include<iostream>
#include<memory>

версия компилятора: g ++ (MinGW.org GCC-6.3.0-1) 6.3.0

Ошибка:

In file included from c:\mingw\lib\gcc\mingw32\6.3.0\include\c++\bits\shared_ptr.h:52:0,
                 from c:\mingw\lib\gcc\mingw32\6.3.0\include\c++\memory:82,
                 from main.cpp:2:
c:\mingw\lib\gcc\mingw32\6.3.0\include\c++\bits\shared_ptr_base.h: In instantiation of 'std::__shared_ptr<_Tp, _Lp>::__shared_ptr(_Tp1*) [with _Tp1 = int; _Tp = int []; __gnu_cxx::_Lock_policy _Lp = (__gnu_cxx::_Lock_policy)2u]':
c:\mingw\lib\gcc\mingw32\6.3.0\include\c++\bits\shared_ptr.h:117:32:   required from 'std::shared_ptr<_Tp>::shared_ptr(_Tp1*) [with _Tp1 = int; _Tp = int []]'
main.cpp:7:42:   required from here
c:\mingw\lib\gcc\mingw32\6.3.0\include\c++\bits\shared_ptr_base.h:885:39: error: cannot convert 'int*' to 'int (*)[]' in initialization
         : _M_ptr(__p), _M_refcount(__p)
                                       ^`

Я относительно новичок и не знаю, что означает эта ошибка.Любые предложения будут полезны.

Ответы [ 2 ]

0 голосов
/ 13 октября 2018

Ошибка «невозможно преобразовать ...» - это ошибка типа, это означает, что вы использовали неправильный тип данных.В этом случае он специально говорит вам, что конструктор для std::shared_ptr<int[]> хочет тип аргумента int (*)[], но вы предоставляете аргумент типа int *.Поскольку компилятор не знает, как конвертировать из int * в int (*)[], вы получаете эту ошибку.

В любом случае, вам не следует использовать shared_ptr для управленияМассивы в стиле C .Я бы посоветовал переключиться на контейнеры STL, либо std::array<int, N>, либо std::vector<int>, в зависимости от того, знаете ли вы размер во время компиляции или нет.Любой тип контейнера STL может быть помещен в интеллектуальный указатель, если хотите.

0 голосов
/ 13 октября 2018

До C++17, std::shared_ptr мог не обрабатывать динамически распределяемые массивы.Обновите ваш компилятор для поддержки C++17, и ваш код с радостью скомпилируется.

РЕДАКТИРОВАТЬ: это обходной путь для более ранних версий.Вы можете использовать:

std::shared_ptr<int> sp(new int[10], custom_deleter<int>{});

, где custom_deleter будет функцией, которая будет использоваться для освобождения выделенной памяти.В этом случае будет достаточно простого delete[] (вместо неявного delete внутри деструктора shared_ptr):

template< typename T >
struct custom_deleter
{
    void operator ()(const T* arr)
    { 
        delete[] arr; 
    }
};

, но так как вы уже используете C++11, можно заменить структурус лямбда-выражением , которое упростит код:

std::shared_ptr<int> sp(new int[10], [](const int* arr){ delete[] arr; });
...