成员函数
- 对象参数
No. | 参数类型 | 建议 | 说明 | |
---|---|---|---|---|
1 |
const 类& /类 const &
|
优先使用 | 函数内部不改变参数对象 | |
2 | 类& |
需要时使用 | 函数内部改变参数对象 | |
3 | 类 |
不要使用,可以使用No.1代替 | 调用复制构造函数,增加开销 | |
4 |
const 类* /类const *
|
特殊情况下使用 | 第三方提供是对象指针时 | |
5 | 类* const |
不使用 | 无意义 | |
6 |
const 类* const /类const * const
|
不要使用,可以使用No.4代替 |
- 对象返回
No. | 返回值类型 | 建议 | 说明 | |
---|---|---|---|---|
1 |
const 类& /类 const &
|
返回值是const 成员变量必须使用 |
成员变量不随函数调用结束而销毁。 | |
2 | 类& |
返回值是非const 成员变量使用 |
成员变量不随函数调用结束而销毁。 | |
3 | 类 |
返回值为函数内部局部对象时使用 | 内部局部对象随函数调用结束而销毁,返回时调用复制构造函数重新生成临时对象,在函数外使用。 | |
4 |
const 类* /类const *
|
特殊情况使用 | 第三方提供是对象指针时/返回动态内存时 | |
5 | 类* const |
不使用 | 无意义 | |
6 |
const 类* const /类const * const
|
不要使用,可以使用No.3 |
析构函数、赋值构造函数、赋值运算符重载函数时机
匿名临时对象
类名(参数);
深拷贝的要点
- 深拷贝必须同时实现拷贝构造函数、赋值运算符重载、析构函数。
- 赋值运算符重载
- 判断是否是自我赋值
- 释放之前的资源
- 获取新资源
- 返回自身引用(
*this
)
运算符重载练习
- 复数:
- 双目算术运算符(
+
-
*
/
) - 单目算术运算符(
+
-
) - 复合赋值运算符(
+=
-=
*=
/=
) - 实数:
- 双目算术运算符(
+
-
*
/
) - 单目算术运算符(
+
-
) - 复合赋值运算符(
+=
-=
*=
/=
) - 比较运算符(
==
!=
>
>=
<
<=
) - 逻辑运算符(
&&
||
!
)
- 双目算术运算符(
- 字符串(模仿
std::string
)- 下标运算符(
[]
) - 连接运算符(
+
) - 连接运算符(
+=
) - 比较运算符(
==
!=
>
>=
<
<=
) - 输入输出运算符(
>>
<<
)
- 下标运算符(
- 指针
- 解引用(
*
) - 箭头访问(
->
)
- 解引用(
- 日期
- 自增减运算符:前缀自增(
++
) 后缀自增(++
) 前缀自减(--
) 后缀自减(--
) - 逻辑运算符(
&&
||
!
)
关键字explicit
- 语法:加在单参构造函数的前面
explicit 类名(参数)
- 作用:单参构造函数禁止隐式转换
- 背景
例如一个类的构造函数只有有一个参数
class Single{
public:
Single(int d):data_(d){
cout<< __func__ << d <<endl;
};
friend bool operator==(const Single& l,const Single& r){
return &l == &r or l.data_ == r.data_;
}
private:
int data_;
};
那么,他可以使用基本方式实例化
Single s1(10);
也可以使用=
实例化,如下:
Single s2 = 11;
不但如此,可以与单参类型隐式转换
void DoSomething(const Single& s){
cout << __func__ << endl;
}
DoSomething(12);
甚至可以直接与单参类型比较(前提是,重载比较运算符)
if(11 == s){
cout << "Equal" << endl;
}else{
cout << "No equal" << endl;
}
隐式转换提供便利的同时,增加了出错排错的困难。所以,有时想禁用单参构造函数的隐式转换功能。只要在单参构造函数前加上关键字explict
即可。
explicit Single(int d):data_(d){
cout<< __func__ << d <<endl;
};
这样,如下代码不能执行隐式转换,都会报错。(使用显示初始化就好了,即明显又不易出错)
Single s2 = 11;
DoSomething(12);
if(11 == s){
cout << "Equal" << endl;
}else{
cout << "No equal" << endl;
}
std::string
有几个单参函数(字符串指针、字符),并没有加上explicit
,所以可以方便与字符串指针、字符相互操作。
std::string
基本应用
实例化
string null_str;// 空串
string str("字符串");
string str(个数,字符);
string str = "字符串";
字符串数组
成员变量
字符串长度
size()
访问字符
[]
比较
==
>
<
应用:遍历字符找出字符串中首个没有重复的字符
把字符串的小写字母全部转化成大写
分别统计出字符串中数字、字母和其它符号出现的次数(cctype中的isdigit...)
连接字符串
+
与append(字符串)
字符串字面量是否可以连接?
- 分割字符串
substr(开始位置,长度)
- 查找子串
find(字符串)
(未找到返回string::npos
)
应用:切割字符串 - 把以逗号分割的字符串,转化成字符串数组
- 取出双引号之间的字符串,存放到字符串数组中
- 替换字符串
replace(开始位置、长度、新子串)
- 删除字符串
erase(开始位置、长度)
- 插入字符串
insert(开始位置、新子串)
应用:替换字符串
- 实现简单的VIM的替换命令
%s/旧串/新串/g
以上操作哪些改变了字符串?
转化
- 字符串对象转字符数组
c_str()
- 字符数组转字符串对象
- 字符串对象转整型
- 整型转字符串对象
- 字符串对象转浮点型
- 浮点型转字符串对象