zoukankan      html  css  js  c++  java
  • c++类的学习笔记

    用结构体数据的安全性得不到保证。

    使用类对数据进行封装,只能通过函数修改类中的数据

    (1)类的定义

    class 类名
    {
        private:
        protected:
        public: 
    }; 

    private:定义私有成员,只能被类本身的成员函数和友元访问。派生类和其他类均不能访问。若数据没有指明类型,默认为私有。

    public:定义公有成员,可被程序中任何代码访问。

    protected:定义保护成员,能被本身成员函数,友元,派生类访问。

    (2)成员函数的定义

    a、在类中定义

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    
    class Student
    {
        private:
            string Name;
            float Phi,Math,Ave;
        public:
            void set(string name,float phi,float math)
            {
                Name=name;
                Phi=phi;
                Math=math;
            }
            void Average()
            {
                Ave=(Phi+Math)/2;
            }
            void Display()
            {
                cout<<Name<<" "<<Phi<<" "<<Math<<" "<<Ave;
            }
    };
    
    int main()
    {
        Student stud;
        stud.set("zhangsan",90,100);
        stud.Average();
        stud.Display(); 
        return 0;
    } 

    b、在类外定义

    (考虑到有些函数复杂,在类内写不好看)

    先在类内声明,然后在类外定义的形式是

    函数返回值类型 类名 ::函数名称()

    {}

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    
    class Date
    {
        private :
            int year,month,day;
        public:
            void setDate(int,int,int);
            bool IsLeap();
            void print()
            {
                cout<<year<<" "<<month<<" "<<day<<endl;
            }
    };
    
    void Date::setDate(int n,int y,int r)
    {
        year=n;month=y;day=r;
    } 
    
    bool Date::IsLeap()
    {
        if(year%400==0||year%4==0&&year%400) return true;
        return false;
    }
    
    int main()
    {
        Date dt;
        dt.setDate(2019,11,22);
        if(dt.IsLeap()) cout<<"yes
    ";
        else cout<<"no
    ";
        dt.print(); 
        return 0;
    } 

    (3)成员的访问

    a、对于数据成员的访问

    对象名.成员名   对象指针名->成员名  (*对象指针名).成员名

    b、对于成员函数的访问

    对象名.成员函数名(参数表)  对象指针名->成员函数名(参数表) (*对象指针名).成员函数名(参数表)

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Student
    {
        private :
            string name;
            float Math,Phi,Ave;
        public:
            void setDate(string nam,float math,float phi)
            {
                name=nam;
                Math=math;
                Phi=phi;
            }
            void Average()
            {
                Ave=(Math+Phi)/2;
            }
            void print()
            {
                cout<<name<<" "<<Math<<" "<<Phi<<" "<<Ave<<endl;
            }
    };
    
    int main()
    {
        Student stu,*stud;
        stud=&stu; 
        //一开始我没定义 stu,直接定义个指针用了,这样不行。 
        stud->setDate("zhangsan",100,90);
        stud->Average();
        stud->print();
        return 0;
    } 
    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Student
    {
        private:
            string Name;
            int Phi,Math;
        public:
            void setDate();
            void print()
            {
                cout<<Name<<" "<<Phi<<" "<<Math<<endl; 
            }
    };
    
    void Student::setDate()//在类作用域内,直接使用即可访问 
    {
        cin>>Name>>Phi>>Math; 
    }
    
    int main()
    {
        Student stud;
        stud.setDate();
        stud.print();
        return 0;
    } 

    (4)构造函数

    构造函数就是对类内的数据初始化,定义一个类的对象时自动调用,如果不写构造函数,就调用系统默认的构造函数。

    性质:与类名相同 无返回值 可以重载

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Cuboid //长方体类 
    {
        private:
            int length,width,height;
        public:
            Cuboid();//无参构造函数 
            Cuboid(int,int,int);//有参构造函数 
            int volume();//计算体积 
    };
    
    Cuboid::Cuboid()
    {
        length=width=height=1;
    }
    
    Cuboid::Cuboid(int l,int w,int h)
    {
        length=l;
        width=w;
        height=h;
    }
    
    int Cuboid::volume()
    {
        return length*width*height;
    }
    
    int main()
    {
        Cuboid cuboid1;
        cout<<"cuboid1的体积为:"<<cuboid1.volume()<<endl;
        Cuboid cuboid2(10,20,30);
        cout<<"cuboid2的体积为:"<<cuboid2.volume()<<endl; 
        return 0;
    } 

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Cuboid
    {
        private:
            int h,w,l;
        public:
    /*        Cuboid(int hh=10,int ww=20,int ll=30)另一种写法 
            {
                h=hh;w=ww;l=ll;
            }*/
            Cuboid(int hh=10,int ww=20,int ll=30);//声明时有默认参数 
            int volume()
            {
                return h*w*l; 
            }
    };
    
    Cuboid::Cuboid(int hh,int ww,int ll)//不用再写默认参数 
    {
        h=hh;
        w=ww;
        l=ll;
    } 
    
    int main()
    {
        Cuboid cuboid1;
        cout<<cuboid1.volume()<<endl;
        Cuboid cuboid2(1);
        cout<<cuboid2.volume()<<endl;
        Cuboid cuboid3(1,2);
        cout<<cuboid3.volume()<<endl; 
        return 0;
    }

     也可以这样初始化

    Cuboid(int h,int w,int len):height(h),width(w),length(len){}

    (5)拷贝构造函数

    拷贝构造函数的名称与类名一致,函数的形式参数是本类型的一个引用变量。

    Sample (Sample &S);

    对象1=对象2 //对象1和对象2必须是同一个类

    将一个对象的非静态数据成员的值一一赋值给另一对象的对应成员。

    Sample S2(S1);

    Sample S2=S1;两者等价

    a、

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Sample
    {
        private:
            int data;
        public:
            Sample(int dt)
            {
                data=dt;
            }
            void print()
            {
                cout<<data<<endl;
            }
    };
    
    int main()
    {
        Sample s1(100);
        Sample s2=s1;
        s2.print();
    } 

     b、自定义拷贝构造函数 并不是一一赋值

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Sample 
    {
        private:
            int a,b,c,d;
        public:
            Sample(int aa,int bb,int cc,int dd)
            {
                a=aa;b=bb;c=cc;d=dd;
            }
            Sample(Sample &S)
            {
                a=S.a;b=S.b;c=S.c; d=S.d+1; 
            }
            void print()
            {
                cout<<a<<" "<<b<<" "<<c<<" "<<d<<endl;
            }
    };
     
    int main()
    {
        Sample dt1(1,2,3,4);
        Sample dt2=dt1;
        //Sample dt2(dt1);
        dt2.print();
        return 0;
    }
    

     (6)析构函数

    作用在对象脱离作用域时释放相关资源,清理善后工作。

    析构函数名和类名相同只是在函数名前面加~符号

    无参数 无返回值 只能有一个析构函数 不写虚构函数调用系统自己的

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Cstudent
    {
        private:
            int num;
            string name;
        public:
            Cstudent(string nam,int nnum)
            {
                name=nam;
                num=nnum;
                cout<<name<<" Constructor called.
    ";
            }
            void display()
            {
                cout<<"学号:"<<num<<endl;
                cout<<"姓名:"<<name<<endl; 
            }
            ~Cstudent()
            {
                cout<<name<<" Destructor called
    "; 
            }
    };
    
    int main()
    {
        Cstudent stud1("zhangsan",1);
        Cstudent stud2("lisi",2);
        return 0;
    }

    一般情况下,调用析构函数的次序与调用构造函数的次序相反。

     (7)静态成员--静态数据成员

    静态数据成员被类内所有的对象共享,某个对象对静态数据成员做出修改,则其他所有对象也共享被修改后的值。

    静态数据成员的存储空间不会随着对象的产生而分配,也不会随着对象的消失而释放。

    因此,静态数据成员不能在类体内初始化。

    如果未对静态数据成员赋初值,默认为0;其他成员不会默认为0;

    class Cstudent
    {
        private:
            string name;
            int score;
            static int studenttotal;
            static int sumscore;
        public:
            Cstudent(string nam,int scor)
            {
                name=nam;
                score=scor;
                studenttotal++;
                sumscore+=scor;
                cout<<name<<" Constructor called.
    ";        
            }
            static float average();
            ~Cstudent();
    };
    
    int Cstudent::studenttotal=0;
    
    int Cstudent::sumscore=0;

     (8)静态成员--静态成员函数

    一般情况下,静态成员函数只访问静态成员。也可以访问非静态成员,但要指明其所属对象,但这样麻烦,所以不用。

    class Cstudent
    {
        private:
            string name;
            int score;
            static int studenttotal;
            static int sumscore;
        public:
            Cstudent(string nam,int scor)
            {
                name=nam;
                score=scor;
                studenttotal++;
                sumscore+=scor;
                cout<<name<<" Constructor called.
    ";        
            }
            static float average();//******* 
            ~Cstudent();
    };
    
    int Cstudent::studenttotal=0;
    
    int Cstudent::sumscore=0;
    
    float Cstudent::average()  //********** 
    {
        return (sumscore*1./studenttotal);
    }

    静态成员的访问

    一般是:类名::静态数据成员名

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Cstudent
    {
        private:
            string name;
            int score;
            static int studenttotal;
            static int sumscore;
        public:
            Cstudent(string nam,int scor)
            {
                name=nam;
                score=scor;
                studenttotal++;
                sumscore+=scor;
                cout<<name<<" Constructor called.
    ";        
            }
            static float average();//******* 
            ~Cstudent();
    };
    
    int Cstudent::studenttotal=0;
    
    int Cstudent::sumscore=0;
    
    float Cstudent::average()  //********** 
    {
        return (sumscore*1./studenttotal);
    }
    
    
    
    Cstudent::~Cstudent()  
    {
        studenttotal--;
        sumscore-=score;
        cout<<name<<" Destructor called.
    "; 
    }
    
    int main()
    {
        Cstudent stud[2]={Cstudent("zhangsan",90),Cstudent("wangwu",100)};
        cout<<Cstudent::average()<<endl;
        return 0;
    }

    (9)对象的存储

    只有数据成员占有的内存空间的大小,而没有包括成员函数。

    //类的空间 
    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Data
    {
        private:
            int a,b;
        public:
            void setdata(int aa,int bb)
            {
                a=aa;
                b=bb;
            }
            void print()
            {
                cout<<a<<" "<<b<<endl;
            }
        
    };
    
    int main()
    {
        Data x,y;
        cout<<sizeof(int)<<endl;
        cout<<sizeof(x)<<endl;
        return 0;
    } 

    ()

     (10)this指针

    this指针指向正在操作该成员函数的对象。

    this指针最常用的两种情况:

    ①该函数的返回的是对调用该函数的对象的引用 即return *this

    ②当参数与成员变量名相同时,由于参数优先,所以对数据成员必须显式使用this指针修饰。

    //this指针 
    #include<iostream> 
    #include<cstdio>
    using namespace std;
    
    class Point
    {
        private:
            int x,y;
        public:
            Point(int xx,int yy)
            {
                x=xx;y=yy;
            }
            Point& setPoint(int x,int y)
            {
                this->x=x+8;
                (*this).y=y+8;
                return *this;
            }
            int getX(){return x;}
            int getY(){return y;}
    };
    
    int main()
    {
        Point p1(8,8);
        cout<<"执行setPoint()前p1:"<<p1.getX()<<","<<p1.getY()<<endl;
        //cout<<p1.setPoint(8,8);这行是错的 
        p1.setPoint(8,8); 
        //p1=p1.setPoint(8,8);和上一行等价 
        cout<<"执行setPoint()后p1:"<<p1.getX()<<","<<p1.getY()<<endl;  
        return 0;
    

    (11)常对象

    常对象是指对象的数据成员的值不能被改变,常对象必须在定义的同时进行初始化。

    且一旦定义就不能再改变。常对象只能调用const成员函数。普通对象可以调用const成员函数。

    //常对象 
    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Rectangle
    {
        private:
            int width;
            int length;
        public:
            Rectangle()
            {
                width=0;
                length=0;
            }
            Rectangle(int w,int len)
            {
                width=w;length=len;
            } 
            int area() const 
            {
                return width*length;
            }
    };
    
    int main()
    {
        Rectangle r1(2,2);
        const Rectangle r2(6,8);
        //Rectangle const r2(6,8);等价 
        cout<<"r1的面积:"<<r1.area()<<endl;
        cout<<"r2的面积:"<<r2.area()<<endl;
        return 0;
    }

    常对象中的某个数据成员想要修改,则声明为mutable

    //常对象 
    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Rectangle
    {
        private:
            int width;
            mutable int length;
        public:
            Rectangle()
            {
                width=0;
                length=0;
            }
            Rectangle(int w,int len)
            {
                width=w;length=len;
            } 
            int area() const 
            {
                length*=2;
                return width*length;
            }
    };
    
    int main()
    {
        Rectangle r1(2,2);
        const Rectangle r2(6,8);
        //Rectangle const r2(6,8);等价 
        cout<<"r1的面积:"<<r1.area()<<endl;
        cout<<"r2的面积:"<<r2.area()<<endl;
        return 0;
    }

    指向常对象的指针变量

    const 类名 *指针变量名

    const Rectangle *ptr=&r1;

    不能通过ptr修改r1,但ptr也可以指向别的。

    (12)常数据成员

    将数据成员声明为const型。不能被赋值,类体类外函数只可读。

    对常数据成员的初始化用构造函数的初始化列表。

    const int Length;
    Rectangle(int w,int len):Length(len)
    {Width=w;}

    (13)常成员函数

    这些函数只是可读函数,不改变类的数据成员,对函数加上const关键字标识,提高程序的可读性。

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Rectangle
    {
        private:
            int width,length;
        public:
            Rectangle(int w,int len)
            {
                width=w;
                length=len;
            }
            int area()const;
            void print()
            {
                cout<<"print"<<endl;
            }
    };
    
    int Rectangle::area()const
    {
        //width=10;不可修改数据成员 
        //print();常成员函数不能调用非const型成员函数 
        return width*length; 
    }
    
    int main()
    {
        Rectangle r(6,8);
        cout<<r.area()<<endl;
        return 0;
    } 

    (14)常指针

    将指针变量声明为const,这样指针始终保持其初值,不会改变。

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Rectangle
    {
        private:
            int width,length;
        public:
            Rectangle(int w,int len)
            {
                width=w;
                length=len;
            }
            int area()
            {
                return width*length;
            }
    };
    
    int main()
    {
        Rectangle r1(6,6);
        Rectangle *const ptr=&r1;
        cout<<ptr->area();
    //    Rectangle r2(3,3);不可以改变常指针的值 
    //    ptr=&r2;
        return 0; 
    } 

    (15)常引用

    一个变量的引用就是这个变量的别名。

    变量名和引用名都指向同一段内存单元。

    如果形参为变量的引用名,实参为变量名,则在调用时虚实结合,

    并不是为形参新开辟了空间,而是把实参变量地址传给形参。

    如果不希望在函数中修改实参的值,则可以把引用变量名声明为常引用。

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Rectangle
    {
        private:
            int width,length;
        public:
            Rectangle(int w,int len)
            {
                width=w;
                length=len;
            }
            void setWL(int w,int len)
            {
                width=w;
                length=len;
            }
            int area()
            {
                return width*length;
            }
    };
    
    void setRec(Rectangle &r1)
    {
        r1.setWL(2,2);     
    }
    
    int main()
    {
        Rectangle r(6,8);
        cout<<r.area()<<endl;
        setRec(r);
        cout<<r.area()<<endl;
        return 0;
    }

     若是常引用,则不可以修改。

     (16)友元

    c++用关键字 friend声明类的友元

    ①位于一个函数说明语句之前,指出该函数为这个类的友元函数。

    ②位于一个类名之前,指出该类是这个类的友元。

    如果要允许一个不属于类的函数取该类中的数据

    法一:数据成员声明为共有

    法二:将类内部声明这个函数为友元,可以访问该类数据。

    后者更好。友元其实破坏了类的封装。

    友元函数是普通(全局)函数

    //友元
    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Rectangle
    {
        private:
            int width,length;
        public:
            Rectangle(int w,int len)
            {
                width=w;
                length=len;
            }
            int area()
            {
                return width*length;
            }
            friend void display(Rectangle &r1);
    };
    
    void display(Rectangle &r1)//必须要加& 
    {
        cout<<r1.length<<" "<<r1.width<<endl;//不能直接使用length必须通过对象使用 
        cout<<r1.area()<<endl;
    }
    
    int main()
    {
        Rectangle r(6,8);
        display(r);
        return 0;
    } 

    友元函数是其他类的成员函数

    #include<iostream>
    #include<cstdio>
    using namespace std;
    class Rectangle;//提前引用说明
    class Cuboid
    {
        private:
            int Height;
        public:
            Cuboid(int h)
            {
                Height=h;
            }
            int Volume(Rectangle &r);//此处只能声明,不能定义。因为Rectangle类还未定义 
    };
    
    class Rectangle
    {
        private:
            int width,length;
        public:
            Rectangle(int w,int len)
            {
                width=w;length=len;
            }
            friend int Cuboid::Volume(Rectangle &r); 
    };
    
    int Cuboid::Volume(Rectangle &r)
    {
        return r.length*r.width*Height;  
    } 
    
    int main()
    {
        Rectangle R(6,8);
        Cuboid C(20);
        cout<<"长方体的体积为:"<<C.Volume(R)<<endl; 
        return 0;
    } 

    (17)友元类

    C++中允许将一个类声明为另一个类的友元,称为友元类。友元类中的所有成员

    函数都可以访问另一个类中的私有成员或保护乘员

    #include<iostream>
    #include<cstdio>
    using namespace std;
    
    class Rectangle;
    
    class Cuboid
    {
        private:
            int Height;
        public:
            Cuboid(int h)
            {
                Height=h;
            }
            int Volume(Rectangle &r);
    };
    
    class Rectangle
    {
        private:
            int width,length;
        public:
            Rectangle(int w,int len)
            {
                width=w;
                length=len;
            } 
            friend class Cuboid;
    };
    
    int Cuboid::Volume(Rectangle &r)
    {
        return r.length*r.width*Height; 
    } 
    
    int main()
    {
        Rectangle r(6,8);
        Cuboid C(20);
        cout<<"长方体的体积为:"<<C.Volume(r);  
        return 0;
    } 

    (18)类模板

    看不下去了。直接抄。

    class A
    {
        private:
            int x,y;
        public:
            A(int xx,int yy){x=xx;y=yy;}
            int sum(){return x+y;}
    };
    
    class A
    {
        private:
            double x,y;
        public:
            A(double xx,double yy){x=xx;y=yy;}
            double sum(){return x+y;} 
    };

    两段代码只是数据类型不同,但做的工作是重复的。

    类模板是一系列相关类的模型或样板,这些类的成员组成相同,成员函数的源代码形式相同。所不同的只是类型。

    (成员的类型以及成员函数的参数类型)。对于类模板,数据类型成了它的参数,因而是一种类型参数化的类。

    类模板是类的抽象,类是类模板的实例。

    template<class T>
    class A
    {
        private:
            T x,y;
        public:
            A(T xx,T yy){x=xx;y=yy;}
            T sum(){return x+y;}
    };

    template<模板参数表>

    class 类模板名

    {}

    
    

    class也可用typename关键字代替 

    类模板不能直接生成对象,因为其参数类型是不确定的,故需要首先对模板参数指定“实参”

    类模板名<具体类型>对象名{(构造函数实参列表)}

    A<int>IntA(6,8);

    A<double>DoubleA(6.6,8.8)

    若函数在类外定义

    T A<T>::sum()

    {return x+y;}

  • 相关阅读:
    redis-client和redis-template存储的key的格式不一样
    dubbo+zookeeper基础
    java面试题1
    Spring线程池(异步、同步)
    Java并发多线程
    Java并发-并发工具类JUC
    Java并发面试题
    ActiveMQ
    一键部署springboot到Docker
    Quartz任务调度学习
  • 原文地址:https://www.cnblogs.com/zzyh/p/11914508.html
Copyright © 2011-2022 走看看