================================================================
day06
================================================================
1.初始化和赋值是有区别的,前者效率更高。有时候必须使用初始化。
如果成员是const、引用,或者属于某种未提供默认构造函数的类类型,我们必须通过构造函数初始值列表为
这些成员提供初始值。
注意:数据成员会按其出现在类定义中的顺序得到初始化,而不是按其在初始值列表中的顺序。
2.如果一个构造函数为【所有参数都提供了默认实参】,则它实际上也定义了默认构造函数。(不是合成的默认构造函数)
举例:
class A {
public:
A(int a = 5, double b = 3.0)
{ v1 = a;
v2 = b;
cout << v1 << " "<< b <<endl;
}

private:
int v1 = 0;
double v2 = 0.0;

};

A m; // 虽然没有显式定义默认构造函数,但因为构造函数所有参数都提供了默认实参,所以正确。会输出: 5 3.0
A n(10,3.14); // 调用了构造函数A(int a = 5,double b = 3.14) 会输出:10 3.14

3.Sales_data obj(); // 这种用法的语法是正确的,但与我们的初衷不符:定义了一个函数【而非对象!】
如果想定义一个使用默认构造函数进行初始化对象,正确方法是去掉对象名之后的空的括号对。
Sales_data obj;
=================================================================================
4.隐式类类型转换。
如果构造函数只接收一个实参,那么它实际上定义了转换为此类类型的隐式转换机制。
eg:
class Sales_data{
public:
Sales_data() = default;
Sales_data(const string &s,unsigned n,double p):bookNo(s),units_sold(n),revenue(p*n){}
Sales_data(const string &s):bookNo(s){}
Sales_data(std::istream &);

Sales_data &combine(const Sales_data&);
private:
string bookNo;
unsigned units_sold = 0;
double revenue = 0.0;
};
这个类只接受一个实参的构造函数为接受string的构造函数和接受istream的构造函数。它们分别定义了从这两种类型
向Sales_data隐式转换的规则。也就是说,需要使用Sales_data的地方,我们可以用string或者istream代替。
string null_book = "9-999-9999-9";
item.combine(null_book); // 因为隐式转换的存在,我们用一个string实参调用了Sales_data的combine成员。
编译器用给定的string自动创建了一个Sales_data对象。然后这个临时对象被传递给combine。
注意!这个临时量是const的,而combine接受的实参也是const引用,所以可以传递。如果换成Sales_data &combine(Sales_data&)则会编译错误!
注意,编译器只会执行一步隐式类型转换。
eg: item.combine("9-999-9999-9"); //错误,不能把“9-999-9999-9”转换为string,再把string转换为Sales_data。编译器只会执行一步类型转换。
可以这样做:item.combine(string("9-999-9999-9"));//显式转换为string,再隐式转换成Sales_data
或: item.combine(Sales_data("9-999-9999-9")); // 隐式转换为string,再显式转换为Sales_data

如果我们不想这样的构造函数进行隐式转换,可以在声明前加explicit阻止。
explicit只对一个实参的构造函数有效,而且【只能在类内声明构造函数时使用,类外定义时不能重复!】

==============================================================================
5.聚合类。
如果一个类:
所有成员都是public、
没有定义任何构造函数、
没有类内初始值、
没有基类和virtual函数, 那么它就是一个聚合类。 聚合类有特殊的初始化语法形式:
struct Data{
int ival;
string s;
}
Data val1 = {0,"Anna"}; 初始值的顺序必须与声明顺序一致!如果初始值列表中的元素个数少于类的成员数量,则靠后的成员被值初始化。

6.有时候,我们需要类的一些成员只与类本身有关,而不是与类的各个对象有关联。
比如,一个银行类可能需要一个数据成员表示利率。从实现的效率来说,没必要每个对象都存储利率信息,
更重要的是,一旦利率浮动,我们希望所有对象都使用新值。
这时,我们可以把它声明为static。 可以用className::来访问静态成员。虽然静态成员不属于某个对象,但
仍然可以通过对象和【.操作符】来访问它。
我们既可以在类的内部,也可以在类的外部定义静态函数,但static关键字只能出现在类内部的声明语句。在类外部定义时,不可以重复static关键字!(与explicit相似)
类内的static数据成员,只能在类外定义,【除非它被声明为const(或者constexpr)】。它的定义一般和类的其他非内联函数的定义放到同一个文件。
class A{
private:
int a = 5; // 正确,表示类内初始值
const int b = 10; // 正确
static int c = 15; // 错误
static constexpr int d = 20; // 正确
}
注意:类的静态方法只能访问类的静态成员!!可以是private和protected。

静态数据成员可以是不完全类型,而非静态数据成员则受到限制,只能声明成它所属类的指针或引用。
class Bar{
public:

private:
static Bar mem1; // 正确,静态成员可以是不完全类型
Bar *mem2; // 正确,指针类型可以是不完全类型
Bar mem3; // 错误,数据成员必须是完全类型
}

C++Primer笔记-----day06的更多相关文章

  1. C++ Primer笔记

    C++ Primer笔记 ch2 变量和基本类型 声明 extern int i; extern int i = 3.14;//定义 左值引用(绑定零一变量初始值,别名) 不能定义引用的引用:引用必须 ...

  2. C++ Primer 笔记(1)基础中的战斗机 输入输出 对输入不定数据处理

    今天打算再重新好好的看一遍C++ Primer这本很经典的书籍,笔记开始: 1.每个C++程序都包含一个或者多个函数,其中必须有一个main,操作系统通过调用main入手运行程序: 2.函数包括:返回 ...

  3. C++ Primer 笔记 第一章

    C++ Primer 学习笔记 第一章 快速入门 1.1 main函数 系统通过调用main函数来执行程序,并通过main函数的返回值确定程序是否成功执行完毕.通常返回0值表明程序成功执行完毕: ma ...

  4. C++primer笔记之顺序容器

    最近又重新拾起C++primer,发现每一次看都会有不同的体验,但每一次看后因为不常用,忘记得很快,所以记笔记是很关键的一环,咋一看是浪费时间,实际上是节省了很多时间.下面就把这一节的内容做一个简单的 ...

  5. c++ primer 笔记 (一)

    昨天开始看的<C++ Primer>,确实不错.希望这周抓紧看完,每天做下笔记,以便以后复习. main函数返回一个值给操作系统   操作系统通过main函数返回的值来确定程序是否成功执行 ...

  6. C++ Primer笔记(1)——连续读取数据、类型对应的尺寸、类型转换、字符串分行写法

    这次要看看C++ Primer,这本基本上就是必读书籍了.下面的内容就是一些之前没有学过的知识的笔记. 读取数量不定的输入数据 虽然很简单,但是还是记一下: #include <iostream ...

  7. C++Primer笔记(3)

    标准库类型string表示可变长的字符序列,使用前先包含string头文件.(哈哈,终于可以逃脱C语言中的str函数系列了.)因为是标准库的一部分,所以string被定义在命名空间std中.所以你懂该 ...

  8. C++ Primer 笔记 第三章

    C++ Primer 第三章 标准库类型 3.1using声明 例: using namespace atd; using std::cin; 3.2string类型 初始化方式 string s1 ...

  9. C++ Primer 笔记 第二章

    C++ Primer 第二章 变量和基本类型 2.1基本内置类型 有算数类型和void类型:算数类型储存空间大小依及其而定. 算数类型表: 类型 含义 最小储存空间 bool 布尔型 - char 字 ...

随机推荐

  1. Java并发--ConcurrentModificationException(并发修改异常)异常原因和解决方法

    在前面一篇文章中提到,对Vector.ArrayList在迭代的时候如果同时对其进行修改就会抛出java.util.ConcurrentModificationException异常.下面我们就来讨论 ...

  2. BZOJ3673 可持久化并查集 by zky 【主席树】

    BZOJ3673 可持久化并查集 by zky Description n个集合 m个操作 操作: 1 a b 合并a,b所在集合 2 k 回到第k次操作之后的状态(查询算作操作) 3 a b 询问a ...

  3. 20179223《Linux内核原理与分析》第一周学习笔记

    第一周实验 尝试创建两个文件,用通配符查找这两个文件:在创建文件的时候,需要同时创建多个文件的方法运行. 根据作业要求,实现一个lilux命令. 根据作业要求添加一个用户loutest,使用sudo创 ...

  4. 实用符号Alt+小键盘快输

    键盘输入:①Word.写字板.QQ2011等,Alt+Unicode之十进制数:②记事簿.网页框.浏览器之地址栏.TM2009等,Alt+GBK之十进制数. 字符 Unicode 十进制数 GBK ...

  5. C#中的依赖注入那些事儿

    目录 目录 1 IGame游戏公司的故事 1.1 讨论会 1.2 实习生小李的实现方法 1.3 架构师的建议 1.4 小李的小结 2 探究依赖注入 2.1 故事的启迪 2.2 正式定义依赖注入 3 依 ...

  6. 笔记:webpack 打包参数 mode development

    webpack 打包参数 mode development 在开发时使用 webpack 打包后不压缩,所以只需要在 webpack 打包命令中加上 --mode mode development 即 ...

  7. ORACLE联机日志文件丢失或损坏的处理方法(转)

    经验总结: 联机日志分为当前联机日志和非当前联机日志,非当前联机日志的损坏是比较简单的,一般通过clear命令就可以解决问题. 损坏非当前联机日志:1.启动数据库,遇到ORA-00312 or ORA ...

  8. Debian初识(选择最佳镜像发布站点加入source.list文件)

    选择最佳镜像发布站点加入source.list文件:netselect,netselect-apt “该将哪个Debian镜像发布站点加入source.list文件?”.有很多方法来选择镜像发布站点, ...

  9. Web 漏洞分析与防御之 XSS(一)

    原文地址:Web 漏洞分析与防御之 XSS(一) 博客地址:http://www.extlight.com 一.全称 跨站脚本攻击(Cross Site Scripting) 二.原理 通过在网站中的 ...

  10. linux常用小技巧(持续更新中)

    一.设置固定ip地址1.config查看用的是哪一个网卡这是假设用的是eth12.修改dns地址vim /etc/resolv.confsearch 域名地址nameserver 192.168.3. ...