前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >C++中与类有关的注意事项(更新中~~~)

C++中与类有关的注意事项(更新中~~~)

作者头像
_DIY
发布2019-09-11 17:22:56
6990
发布2019-09-11 17:22:56
举报
  • 关于构造函数的调用次序,见下列代码
代码语言:javascript
复制
#include<iostream>
using namespace std;
class A
{
    private:
        int x;
    public:
        A():x(0){ cout << "Construct A----" << x << endl; }
        A(int i):x(i){ cout << "Construct A----" << x << endl; }
        ~A() { cout << "Des A----" << x << endl; }
};
class B
{
    private:
        int y;
    public:
        B():y(0){ cout << "Construct B----" << y << endl; }
        B(int i):y(i){ cout << "Construct B----" << y << endl; }
        ~B() { cout << "Des B----" << y << endl; }
};
class C
{
    private:
        int z;
    public:
        C(int i):z(i){ cout << "Construct C----" << z << endl; }
        ~C() { cout << "Des C----" << z << endl; }
};
class D:public B
{
    public:
        A a0, a4;       // l1
        B b2, b1;       // l2
        C c1, c2;       // l3                //其构造函数调用次序与这里的顺序有关
        D():c2(2), c1(1), a4(4),B(1), b2(3) { cout << "Construct D----5" << endl; }         //B(i)调用的是D的基类构造函数,它首先开始
        ~D(){ cout << "Des D----5" << endl; }

};
int main()
{
    D d;
}

当然了,首先调用基类的构造函数是不容置疑的,不管它在哪里,记住即可,不过关于对象成员的构造函数的调用还需注意, 见 L1, L2, L3, 它们的构造函数的调用次序与它们在此的相对次序有关,如类A排在第一行,因此先调用关于它的对象,这里还应再注意一点,尽管先定义了它的对象成员,不过它不会立即调用其默认构造函数,而是去看看你有没有写相应的初始化(注意:这里是指在类里面,而不是指main函数内以及类外函数,对于类外函数应注意,在定义类的同时必须给它附上一定的值,不过这根据需要而定,如果你已经设置了无参构造函数了或者你在类内定义了一些set函数),比如调用完基类构造函数后优先调用a0的构造函数,但初始化列表中并没有它,故调用它的默认构造函数,然后调用a4的构造函数,依此类推,就不难理解编译运行后的结果了。

关于析构函数的调用只需知道它与构造函数的调用刚好对称即可。

针对继承,其构造函数的一般调用顺序为基类构造函数 ---> 成员对象的构造函数 ---> 它自身的构造函数(这里是指初始化列表后大括号内的内容)

  • 类的静态成员(static member)必须在类内声明,在类外初始化。
  • 类里面的任何成员变量在定义时是不能初始化的,尽管你可以编译过。
  • 类的一个对象调用了一次构造函数之后,是不允许再次调用构造函数的。
  • 如果一个类是另一个类的友元类,那么该类的友元函数将不能访问另一个类的私有成员。
代码语言:javascript
复制
class Base
{
    public:
        int x = 0;
    protected:
        double y = 0;
    private:
        float z = 0;
        friend class Deri;
};
class Deri:public Base
{
    protected:
        int dx = 1;
    public:
        friend void f2(Base b);
        void f3(Base b){cout << b.x << " " << b.y << " " << b.z << endl; }  //在友元类内部可以访问另一个类私有成员
};
void f2(Base b)
{
    cout << "======" << endl;
    cout << b.x << " " << b.y << " " << b.z<< endl; //error, y和z都无法访问
    
}
  • 类内的函数名不可和数据成员的名称重复。

代码语言:javascript
复制
class First1
{
    private:
        int memi; 
        double memd; 
    public:
        int memi(){return memi;} //error
        double getmemd(){return memd;}

};
  • 无论是在类外还是在类内访问私有成员时最好设置一个接口,养成一个习惯。
代码语言:javascript
复制
#include<iostream>
using namespace std;
class First1
{
    private:
        double memd; 
    public:
        double getmemd(){return memd;}

};
int main()
{
    First1 a;
    cout << a.getmemd() << endl;
}
  • 当该类的对象离开了它的域或者delete表达式应用到一个该类的对象的指针上时,析构函数会自动被调用。
  • 看一段代码:
代码语言:javascript
复制
#include<iostream>
using namespace std;
class Data
{
    private:
        int d;
    public:
        Data():d(0){}
        Data(const int dd):d(dd){}
        Data operator+(Data b)
        {
            Data c = d + b.d;
            return c;
            /*
                int c = d + b.d;
                return Data(c);         //true,和下面一样
                Data c_(c);
                return c_;      //true
            
            
            */
        }
        friend ostream &operator<<(ostream& os, const Data &b);


};
ostream& operator<<(ostream& os, const Data &b)
{
    os << b.d;
    return os;
}

int main()
{
    Data a, b(1);
    cout << a + b << endl;
    return 0;
}
  • 在使用类模板对象时,必须显示地指定模板实参,否则就会报错。
  • 在写类模板时,如何需要用到另一个类模板的私有成员,注意千万不要写成普通的友元形式,那样报错报到你哭,步骤自身感觉比较繁琐,见链接https://blog.csdn.net/lezardfu/article/details/61433246,实在不行在另一个类中写相应的public接口。

关于运算符重载需要注意以下几点:

  • 作为类成员的重载函数(以类名作为返回类型),其形参看起来比操作数少一个,因为隐式传递,限定为第一个操作数,举个例子,如下:
代码语言:javascript
复制
T operator + (const Data<T> d) {
        return value + d.value;
    }                /*这是一个类模板内的一个成员函数,注意人家
                        在使用类对象时显示的指定模板实参了,不要忘了,另外系统隐藏了一个
                        类对象,一般两个对象中隐藏第一个*/
  • 重载>> 和 << 时一般在public处声明(声明时不要忘记它是友元函数),在类外定义,注意它是非成员函数(这其中包括普通函数,友元函数)。但也有例外,比如你写了个类模板,并且你给它写了个运算符重载<<,这时你就不能按照常规做了,要既在类内声明,又在类内定义,同时不要忘记显示指定模板实参。注意:重载<<时写成 “friend ostream& operator<<(ostream &os, const Data<T> &s)”形式,避免出现发现不出来的错。
  • 一般将算术操作符定义为非成员函数,如+, - , *, /,不过也可以将它定义成成员函数
代码语言:javascript
复制
friend Complex operator+(Complex a, double b) {return Complex(a.r + b, a.i);   /*这里定义成友元函数比较好*/

/*注意:写成这样就不对了*/
//例如
Complex(Complex a, Complex b){}     /*如果你想得到a和b相加后的结果,这样写是不对的,因为多了一个*/
/*应写成这样*/
Complex(Complex b)
{
    Complex c;
    ... 
    return c(...);
}
  • 一元运算符(如++,--)因为其位置不同而导致重载形式不同,如
代码语言:javascript
复制
friend X& operator++(X& o);         /*相当于++a,自增完后直接返回引用*/
/*或者这样*/ X& operator++();        /*相当于++a,自增完后直接返回引用*/
friend X operator++(X& o, int );    /*相当于a++,自增完后返回一个临时的*/
  • 赋值必须返回对*this的引用(如+=, =)
  • 下标运算符 [ ] 一般作为类成员函数,中间加上const就更好了。
  • 重载类型转化操作符时应注意1.必须定义为类成员函数. 2.不能指定返回类型。 3.必须返回要转换成的类型
代码语言:javascript
复制
class Circle
{
    private:
        double x, y, r;
    public:
        Circle(double a, double b, double c):x(a), y(b), r(c){}
        operator int(){return int(r);}

};
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2019-05-12 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 关于运算符重载需要注意以下几点:
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档