第七讲:构造函数与析构函数

2023-11-05

第七讲:构造函数与析构函数

本讲基本要求

    * 掌握:构造和析构函数概念、初始化、作用。
    * 理解:构造构函的重载; 带参数的构造函数两种表达格式。
重点、难点
    * 构造和析构函数概念、初始化、作用。

    通过前两章的学习,我们已经对类和对象有了初步的了解。在本章中将对类和对象进行进一步的讨论。在这一章中将会遇到一些稍为复杂的概念,请同学们多用心学,是C++的基础,也是面象对象编程的基础。

一、构造函数

1、对象的初始化

    对象的初始化和结构体变量的初始化是差不多的,在一个花括号内顺序列出各公用数据成员的值,两个值之间用逗号分隔。如:
      class Time
          { public: //声明为公用成员
            hour;
            minute;
            sec;
          };
      Time t1={14,56,30}; //将tl初始化为14:56:30

说明:
    1、如果数据成员是私有的,或者类中有private或protected的成员,就不能用这种方法初始化。
在建立一个对象时,常常需要作某些初始化的工作。
    2、类的数据成员是不能在声明类时初始化的。下面的写法是错误的:
        class Time
            { hour=0;     //不能在类定义中对数据成员初始化
              minute=0;
              sec=0;
            };
    因为类并不是一个实体,而是一种抽象类型,并不占存储空间,显然无处容纳数据。如果一个类中所有的成员都是公用的,则可以在定义对象时对数据成员进行初始化。
    3、用成员函数来对对象中的数据成员赋初值的(例如上章中3中的set_time函数)。从例3中可以看到,用户在主函数中调用set_time函数来为数据成员赋值。如果对一个类定义了多个对象,而且类中的数据成员比较多,那么,程序就显得非常臃肿烦琐,这样的程序哪里还有质量和效率?

2、构造函数的作用

    构造函数的作用:面象对象的编程提供了构造函数(constructor)来处理对象的初始化。构造函数是一种特殊的成员函数,与其他成员函数不同,不需要用户来调用它,而是在建立对象时自动执行。构造函数的名字必须与类名同名,而不能由用户任意命名,以便编译系统能识别它并把它作为构造函数处理。它不具有任何类型,不返回任何值。构造函数的功能是由用户定义的,用户根据初始化的要求设计函数体和函数参数。

无参数构造函数格式:
    
类名( )
    {  类数据成员的初始化定义  }

例1 在例2.3的基础上定义构造成员函数。
    
#include <iostream>
    using namespace std;
    class Time
       { public:     //私有数据成员
       
  Time()    //定义构造成员函数,函数名与类名相同
          { hour=0;
//利用构造函数对对象中的数据成员赋初值
            minute=0;
            sec=0;
          }
          void set_time();  
//函数声明
          void show_time();
//函数声明
         private:           //私有数据成员
          int hour;
          int minute;
          int sec;
         };

         void Time::set_time() //定义成员函数,向数据成员赋值
            { cin>>hour;
              cin>>minute;
              cin>>sec;
            }

         void Time::show_time() //定义成员函数,输出数据成员的值
            { cout<<hour<<":"<<minute<<":"<<sec<<endl; }

         int main()
          { Time t1;       //建立对象t1,同时调用构造函数t1.Time()
            t1.set_time(); //对tl的数据成员赋值
            t1.show_time(); //显示t1的数据成员的值
            Time t2;       //建立对象t2,同时调用构造函数t2.Time()
            t2.show_time(); //显示t2的数据成员的值
            return 0;
           }

程序运行的情况为:
    10 25 54/     (从键盘输入新值赋给t1的数据成员)
    l0:25:54     (输出t1的时、分、秒值)
    0:0:0        (输出t2的时、分、秒值)

构造函数的使用说明:
    (1)在类对象进入其作用域时调用构造函数,构造函数一般声明为 public。
    (2)构造函数没有返回值,因此也不需要在定义构造函数时声明类型,构造函数的作用主要是用来对对象进行初始化。
    (3)构造函数不需用户调用,也不能被用户调用。
    (4)在构造函数的函数体中不仅可以对数据成员赋初值,而且可以包含其他语句。但是一般不提倡在构造函数中加入与初始化无关的内容,以保持程序的清晰。
    (5)如果用户自己没有定义构造函数,则系统会自动生成一个构造函数,只是这个构造函数的函数体是空的,也没有参数,不执行初始化操作。

3、带参数的构造函数

   带参数的构造函数定义格式为:
      构造函数名(类型1形参1,类型2形参2,…)

   调用带参数的构造函数在定义对象一般格式为
      类名对象名(实参1,实参2,…);

例2 有两个长方柱,其长、宽、高分别为:(1)12,25,30;(2)15,30,2l。分别求它们的体积。编写一个程序,在类中用带参数的构造函数。
#include <iostream>
using namespace std;
class Box
    { public:
      Box(int,int,int); //声明带参数的构造函数
      int volume();     //声明计算体积的函数
      private:
      int height;
      int width;
      int length;
    };
//在类外定义带参数的构造函数
Box::Box(int h,int w,int len)
  { height=h;
    width=w;
    length=len;}

int Box::volume() //定义计算体积的函数
  { return(height*width*length); }

int main()
  { Box box1( 12,25,30 );
    cout<<"The volume of box1 is "<<box3.volume()<<endl;
    Box box2(15,30,20);
    cout<<"The volume of box2 is "<<box2.volume()<<endl;
    return 0; }

程序运行结果如下:
   The volume Of boxl is 9000
   The volume of box2 is 9450

可以知道:
   (1)带参数的构造函数中的形参,其对应的实参在定义对象时给定。
   (2)用这种方法可以方便地实现对不同的对象进行不同的初始化。

4、用参数初始化表对数据成员初始化

   参数初始化表来实现对数据成员的初始化。这种方法不在函数体内对数据成员初始化,而是在函数首部实现。

用参数初始化表对数据成员初始化定义格式:
   
类名(型参列表):数据成员(型参)列表{ }

例如:例2中定义构造函数可以改用以下形式:

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

 

5、构造函数的重载

   在一个类中可以定义多个构造函数,以便对类对象提供不同的初始化的方法,供用户选用。这些构造函数具有相同的名字,而参数的个数或参数的类型不相同。这称为构造函数的重载。


例3 在例2的基础上,定义两个构造函数,其中一个无参数,一个有参数。
#include <iostream>
using namespace std;
class Box
    { public:
        Box(); //声明一个无参的构造函数
        Box(int h,int w ,int len):height(h),width(w),length(len){}
        //声明一个有参的构造函数,用参数的初始化表对数据成员初始化
        int volume();
      private:
        int height;
        int width;
        int length;  };

Box::Box()   //定义一个无参的构造函数
  { height=10;
    width=10;
    length=10; }

int Box::volume()
  { return(height*width*length); }

int main()
  { Box box1;   // 建立对象box1不指定实参
    cout<<"The volume of box1 is "<<box1.volume()<<endl;
    Box box2(15,30,25); //建立对象box2,指定3个实参
    cout<<"The volume of box2 is "<<box2.volume()<<endl;
    return 0; }

说明:
    (1)在调用构造函数时不必给出实参的构造函数,称为默认构造函数(default constructor),无参的构造函数属于默认构造函数。
    (2)如果在建立对象时选用的是无参构造函数,应注意正确书写定义对象的语句。在程序中不应出现调用无参构造函数(如Box()),请记住:构造函数是不能被用户显式调用的。
    (3)尽管在一个类中可以包含多个构造函数,但是对于每一个对象来说,建立对象时只执行其中一个构造函数,并非每个构造函数都被执行。

6、使用默认参数的构造函数

    例4 将例3程序中的构造函数改用含默认值的参数,长、宽、高的默认值均为10。

#include <iostream>
using namespace std;
class Box
    { public:
        Box(int w=10,int h=10,int len=10); //在声明构造函数时指定默认参数
        int volume();
      private:
        int height;
        int width;
        int length;  };

    Box::Box(int w,int h,int len) //在定义函数叫可以不再指定参数的默认值
      { height=h;
        width=w;
        length=len; }

int Box::volume()
  { return(height*width*length); }

int main()
{ Box box1;         //没有给定实参
  cout<<"The volume of box1 is "<<box1.volume()<<endl;
  Box box2(15);     //只给定1个实参
  cout<<"The volume of box2 is "<<box2.volume()<<endl;
  Box box3(15,30);  //只给定2个实参
  cout<<"The volume of box3 is "<<box3.volume()<<endl;
  Box box4(15,30,20); //给定3个实参
  cout<<"The volume of box4 is "<<box4.volume()<<endl;
  return 0; }


程序运行结果为
    The volume Of boxl is 1000
    The volume Of box2 is 1500
    The volumc Of box3 is 40500
    The volume Of box4 is 9000

说明:
    (1)指定构造函数的默认参数?应该在声明构造函数时指定默认值,而不能只在定义构造函数时指定默认值。
    (2)程序第5行在声明构造函数时,形参名可以省略。
    (3)如果构造函数的全部参数都指定了默认值,则在定义对象时可以给一个或几个实参,也可以不给出实参。由于不需要实参也可以调用构造函数,因此全部参数都指定了默认值的构造函数也属于默认构造函数。前面曾提到过:一个类只能有一个默认构造函数,也就是说,可以不使用参数而调用的构造函数,一个类只能有一个。其道理是显然的,为了避免调用时的歧义性。如果同时定义了下面两个构造函数,是错误的。
    Box(); //声明一个无参的构造函数
    Box(int=10,int=10,int=10);//声明一个全部参数都指定了默认值的构造函数因为在建立对象时,如果写成
    Box boxl;
编译系统无法识别应该调用哪个构造函数,编译时报错, 应该避免出现了歧义性。
    (4)在一个类中定义了全部是默认参数的构造函数后,不能再定义重载构造函数。一般不应同时使用构造函数的重载和有默认参数的构造函数。

二、析构函数

    定义:析构函数(destructor)也是一个特殊的成员函数,它的作用与构造函数相反,它的名字是类名的前面加一个“-”符号。

    作用:析构函数是与构造函数作用相反的函数。当对象的生命期结束时,会自动执行析构函数。具体地说如果出现以下几种情况,程序就会执行析构函数:

    1、如果在一个函数中定义了一个对象(它是自动局部对象),当这个函数被调用结束时,对象应该释放,在对象释放前自动执行析构函数。
    2、static局部对象在函数调用结束时对象并不释放,因此也不调用析构函数,只在main函数结束或调用exit函数结束程序时,才调用static局部对象的析构函数。
    3、如果定义了一个全局对象,则在程序的流程离开其作用域时(如main函数结束或调用exit函数)时,调用该全局对象的析构函数。
    4、如果用new运算符动态地建立了一个对象,当用delete运算符释放该对象时,先调用该对象的析构函数。

例5 包含构造函数和析构函数的程序。
#include <iostream>
#include <string>
using namespace std;
class Student     //声明Student类
    { public:
        Student(int n,string nam,char s) //定义构造函数
          { num=n; name=nam; sex=s;
            cout<<"Constructor called."<<endl;//输出有关信息
           }
        ~Student()         //定义析构函数
          { cout<<"Destructor called."<<endl;} //输出有关信息
        void display()   //定义成员函数
          { cout<<"num:"<<num<<endl;
            cout<<"name:"<<name<<endl;
            cout<<"sex:"<<sex<<endl<<endl; }
      private:
        int num;
        string name;
        char sex;
     };
int main()
  { Student stud1(10010,"Wang_li",'f'); //建立对象studl
    stud1.display();         //输出学生l的数据
    Student stud2(10011,"Zhang_fun",'m'); //定义对象stud2
    stud2.display();        //输出学生2的数据
    return 0; }

程序运行结果如下:
    Constructor called. (执行studl的构造函数)
    num:lOOlO
          (执行studl的display函数)
    name:Wang_li
    sex:f

    Constructor called. (执行stud2的构造函数)
    num:10011
           (执行stud2的display函数)
    name:Zhang_fun
    sex:m
    Destructor called.
(执行stud2的析构函数)
    Destructor called.
(执行smdl的析构函数)

三、调用构造函数和析构函数的顺序

    在使用构造函数和析构函数时,需要特别注意对它们的调用时间和调用顺序。
    其对应的析构函数最先被调用。如图所示。可简记为:先构造的后析构,后构造的先析构。它相当于一个栈,先进后出。


下面归纳一下什么时候调用构造函数和析构函数
    (1)在全局范围中定义的对象(即在所有函数之外定义的对象),它的构造函数在文件中的所有函数(包括main函数)执行之前调用。但如果一个程序中有多个文件,而不同的文件中都定义了全局对象,则这些对象的构造函数的执行顺序是不确定的。当main函数执行完毕或调用exit函数时(此时程序终止),调用析构函数。
    (2)如果定义的是局部自动对象(例如在函数中定义对象),则在建立对象时调用其构造函数。如果函数被多次调用,则在每次建立对象时都要调用构造函数。在函数调用结束、对象释放时先调用析构函数。
    (3)如果在函数中定义静态(static)局部对象,则只在程序第一次调用此函数建立对象时调用构造函数一次,在调用结束时对象并不释放,因此也不调用析构函数,只在main函数结束或调用exit.函数结束程序时,才调用析构函数。

  

 

 

 

 

 

           转自:http://210.44.195.12/cgyy/text/HTML/text/07.htm

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

第七讲:构造函数与析构函数 的相关文章

  • 如何动态构造方法?

    我设计了一个类 它非常标准 具有一些方法属性 class foo def f1 self print f1 def f2 self print f2 def fn self print fn 现在我想创建一个包含一组 foo 实例的类 cl
  • 如何使构造函数只能由基类访问?

    如果我想要一个只能从子类访问的构造函数 我可以使用protected构造函数中的关键字 现在我想要相反的 我的子类应该有一个构造函数 该构造函数可以由其基类访问 但不能从任何其他类访问 这可能吗 这是我当前的代码 问题是子类有一个公共构造函
  • Sinatra 请求对象

    我可能在这里遗漏了一些非常明显的东西 但我似乎无法找到答案 或者自己解决它 在西纳特拉 他们有一个self get方法 捕获块 当调用块时 您可以使用request里面有变量 这怎么可能 Sinatra module Sinatra cla
  • 重写继承的构造函数字段时的差异?

    考虑这个简单的 Scala 类 class A val d Int Scala 之间是否存在差异 无论是行为还是生成的字节码 class B d Int extends A d and class B override val d Int
  • Haskell/GHC:使用相同模式匹配多个一元构造函数

    所以我正在尝试定义 TrieSet 数据类型 尽管我知道我不需要 http hackage haskell org package TrieMap module Temp where import Data Map data TrieSet
  • 为什么可以从 console.log 访问 JavaScript 私有方法

    我写了一个简单的代码 const secure new class privateProperty 4 privateMethod console log The property this privateProperty should n
  • java“void”和“非void”构造函数

    我用 java 编写了这个简单的类 只是为了测试它的一些功能 public class class1 public static Integer value 0 public class1 da public int da class1 v
  • .class 与 .java

    class 文件和 java 文件有什么区别 我正在尝试让我的小程序工作 但目前我只能在 Eclipse 中运行它 还不能嵌入 HTML 谢谢 编辑 那么如何使用 JVM 进行编译呢 class 文件是编译后的 java 文件 java 都
  • C# 中的类和模块有什么用

    有人可以解释一下类和模块之间的区别吗 你什么时候使用其中一种而不是另一种 我正在使用 C 更新 我的意思是相当于 VB 模块的 C 版本 这在很大程度上取决于您所指的 模块 Visual Basic 的模块 C 中没有真正等效的 VB Ne
  • 反对“initialize()”方法而不是构造函数的争论

    我目前负责查找代码库中的所有不良做法 并说服我的同事修复有问题的代码 在我的探索过程中 我注意到这里很多人都使用以下模式 class Foo public Foo Do nothing here bool initialize Do all
  • 在类中使用 std::chrono::high_resolution_clock 播种 std::mt19937 的正确方法是什么?

    首先 大家好 这是我在这里提出的第一个问题 所以我希望我没有搞砸 在写这篇文章之前我用谷歌搜索了很多 我对编码 C 很陌生 我正在自学 考虑到有人告诉我 只为任何随机引擎播种一次是一个很好的做法 我在这里可能是错的 什么是正确 最佳 更有效
  • 在继承的ctypes.Structure类的构造函数中调用from_buffer_copy

    我有以下代码 class MyStruct ctypes Structure fields id ctypes uint perm ctypes uint 定义类后 我可以直接从缓冲区复制数据到我的字段上 例如 ms MyStruct fr
  • 类、模块、它们的特征类和方法查找

    我们来开公开课吧Module并向其中添加一个方法 class Module def foo puts phew end end 我可以通过这样做来调用这个方法 Class foo 这是可以理解的 因为类Class is Class 其超类是
  • 将构造函数传递给 Array.map?

    我怎样才能做这样的事情 var a 1 2 3 4 a map Date constructor 此代码在 Google V8 上引发错误 SyntaxError Unexpected number 我也尝试过 a map Date con
  • 默认析构函数做了多少事情

    C 类中的默认析构函数是否会自动删除代码中未显式分配的成员 例如 class C public C int arr 100 int main void C myC new C delete myC return 0 删除 myC 会自动释放
  • 检查一个类是否是另一个类的子类

    我想在不创建实例的情况下检查一个类是否是另一个类的子类 我有一个类 它接收类名作为参数 作为验证过程的一部分 我想检查它是否属于特定的类系列 以防止安全问题等 有什么好的方法可以做到这一点吗 is subclass of http php
  • 类的成员复制

    在学习 复制成员 概念时 书中给出了如下说法 此外 如果非静态成员是引用 const 或没有复制赋值的用户定义类型 则无法生成默认赋值 我不太明白这个声明到底想传达什么 或者说这个说法指的是哪一种场景 谢谢 该语句与编译器自动为您编写的类
  • 为什么在标头内的类声明中声明变量时会出现错误?

    我正在尝试创建一个包含简单整数的类 当然 它使用头文件之类的 这是代码 class h class consolBuf private int buffersize1 10 Data member initializer is not al
  • 为什么使用 Create 方法而不是使用“new”?

    静态构造函数有什么优点以及什么时候适合使用 public class MyClass protected MyClass public static MyClass Create return new MyClass 然后通过创建该类的实例
  • 为什么基类必须有一个带有 0 个参数的构造函数?

    这不会编译 namespace Constructor0Args class Base public Base int x class Derived Base class Program static void Main string a

随机推荐

  • 超级无敌简单题(鸽子数)

    欢迎访问https blog csdn net lxt Lucia 宇宙第一小仙女 o 萌量爆表求带飞 o dalao们点个关注呗 我只是一条可爱哒分界线 一 问题 Description 通常来说 题面短的题目一般都比较难 所以我要把题面
  • 【线性回归——从简单构建到实现数据预测】

    深度学习与神经网络day03 线性回归 一 简单的线性回归 1 1 数据集的构建 1 2 模型构建 1 3 损失函数 1 4 模型优化 1 5 模型训练 1 6 模型评估 1 7 样本数量和正则化系数的影响 二 多项式回归 1 1 数据集的
  • Qt(day1)

    思维导图 Qt实现第一个web的基础页面 include mywnd h include ui mywnd h include
  • IP协议-NAT机制(理解网络结构的关键要点)

    前言 我们现在使用得最多的IP协议版本是IPv4 IPv4是4个字节 32位 也就是说我们的IP地址最多就只有2 32 42亿 个 在日常生活中 我们需要联网的设备都需要有IP地址才能进行通讯 很明显现在42亿个IP地址已经完全不足以满足我
  • 矩阵乘法的本质是什么

    作者 知乎用户 链接 https www zhihu com question 21351965 answer 31050145 本题目前下面的解释都是线性代数教材上的各种定义 但都太过复杂了 我尝试写一个浅显的解释 小明今天要做饭 消耗2
  • 解决servlet filter中if语句无效问题

    今天写代码学习filter时 遇到了一件很奇怪的事情 就是当在jsp中同时提交文件和一个text组件时 servlet中的if语句无法返回true 代码如下 这是jsp代码
  • html三个组成部分组成部分,JavaScript组成(三个组成部分)

    JavaScript主要包括三部分 1 ECMAScript JavaScript的核心 仅仅是一个描述 定义了脚本语言的所有属性 方法和对象 具体地 ECMAscript描述了以下内容 语法 类型 语句 关键字 保留字 运算符 对象 每个
  • .cu的cuda程序的kernel函数中调用std函数无法编译的问题解决; error: calling a constexpr __host__ function

    引言 本人初学cuda编程 在jetson nano 4G上想实现sliced wasserstein distance的各个slice并行加速计算 遇到一些问题和解决方案 记录在此 基础知识 在cuda编程中 cpu称作host gpu称
  • vue文件夹上传组件选哪个好?

    一 功能性需求与非功能性需求 要求操作便利 一次选择多个文件和文件夹进行上传 支持PC端全平台操作系统 Windows Linux Mac 支持文件和文件夹的批量下载 断点续传 刷新页面后继续传输 关闭浏览器后保留进度信息 支持文件夹批量上
  • MYSQL服务器在federated引擎下备份还原错误实例

    MYSQL服务器备份还原错误实例 大家好 我是服务器新人 在使用过程中经常出现一些问题 希望各位大神指点一下 感激 我们服务器用的是MYSQL5 0 维护一般用Navicat8 0 有两台主机 192 168 2 2 192 168 2 3
  • Open3D 点云均值滤波

    目录 一 算法原理 1 均值滤波 2 参考文献 二 代码实现 三 结果展示 本文由CSDN点云侠原创 原文链接 如果你不是在点云侠的博客中看到该文章 那么此处便是不要脸的爬虫 一 算法原理 1 均值滤波 对待处理的当前采样点 选择一个模板
  • 区块链能否助力重塑物流业?

    去年年底 就有部分技术人士看准了市场潜力巨大的快递市场 提出 用区块链技术整顿快递行业 的观点 引发热议 那么区块链到底能否拯救快递行业 基于区块链技术具有去中心化 集体维护 高度透明 去信任 匿名 不可篡改 可追溯等显著特点 可以在以下快
  • 【转】do{...}while(0)的意义和用法

    转自 http blogread cn it article 5907 linux内核和其他一些开源的代码中 经常会遇到这样的代码 do while 0 这样的代码一看就不是一个循环 do while表面上在这里一点意义都没有 那么为什么要
  • 2020软件测试工程师面试题汇总(内含答案)-看完BATJ面试官对你竖起大拇指!

    2020最新软件测试面试题汇总 内附参考答案 测试技术面试题 1 什么是兼容性测试 兼容性测试侧重哪些方面 参考答案 兼容测试主要是检查软件在不同的硬件平台 软件平台上是否可以正常的运行 即是通常说的软件的可移植性 兼容的类型 如果细分的话
  • Spring MVC 学习总结(四)——校验与文件上传

    Spring MVC不仅是在架构上改变了项目 使代码变得可复用 可维护与可扩展 其实在功能上也加强了不少 验证与文件上传是许多项目中不可缺少的一部分 在项目中验证非常重要 首先是安全性考虑 如防止注入攻击 XSS等 其次还可以确保数据的完整
  • 【Python基础】python必会的10个知识点

    来源 Towards Data Science 作者 Soner Yildirim 编译 VK Python在数据科学生态系统中占据主导地位 我认为 占据主导地位的两大原因是相对容易学习和数据科学库的丰富选择 Python是一种通用语言 因
  • android main system log,Log中'main', 'system', 'radio', 'events'以及android log分析

    1log文件分类简介 实时打印 的主要有 logcat main logcat radio logcat events tcpdump 还有高通平台的还会有 QXDM 日志 状态信息 的有 adb shell cat proc kmsg a
  • vlan中ACL inbound与outbound详解

    关键字 华为ACL配置 Cisco ACL配置 Vlan ACL配置 ACL一般有两种应用场景 应用到交换机物理端口和应用到Vlan 场景一 应用到交换机物理端口 网络拓扑 PC连接在交换机的Gig0 0 1端口 实现目的 禁止在PC上能够
  • 在python中使用nohup命令说明

    nohup功能 nohup 是 no hang up 的缩写 就是不挂断的意思 如果你正在运行一个进程 而且你觉得在退出帐户时该进程还不会结束 那么可以使用nohup命令 该命令可以在你退出帐户 关闭终端之后继续运行相应的进程 1 代码 n
  • 第七讲:构造函数与析构函数

    第七讲 构造函数与析构函数 本讲基本要求 掌握 构造和析构函数概念 初始化 作用 理解 构造构函的重载 带参数的构造函数两种表达格式 重点 难点 构造和析构函数概念 初始化 作用 通过前两章的学习 我们已经对类和对象有了初步的了解 在本章中