C++类中成员变量的初始化有两种方式:构造函数初始化列表和构造函数体内赋值。

一、内部数据类型(char,int……指针等)

  class Animal

  {

  public:

    Animal(int weight,int height): //A初始化列表

      m_weight(weight),

      m_height(height)

    {

    }

    Animal(int weight,int height) //B函数体内初始化

    {

      m_weight = weight;

      m_height = height;

    }

  private:

    int m_weight;

    int m_height;

  }

  对于这些内部类型来说,基本上是没有区别的,效率上也不存在多大差异。

  当然A和B方式不能共存的。

二、无默认构造函数的继承关系中

  class Animal

  {

  public:

    Animal(int weight,int height): //没有提供无参的构造函数

      m_weight(weight),

      m_height(height)

    {

    }

  private:

    int m_weight;
    int m_height;
  };
 
  class Dog: public Animal
  {
  public:
    Dog(int weight,int height,int type) //error 构造函数 父类Animal无合适构造函数
    {
    }
  private:
    int m_type;
  }
  上面的子类和父类编译会出错:
  因为子类Dog初始化之前要进行父类Animal的初始化,但是根据Dog的构造函数,没有给父类传递参数,使用了父类Animal的无参数构造函数。而父类Animal提供了有参数的构造函数,
  这样编译器就不会给父类Animal提供一个默认的无参数的构造函数了,所以编译时报错,说找不到合适的默认构造函数可用。要么提供一个无参数的构造函数,
  要么在子类的Dog的初始化列表中给父类Animal传递初始化参数,如下:
  class Dog: public Animal
  {
  public:
    Dog(int weight,int height,int type) : Animal(weight,height) //必须使用初始化列表增加对父类的初始化
    {
      this->m_type = type;
    }
  private:
    int m_type;
  }
三、类中const数据成员、引用数据成员,必须在初始化列表中初始化,不能使用赋值的方式初始化
  class Dog: public Animal
  {
  public:
    Dog(int weight,int height,int type) : Animal(weight,height),m_type (type),LEGS(4) //必须在初始化列表中初始化
    {
      this->m_type = type;//error
      //LEGS = 4; //error
    }
  private:
    int& m_type;
    const int LEGS;
  }
四、包含有自定义数据类型(类)对象的成员初始化 
  class Food
  {
  public:
    Food(int type = 10)
    {
      m_type = type;
    }
    Food(Food &other) //拷贝构造函数
    {
      m_type = other.m_type;
    }
    Food & operator =(Food &other) //重载赋值=函数
    {
      m_type = other.m_type;
      return *this;
    }
  private:
    int m_type;
  };
  (1)、构造函数赋值方式 初始化成员对象m_food
  class Dog: public Animal
  {
  public:
    Dog(Food &food)
      //:m_food(food)
    {
      m_food = food; //初始化 成员对象
    }
  private:
    Food m_food;
  };
  //使用
  Food fd;
  Dog dog(fd);
  结果:
  先执行了对象类型构造函数Food(int type = 10)
  然后再执行对象类型构造函数Food & operator =(Food &other)
  想象是为什么?
  (2)、构造函数初始化列表方式
  class Dog: public Animal
  {
  public:
    Dog(Food &food) : m_food(food) //初始化 成员对象
    {
      //m_food = food;
    }
  private:
    Food m_food;
  };
  //使用
  Food fd;
  Dog dog(fd);
  结果:执行Food(Food &other)拷贝构造函数完成初始化
 

  不同的初始化方式得到不同的结果:明显构造函数初始化列表的方式得到更高的效率。

C++构造函数初始化列表与构造函数中的赋值的区别的更多相关文章

  1. C++基础 (3) 第三天 构造函数 构造函数初始化列表 拷贝构造函数 析构函数 静态成员变量

    // 同类之间无私处 2构造函数 3析构函数 4构造函数的种类和析构函数的顺序 结论:析构函数的调用顺序,跟对象的构造顺序相反,谁先构造,谁最后一个被析构. 拷贝构造函数: 注意: 等号写在下面和写在 ...

  2. C++构造函数初始化列表与赋值

    C++中类的初始化操作一般有四个部分组成: 1.构造函数初始化列表 2.构造函数体内赋值 3.类外部初始化 4.类声明时直接赋值 对于内部数据类型(char,int,float...),构造函数初始化 ...

  3. C++中为什么构造函数初始化列表

    已经有个构造函数负责初始化,为什么还需要构造函数初始化表呢? 在以下三种情况下需要使用初始化成员列表:一,需要初始化的数据成员是对象的情况:二,需要初始化const修饰的类成员:三,需要初始化引用成员 ...

  4. c++中的构造函数初始化列表

    三种情况下,必须在构造函数初始化列表中初始化成员: 1.const成员 2.引用成员 3.没有默认构造函数的成员

  5. C++类构造函数初始化列表

    C++类构造函数初始化列表 构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式.例如: class CExample {public:     ...

  6. 10.C++-构造函数初始化列表、类const成员、对象构造顺序、析构函数

    首先回忆下,以前学的const 单独使用const修饰变量时,是定义的常量,比如:const int i=1; 使用volatile const修饰变量时,定义的是只读变量 使用const & ...

  7. C++类构造函数初始化列表(转)

    构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式.例如: { public:     int a;     float b;     //构 ...

  8. const成员或者引用成员必须使用构造函数初始化列表的方式

    #include<iostream.h> class A { const int a; int b; }; void main() { A obja; }编译出现如下错误:error C2 ...

  9. <四>构造函数初始化列表

    示例代码1 点击查看代码 class CDate{ public: CDate(int _year,int _month, int _day){ this->year=_year; this-& ...

随机推荐

  1. phpstudy中apache的默认根目录的配置

    默认配置文件是:vhosts.conf. 安装laravel后需要把根目录配置到public. 下面的配置需要在本地计算机的host文件配置域名,一个是“localhost”,一个是“www.goho ...

  2. 007_linux显示一个文件的某几行(中间几行)

    <1>从第3000行开始,显示1000行.即显示3000~3999行 cat -n filename | tail -n +3000 | head -n 1000 cat -n anaco ...

  3. windows系统实现mysql数据库数据库主从复制

    环境: master mysql服务器 192.168.8.201 slave mysql服务器 192.168.8.89 目标: 实现主从复制 1.将MySQL5.5安装文件分别拷贝到两台机器的c盘 ...

  4. CentOS 6.5自动化运维之基于cobbler服务的自动化安装操作系统详解

    一.Cobbler安装 前提:cobbler由epel源提供,故此需要事先配置指向epel的yum源方可进行类似下面的安装过程. # yum install -y epel-release # yum ...

  5. SPI、IIC、IIS、UART、CAN、SDIO、GPIO、USB总线协议

    SPI.IIC.IIS.UART.CAN.SDIO.GPIO总线协议 SPI(Serial Peripheral Interface:串行外设接口)SPI总线由三条信号线组成:串行时钟(SCLK).串 ...

  6. matplotlib画堆叠条形图

    import matplotlib.pyplot as plt%matplotlib inlineplt.style.use('ggplot') plt.style.use("ggplot& ...

  7. java多线程快速入门(七)

    什么是守护线程 守护线程是为用户线程服务的这么一个线程,主线程结束,守护线程也结束 package com.cppdy; class MyThread3 extends Thread{ @Overri ...

  8. MarkDown常用语法及word转MarkDown

    介绍 Markdown 的目标是实现「易读易写」. 可读性,无论如何,都是最重要的.一份使用 Markdown 格式撰写的文件应该可以直接以纯文本发布,并且看起来不会像是由许多标签或是格式指令所构成. ...

  9. hive遇到FAILED: Execution Error, return code 2 from org.apache.hadoop.hive.ql.exec.mr.MapRedTask错误

    hive遇到FAILED: Execution Error, return code 2 from org.apache.hadoop.hive.ql.exec.mr.MapRedTask错误 起因 ...

  10. 《剑指offer》-数组中出现次数超过一半的数字

    /* 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2.如果 ...