try { mpdf* mtmp = UI::build<mpdf> (_Sources, i); Sources (i) = mtmp; } catch (UIException &exc) {
This saves one call to copy constructor and prevents slicing.
Objects allocated by operator new should be assigned to a smart pointer instance immediately upon their construction, so that they can be automatically deleted after use. BDM implements its own reference-counted smart pointer template, bdm::shared_ptr, whose interface and semantics are close to the proposed standard std::tr1:shared_ptr (which is planned to replace bdm::shared_ptr once it becomes widely available). Note that objects allocated on the stack must not have their addresses passed to shared_ptr - that is a bug leading to intermittent runtime errors.
Non-null heap pointers may also be kept in instances of object_ptr, which is convertible to (and from) shared_ptr. The SHAREDPTR macro (or SHAREDPTR2, for templated types) defines standartized names for these instances, simplifying library usage:
/* egamma_ptr is typedef for object_ptr<egamma>, whose default constructor calls new egamma() */ egamma_ptr eG; eG->set_parameters ( a, b ); epdf_array Coms ( 2 ); // epdf_array is typedef for Array<shared_ptr<epdf> > Coms ( 0 ) = eG; // object_ptr<T> is derived from shared_ptr<T> /* The egamma instance doesn't leak: if the shared_ptr instance which wraps it isn't assigned to anything else, the pointer is deleted by the destructor of either object_ptr, or Array, whichever runs last. */
Pointers kept in object fields should be wrapped in a shared_ptr instance, which will automatically keep them valid (at least) for the lifetime of the containing object. When that isn't possible, it should be documented why their containing class doesn't delete them and who does.
Pointers passed as arguments into functions (and methods) are generally not expected to stay valid after the function returns - when that is required, the parameter's documentation should specify the required scope:
/* the pointer must stay valid for the lifetime of the object */ CurrentContext ( const char *name, int idx );
A simpler alternative is just to pass a shared pointer:
class mepdf : public mpdf { shared_ptr<epdf> iepdf; public: mepdf (shared_ptr<epdf> em) { iepdf = em;
In the case above, passing a (constant) reference to shared_ptr<epdf> might be more efficient, but no measurements have been performed.
Functions returning raw pointers should document the scope of their validity:
/* Returns the stored pointer (which remains owned by this instance). */ T *get();
Functions generally shouldn't return raw pointers allocated by operator new - such pointers should be wrapped in an instance of shared_ptr, so that the pointer's unlimited life expectancy is encoded in the function signature:
virtual shared_ptr<epdf> marginal (const RV &rv) const;