功夫阿拉克的博客

effective c plus plus

1.尽量以const,enum,inline替换#define

例如:
# define ASPECT_RATIO 1.653
const double AspectRatio=1.653
宏定义在编译前被预处理,编译器看不见

2.尽量使用const

在实际使用中,const对象大多用于passed by pointer-to-const或passed by 
reference-to-const的传递结果

const对象调用const方法,non-const对象调用non-const方法,这种效率太低,
需要non-const对象调用const方法(转型)

例:
class TextBlock{
    public:/*...*/
    const char & operator[](std::size_t position) const//一如既往
    {/*...*/ return text[position];}
    char & operator[](std::size_t position)//现在只调用const op[]
    {
        return const_cast<char &>(static_cast<const TextBlock &>(*this)[position]);
    }
}

3.确定对象被使用前已先被初始化

c++规定,对象的成员变量的初始化动作发生在进入构造函数本体之前,构造函数的一个
较佳写法是,使用所谓的member initialization list(成员初值列)替换赋值动作。
例:
ABEntry::ABEntry(const std::string &name,const std::string &address,
                 const std::list<PhoneNumber> & phones)
                 :theName(name),theAddress(address),thePhones(phones),numTimeConsulted(0)
{}//构造函数本体不必有任何动作,通常效率高

4.如果不想使用编译器提供的函数

为驳回编译器自动提供的机能,可将相应的成员函数声明为private并且不予实现。使用像Uncopyable
这样的base class 也是一种做法。
class Uncopyable{
    protected://允许derived对象构造和析构
        Uncopyable(){};
        ~Uncopyable(){};
    private:
        Uncopyable(const Uncopyable&);//阻止copy
        Uncopyable & operator=(const Uncopyable &));
};//只需继承这个类

5.为多态基类声明virtual析构函数

1.当derived class对象经由一个base class指针被删除,而该base class带着一个non-virtual析构函数,
其结果未有定义——实际上执行时通常发生的是对象的derived成分没被销毁。
2.如果class不含virtual函数,通常表示它并不意图被用做一个base class。如果class带有任何virtual
函数,它就应该拥有一个virtual析构函数。

6.析构函数绝对不要吐出异常

1.如果一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉任何异常,然后吞下它们(不传播)
或结束程序。
2.如果客户需要对一个操作函数运行期间抛出的异常做出反应,那么class应该提供一个普通函数
(而非在析构函数中)执行该操作。

7.绝不在构造和析构过程中调用virtual函数

由于base class构造函数的执行更早于derived class构造函数,当base class构造函数执行时,
derived class的成员变量尚未初始化。

8.令operator=返回一个reference to *this

class Widget{
    public:
        Widget & operator=(const Widget & rhs){
            /*...*/
            return *this;
        }
};

9.在operator=中处理“自我赋值”

class Widget{
    /*...*/
    public:
        void swap(Widget &rhs);//交换*this和rhs是数据
};
Widget &Widget::operator=(const Widget &rhs)
{
    Widget temp(rhs);//为rhs数据制作一份副本
    swap(temp);//将*this数据和上述副本的数据交换
    return *this;
}

10.让接口容易被正确使用,不易被误用

比较安全的写法是预先定义所有有效的Months
class Month{
    public:
        static Month Jan(){return Month(1);}
        /*...*/
        static Month Dec(){return Month(12);}
    private:
        explicit Month(int m);//阻止生成新的月份
        /*...*/
};

Date d(Month::Mar(),Day(30),Year(1995));

11.宁以pass-by-reference-to-const替换pass-by-value

减少复制,提高效率

12.必须返回对象时,别妄想返回其reference

错误1:
const Rational & operator*(const Rational &lhs,const Rational &rhs)
{
    Rational result(lhs.n*rhs.n,lhs.d*rhs.d);
    return result;//引用的是已被销毁的local对象
}
错误2:
const Rational & operator*(const Rational &lhs,const Rational &rhs)
{
    Rational *result=new Rational(lhs.n*rhs.n,lhs.d*rhs.d);
    return *result; 
}//问题:必须付出一个“构造函数调用”代价,同时如何对new出来的对象实施delect?
正确写法:
inline const Rational operator*(const Rational &lhs,const Rational &rhs)
{
    return Rational(lhs.n*rhs.n,lhs.d*rhs.d);
}
绝不要返回pointer或reference指向一个local stack对象,或返回reference 指向
一个heap-allocated对象,或返回pointer或reference指向一个local static对象
而有可能同时需要多个这样的对象。

13.将成员变量声明为private,protected并不比public更具有封装性

14.宁以non-member,non-friend替换member函数

15.使用非成员函数(non-member)处理函数的所有参数都需要类型转换

因为成员函数的发起者必须是类,若出现double类型在前的情况,则无法执行乘法,
所以应使用非成员函数。
例:
class Rational{
    public:
        Rational(double numerator=0,double denominator=1)
                :m_n(numerator),m_d(denominator){}
        double numerator() const{return m_n;}
        double denominator() const{return m_d;}
        double value(){return (m_n/m_d);}
    private:
        double m_n;
        double m_d;
};
const Rational operator *(const Rational &lhs,const Rational &rhs)
{
    return Rational(lhs.numerator()*rhs.numerator(),lhs.denominator()*rhs.denominator());
}

16.inline函数

inline函数背后的整体观念是,将“对此函数的每一个调用”都以函数本体替换之。

17.绝对不要重新定义继承而来的non-virtual函数

18.如果派生类的函数与基类函数同名,但是参数不同。

此时,无论其有无virtual关键字,基类的函数将被隐藏。

19.如果派生类的函数与基类函数同名,而且参数也相同。

基类没有关键字,此时基类的函数被隐藏。

20.c++纯虚函数

virtual void function()=0;

21.new 和 malloc的区别

(1)new是c++中的操作符,malloc是c中的一个函数
(2)new不止是分配内存,而且会调用类的构造函数,同理delect会调用类的析构函数,
而malloc则只分配内存,不会进行初始化类成员的工作,同样free也不会调用析构函数
(3)new出来的指针是直接带类型信息的,而malloc返回的都是void指针