Указатель NULL с boost:: shared_ptr?
Что эквивалентно следующему:
std::vector<Foo*> vec;
vec.push_back(NULL);
при работе с boost::shared_ptr
? Это следующий код?
std::vector< boost::shared_ptr<Foo> > vec;
vec.push_back(boost::shared_ptr<Foo>());
Примечание. Я могу отбросить много таких объектов. Должен ли я объявлять глобальный статический объект nullPtr
где-нибудь? Таким образом, только один из них должен быть построен:
boost::shared_ptr<Foo> nullPtr;
Ответы
Ответ 1
Ваше предложение (вызов конструктора shared_ptr<T>
без аргумента) верен. (Вызов конструктора со значением 0 эквивалентен.) Я не думаю, что это будет медленнее, чем вызов vec.push_back()
с уже существующим shared_ptr<T>
, поскольку в обоих случаях требуется строительство (либо прямая конструкция, либо копия -конструкция).
Но если вы хотите "более симпатичный" синтаксис, вы можете попробовать следующий код:
class {
public:
template<typename T>
operator shared_ptr<T>() { return shared_ptr<T>(); }
} nullPtr;
Это объявляет один глобальный объект nullPtr
, который позволяет использовать следующий естественный синтаксис:
shared_ptr<int> pi(new int(42));
shared_ptr<SomeArbitraryType> psat(new SomeArbitraryType("foonly"));
...
pi = nullPtr;
psat = nullPtr;
Обратите внимание, что если вы используете это в нескольких единицах перевода (исходные файлы), вам нужно указать классу имя (например, _shared_null_ptr_type
), переместить определение объекта nullPtr
в отдельный .cpp файл, и добавьте объявления extern
в файл заголовка, где определен класс.
Ответ 2
Ну, это законно:
shared_ptr<Foo> foo; /* don't assign */
И в этом состоянии он ничего не указывает. Вы можете даже проверить это свойство:
if (foo) {
// it points to something
} else {
// no it doesn't
}
Так почему бы не сделать это:
std::vector < shared_ptr<Foo> > vec;
vec.push_back (shared_ptr<Foo>); // push an unassigned one
Ответ 3
В С++ 0x вы можете просто преобразовать из nullptr
в std::shared_ptr
:
std::vector< boost::shared_ptr<Foo> > vec;
vec.push_back(nullptr);
Ответ 4
Вы можете объявить глобальный nullPtr
для shared_ptr<Foo>
. Но если вы загрязняете глобальное пространство имен, что бы вы назвали глобальным nullPtr
для shared_ptr<Bar>
?
Обычно я объявляю нулевой ptr как статичный в классе указателя.
#include <boost\shared_ptr.hpp>
class Foo; // forward decl
typedef boost::shared_ptr<Foo> FooPtr;
class Foo
{
public:
static FooPtr Null;
}
...
// define static in cpp file
FooPtr Foo::Null;
...
// use Foo Null
vec.push_back(Foo::Null);
Таким образом, каждый класс имеет статический Null.
Ответ 5
Здесь что-то, что я думаю, немного проще и прекрасно работает
(помните, что typedef
- ваш друг):
#include <cstdlib>
#include <vector>
#include <iostream>
#include <boost/shared_ptr.hpp>
typedef boost::shared_ptr< std::vector<char> > CharVecHandle;
inline CharVecHandle newCharVec(std::vector<char>::size_type size) {
return CharVecHandle(new std::vector<char>(size));
}
inline CharVecHandle newCharVec(void) {
return CharVecHandle();
}
int main ( void )
{
CharVecHandle cvh = newCharVec();
if (cvh == NULL)
std::cout << "It NULL" << std::endl;
else
std::cout << "It not NULL" << std::endl;
std::vector< CharVecHandle > cvh_vec;
cvh_vec.push_back(newCharVec(64));
cvh_vec.push_back(newCharVec());
// or call the NULL constructor directly
cvh_vec.push_back(CharVecHandle());
return EXIT_SUCCESS;
}
Ответ 6
Да, объявите глобальный статический нулевой указатель.