第六章:类与对象

  首先我个人认为类是非常难的一个知识点,并且是一个非常重

要的知识点。我认为自己在这章掌握并不是很成功,为此我会

更多时间去学习这方面知识,下面是我对这章知识的认识:

第一点类的定义:

类是对具有相同属性和行为的一组对象的抽象与统一描述。是用户自定义的数据类型

类的定义包括行为和属性两个部分。

属性以数据表示,行为通过函数实现

格式:类的最后一定加“;”

各成员函数的实现

  1. <span style="font-family:SimSun;">class  类名  
  2.   
  3. {  
  4.   
  5.    public:  
  6.   
  7.            公有数据成员和成员函数;  
  8.   
  9.    protected:  
  10.   
  11.            保护数据成员和成员函数;                  类成员可以是数据类型,也可以是函数;  
  12.   
  13.    private:  
  14.   
  15.            私有数据成员和成员函数;  
  16.   
  17. };</span>  

由于隐藏数据是opp主要目标之一,数据项通常放在私有部分,组成类接口的成员函数放在公有部分;

不必要在类声明中使用关键字privat,因为这是类对象的默认访问控制;

public与private之间唯一的区别是,结构的默认访问类型是public,而类为prite;

实现类成员函数:

  1. <span style="font-family:SimSun;">返回值类型  类名::成员函数名(参数表)  
  2. {  
  3.           函数体  
  4. } </span>  

访问类成员

  1. <span style="font-family:SimSun;">对象成员的访问包括:  
  2.        ●圆点访问形式:对象名.公有成员  
  3.        ●指针访问形式</span>  

对象指针变量名->公有成员

*注意事项

1.类的成员可以是其他人的对象,但不能以类自身的对象作为本类的成员,而类自身的指针和引用可以作为类的成员。

2.类定义必须以分号“;”结束。

3.类与结构体的区别:没有明确指定类成员的访问权限时,C++结构体的成员是公有的,而类的成员是私有的。

类的构造函数和析构函数

构造函数的作用是:

 为对象分配空间;对数据成员赋初值;请求其他资源

c++提供了一个特殊的成员函数--类构造函数,专门用于构造新对象,将值赋给他们的数据成员

构造函数没有返回值,实际上构造函数没有声明类型;

构造函数的参数标识的不是类成员,而是赋给类成员的值,因此,参数名不能与类成员相同;

析构函数是用于取消对象的成员函数

当一个对象作用域结束时,系统自动调用析构函数

析构函数的作用是进行对象消亡时的清理工作

没有用户定义析构函数时,系统提供缺省版本的析构函数
析构函数名为:~类名
析构函数没有参数,也没有返回类型

c++提供了两种使用构造函数来初始化对象的方式:

第一种方式:显示地调用析构函数  例如:stock food=stock(“world cabbage”,250,1.25);

第二种方式:隐式地调用析构函数   例如:stock garment(”furry mason“,50,2.5);

默认构造函数:

默认构造函数是在未提供显式初始值时,用来创建对象的构造函数。

格式:

  1. <span style="font-family:SimSun;">格式:  
  2. funname(参数列表):初始化列表  
  3. {  函数体,可以是空函数体  }   
  4. 初始化列表的形式:   
  5. 成员名1(形参名1),成员名2(形参名2),成员名n(形参名n)   
  6. </span>  

对象生存期结束时,需要做清理工作,比如:释放成员(指针)所占有的存储空间

析构函数可以完成上述工作。

析构函数自动调用(隐式调用)

析构函数没有返回值,不能有参数,也不能重载

  1. <span style="font-family:SimSun;">类名::~类名()  
  2. {  
  3.        函数语句  
  4. }</span>  
函数重载:函数名相同,但参数不相同(类型不同,或者个数不同)的一组函数。

参数个数相同      参数类型不同

定义成员函数时,使用作用域解析运算符(::)来标示函数所属的类;

类方法可以访问类的private组件;

例如:void stock::update(double price)

类方法的完整名称中包括类名。

其定义位于类声明中的函数都将自动成为内联函数;

内联函数:

内联函数作用:减少频繁调用小子程序的运行时间的开销;

内联函数机制:编译器在编译时,将内联函数的调用以相应代码代替;

内联函数声明:inline函数原型;

注:内联函数仅在函数原型做一次声明;适用于只有1-5行的小函数;不能含有复杂结构控制语句,不能递归调用。

this 指针

一般来说,所有类方法都将this指针设置为调用它的对象的地址;

当参数与成员变量名相同时,如this->x = x,不能写成x = x。

浅复制与深复制

当类的数据成员是简单的数据类型时,创建对象时的数据复制系统机制工作得很好。但是,如果数据成员资源是由指针指示的堆,系统复制对象数据时只进行指针(地址值)复制,而不会重新分配内存空间。这样,程序运行时会产生对象操作异常;另外,当对象作用域结束后,又会错误地重复释放堆。这种情况下,需要使用用户自定义的复制构造函数。

*浅复制:

在用一个对象初始化另一个对象时,只复制了数据成员,而没有复制资源,使两个对象同时指向了同一资源的复制方式称为浅复制;

即:对于复杂类型的数据成员只复制了存储地址而没有复制存储内容;

默认复制构造函数所进行的是简单数据复制,即浅复制 。

*深复制:

通过一个对象初始化另一个对象时,不仅复制了数据成员,也复制了资源的复制方式称为深复制;

自定义复制构造函数所进行的复制是浅复制。 

静态成员:

静态数据成员在定义或说明时前面加关键字static,如:

  1. class A  
  2. {  
  3.     int n;  
  4.     static int s;  
  5. };  

友元函数

如果在本类(类A)以外的其他地方定义了一个函数(函数B)
这个函数可以是不属于任何类的非成员函数,
也可以是其他类的成员函数,
在类体中用friend对其(函数B)进行声明,此函数就称为本类(类A)的友元函数。
友元函数(函数B)可以访问这个类(类A)中的私有成员
用友元函数计算两点间距离
  1. #include<iostream>  
  2. using namespace std ;  
  3. #include<math.h>  
  4. class Point  
  5. public:  
  6.       Point(double xi, double yi) { X = xi ; Y = yi ;}  
  7.       double GetX() { return X ; }  
  8.       double GetY() { return Y ; }  
  9.       friend double Distance ( Point & a, Point & b ) ;  
  10.   private:  double X, Y ;  
  11. } ;  
  12. double Distance(Point & a, Point & b )    
  13.   { double dx = a.X - b.X ;  
  14.      double dy = a.Y - b.Y ;  
  15.      return sqrt ( dx * dx + dy * dy ) ;  
  16.   }  
  17. int main()  
  18. { Point  p1( 3.0, 5.0 ) ,  p2( 4.0, 6.0 ) ;  
  19.   double  d = Distance ( p1, p2 ) ;  
  20.   cout << "This distance is " << d << endl ;  
  21. }   

类的包含

*类的包含是程序设计中一种软件重用技术。即定义一个新的类时,通过编译器把另一个类“抄”进来;
*当一个类中含有已经定义的类类型成员,带参数的构造函数对数据成员初始化,须使用初始化语法形式;

*构造函数 ( 形参表 ) : 对象成员1(形参表 ) , … , 对象成员n (形参表 ) ;

*出现成员对象时,该类的构造函数要包含对象成员的初始化。如果构造函数的成员初始化列表没有对成员对象初始化时,则使用成员对象的无参(缺省构造函数。建立一个类的对象时,要先执行成员对象自己的构造函数,再执行当前类的构造函数。

*用类包含计算两点间距离:

  1. #include<cmath>  
  2. #include<iostream>  
  3. using namespace std;  
  4. class  Point   
  5. public:  
  6.        Point( int xi=0, int yi=0 ) { x = xi; y = yi; }  
  7.        int GetX()  {  return x;  }  
  8.        int GetY()  {  return y;  }  
  9.   private:  int x;  int y;  
  10. };  
  11. class Distance   
  12. public:  
  13.       Distance( Point xp1, Point xp2 );  
  14.       double GetDis()  {  return dist;  }  
  15.   private:  
  16.      Point p1, p2;  
  17.      double dist;  
  18. };  

学到这里我感受到了学习编程语音的压力但我也感受到了其中的快乐。

下面是我做的一道习题studentdemopic

  1. #include<iostream>  
  2. #include<string>  
  3. #include<algorithm>  
  4. using namespace std;  
  5. class student  
  6. {  
  7.     string name;  
  8.     int no;  
  9.     int score[3];  
  10.     float average;  
  11.     int order;  
  12.     public:  
  13.     student(int id,string na,int x,int y,int z):name(na),no(id)  
  14.     {  
  15.     score[0]=x;  
  16.     score[1]=y;  
  17.     score[2]=z;  
  18.     order=1,average=(score[0]+score[1]+score[2])/3;  
  19.     }  
  20.     student()  
  21.     {  
  22.         score[0]=score[1]=score[2]=0;  
  23.         order=1,average=0;  
  24.     }  
  25.     int getNo(){return no;}  
  26.     float getAverage(){return average;}  
  27.     void setAverage(int avg){average=avg;}  
  28.     void setOrder(int x){order=x;}  
  29.     int getOrder(){return order;}  
  30.     string getName(){return name;}  
  31.     void setName(string name){this->name=name;}  
  32.     void display();  
  33. };  
  34. void student::display()  
  35. {  
  36.     cout<<name<<"\t"<<no<<"\t"<<score[0]<<"\t"<<score[1]<<"\t"<<score[2]<<"\t"<<average<<"\t"<<order<<endl;  
  37. }  
  38. bool cmp1(student stu1,student stu2)  
  39. {  
  40.     if(stu1.getAverage()-stu2.getAverage()>=1e-9)return 1;  
  41.     else return 0;  
  42. }  
  43. bool cmp2(student stu1,student stu2)  
  44. {  
  45.     return stu1.getNo()<stu2.getNo();  
  46. }  
  47. class student_list  
  48. {  
  49.     student list[60];  
  50.     int n;  
  51.     public:  
  52.     student_list():n(0){};  
  53.     void add();  
  54.     void deleteStu();  
  55.     void query();  
  56.     void change();  
  57.     void display(int flag);  
  58.     void menu();  
  59.     int search(int no);  
  60.     void sortList();  
  61. };  

对于这道习题我还有很多困惑,我会在以后一一完善。

学习总结

经过这一章节的学习我学会了类这一部分很多新知识,但让我在编程方面的疑惑多了很多我认为这是一门需要不断探索的课程,我相信在我们的努力下会收获很多乐趣,也会学到很多新知识。我认为老师说的很对这一章的重要性显而易见,我认为这需要我们不断的努力。我认为所学的东西都非常浅,我的路还很长,以后也要多加努力,相信自己会做得更好!

猜你喜欢

转载自blog.csdn.net/lg20171771/article/details/80076653