C++

C.hier: Class hierarchies (OOP)[

2021-09-08  本文已影响0人  greatseniorsde

C.hierclass: Designing classes in a hierarchy:

C.126: An abstract class typically doesn’t need a user-written constructor

class Shape {
public:
   // no user-written constructor needed in abstract base class
   virtual Point center() const = 0;    // pure virtual
   virtual void move(Point to) = 0;
   // ... more pure virtual functions...
   virtual ~Shape() {}                 // destructor
};

class Circle : public Shape {
public:
   Circle(Point p, int rad);           // constructor in derived class
   Point center() const override { return x; }
};

C.127: A class with a virtual function should have a virtual or protected destructor

Reason

A class with a virtual function is usually (and in general) used via a pointer to base. Usually, the last user has to call delete on a pointer to base, often via a smart pointer to base, so the destructor should be public and virtual. Less commonly, if deletion through a pointer to base is not intended to be supported, the destructor should be protected and non-virtual; see C.35.

Example, bad
struct B {
    virtual int f() = 0;
    // ... no user-written destructor, defaults to public non-virtual ...
};

// bad: derived from a class without a virtual destructor
struct D : B {
    string s {"default"};
    // ...
};

void use()
{
    unique_ptr<B> p = make_unique<D>();
    // ...
} // undefined behavior, might call B::~B only and leak the string

Note

There are people who don’t follow this rule because they plan to use a class only through a shared_ptr: std::shared_ptr<B> p = std::make_shared<D>(args); Here, the shared pointer will take care of deletion, so no leak will occur from an inappropriate delete of the base. People who do this consistently can get a false positive, but the rule is important – what if one was allocated using make_unique? It’s not safe unless the author of B ensures that it can never be misused, such as by making all constructors private and providing a factory function to enforce the allocation with make_shared.

Enforcement

这个是经常看到的很重要的一条,通常的case就是base class 的dtor要public and virtual,或者protected and non-virtual(少见)

C.128: Virtual functions should specify exactly one of virtual, override, or final

Reason

Readability. Detection of mistakes. Writing explicit virtual, override, or final is self-documenting and enables the compiler to catch mismatch of types and/or names between base and derived classes. However, writing more than one of these three is both redundant and a potential source of errors.

It’s simple and clear:

Example, bad
struct B {
    void f1(int);
    virtual void f2(int) const;
    virtual void f3(int);
    // ...
};

struct D : B {
    void f1(int);        // bad (hope for a warning): D::f1() hides B::f1()
    void f2(int) const;  // bad (but conventional and valid): no explicit override
    void f3(double);     // bad (hope for a warning): D::f3() hides B::f3()
    // ...
};

Example, good
struct Better : B {
    void f1(int) override;        // error (caught): Better::f1() hides B::f1()
    void f2(int) const override;
    void f3(double) override;     // error (caught): Better::f3() hides B::f3()
    // ...
};

Discussion

We want to eliminate two particular classes of errors:

Note: On a class defined as final, it doesn’t matter whether you put override or final on an individual virtual function.

Note: Use final on functions sparingly. It does not necessarily lead to optimization, and it precludes further overriding.

Enforcement

很简单明确,就是explicitly 写清楚virtual, override或者final. 既增加readability又能catch错误.

上一篇 下一篇

猜你喜欢

热点阅读