C++威力强大的助手 --- const

 Welcome to 9ilk's Code World

       

(๑•́ ₃ •̀๑) 个人主页:        9ilk

(๑•́ ₃ •̀๑) 文章专栏:     C++之旅 


const是个奇妙且非比寻常的东西,博主从《Effective C++》一书中认识到关于const更深层次的理解,写此博客进行巩固。


🏠 const的作用

  • const可以指定一个“不该被改动的对象”,编译器会强制实施这个约束
  • const可修饰类外的作用域(全局域,命名空间域等)中的常量,文件,函数或静态常量等
  • const可修饰类内的static和non-static成员

🏠 const修饰指针和迭代器

📌 const修饰指针

const修饰指针无外乎三种:

char greet[] = "hello";
char* pg = greet;

//指针指向内容不可变
const char * pg = greet;
char const * pg = greet;

//指针本身不可变
char * const pg = greet;

//指针本身和指针执行内容都不可变
const char * const pg = greet;

总结:

1.const出现在*号左边时,表示指针指向的内容是常量。

2.const出现在*号右边时,表示指针本身是常量。

3.const出现在*号左右边时,表示指针本身和指针指向的内容都是常量

📌 const与迭代器

  • 声明迭代器为const

我们把迭代器看作一个类型,它的作用就像个T*的指针,当声明迭代器为const时,类比const int是int这个int类型变量不可能,此时说明迭代器类型变量是不能变的,也就是类比修饰一个T* const的指针。

vector<int> v  = {4,1,2,9,10};
const vector<int>::iterator it = v.begin();

it++; //错误,迭代器类型对象不可变
(*it)++; //正确,指向内容可以变 

我们也可以利用下面代码进行验证,当生成解决方案时会对其进行报错:

template<class T>
void fun()
{
	const T x;
	cout << typeid(x).name() << endl;
	x++;
}
int main()
{
	//op o;
	//cout << o[1];
	vector<int> v = {3,5};
	vector<int>::iterator it = v.begin();;
	fun<vector<int>::iterator>();
	return 0;
}
  • const迭代器

  如果你希望迭代器所指的东西不可被改动(也就是类似希望模拟个const T*指针),你需要的是const_iterator.

std::vector<int> v = {1,5,2,3,4};

std::vector<int>::const_iterator it = v.begin();

*it  = 10; //错误,此时是const迭代器,指向内容不可改变。
it++; //此时不是声明迭代器为const,本身迭代器可以改变。

注:我们平时所说的const迭代器就是const_iterator,请注意区分与前者声明迭代器为const进行区分。

🏠 const与函数

📌 const与函数参数

建议:如果对于局部对象或对于函数参数没有改动的需求时,建议将他们声明为const,此时可以避免不必要的麻烦。比如:将“==”键值的“=”。

📌const与函数返回值

  • 函数返回一个常量,往往可以降低因客户错误而造成的意外,而且具有安全性和高效性

    假设有这样的一个有理数类:

class Rational;

const Rational operator*(const Rational& r1,const Rational& r2)
{//...};

Rational a,b,c;
...
(a*b) = c;

由于是有理数类,客户可能会将乘积再做一次赋值,此时将operator*的返回值类型设置为const就可以避免这样无意义的赋值动作。

🏠 const与成员函数

📌 const成员函数的好处

1. const成员函数易使class接口比较容易被理解,清楚知道哪个函数可以改动对象内容而哪个函数不行。

2. 有了const成员函数,const对象就能被“操作”,因为const对象只能调用const成员函数;能操作const对象,就能利用传引用(const 类类型 &)提高效率。

  • const成员函数与普通成员函数的区别

我们知道普通成员函数参数都有一个隐含的this指针,是不能修改的,也就是类 * const this;而const成员函数的this指针,类型是const 类 * const this,也就是说此时对象的内容也不能被修改

class Date
{
public:
//... 
	const char& operator[](size_t position)const //operator[]for const对象
	{
		return _date[position];
	}

	char& operator[](size_t position) //operator[]for non-const 对象
	{
		return _date[position];
	}

private:
	string _date;

};

int main()
{

 Date d1("2024/08/04");
 cout << d1[0] << endl; //调用的是non-const版本

 const Date d2("2024/08/04")
 cout << d2[0]; //调用的是const版本
 return 0;
}

  • 说明

1. 两成员函数如果只是常量性不同,也是可以被重载的。因为两个版本隐含的this类型不同,同时

返回值类型也跟着不同。

2.const对象d2的对象指针为const Date* ,更匹配const版本的operator[],因此调用const版本;

而非const对象d1的对象指针为Date*,更匹配非const版本。

📌 bitwise const VS logical const

        到这里,我们思考一下什么成员函数如果是const意味着什么?目前有以下两个流行概念需要向大家介绍一下:

  • bitwise constness

这种观点的人认为,成员函数只有在不改变对象之任何成员变量时才可以说是const,也就是说不改变对象内的任何一个bit。这种论点的好处是很容易找到违反点:只需找到对成员变量的赋值动作即可。

这种观点正是C++对常量性的定义,因此const成员函数不可以更改对象内任何非静态成员变量。

如果只有指针(而非所指向内容)隶属于对象,(比如有这样的一个类,将数据存储于char*而不是string),不修改char*而修改char*指向内容,此时编译器是认为是bitwise constness的,可以正常通过编译。

class Block
{
public:
	Block( char* ch )
		:pText(ch)
	{}
	 char& operator[](size_t position)const
	{
		return pText[position];
	}


private:
	char* pText;
};

int main()
{
	char arr[] = "hello";
	char* ch = arr;
	const Block cctb(ch);
    char* pc = &cctb[0];
	*pc = 's';
	return 0;
}

此时可以通过这个漏洞修改成员变量指针指向的内容而不违法const

(注:如果成员变量是string,由于operator[]需要访问pText成员变量,并且你需要保证Block对象的状态不被修改,所以pTextoperator[]调用也必须是const的,因此不会出现上述漏洞。)

这种情况导出所谓的logical constness.

  • logical constness
class BigArray 
{
    vector<int> v; 
    int accessCounter;
public:
    int getItem(int index) const 
   { 
        accessCounter++;
        return v[index];
   }
};

此类提供了一个 getItem 接口,除此之外,为了计算外部访问数组的次数,该类还设置了一个计数器 accessCounter ,可以看到用户每次调用 getItem 接口,accessCounter 就会自增,很明显,这里的成员 v 是核心成员,而 accessCounter 是非核心成员

我们希望接口 getItem 不会修改核心成员,而不考虑非核心成员是否被修改,此时 getItem 所具备的 const 特性就被称为 logic constness

问题:在这个函数中虽然accesCounter修改对对象而言可以被接受,但是编译器只认bitwise constness不允许修改怎么办?

  • mutable

mutable是C++中一个与const相关的摆动场,他可以释放掉non-static成员变量的bitwise constness约束。

class BigArray {
    vector<int> v; 
    mutable int accessCounter; //像这样的成员变量可能总是会被更改。
public:
    int getItem(int index) const { 
        accessCounter++;
        return v[index];
    }
};

总结:当const成员函数接受某些修改之后不改变成员函数逻辑状态的成员变量时,这时可以使用mutable来释放const约束。但注意mutable可能会违反对象的不变性,需要慎用。

📌 在const和non-const成员函数中避免重复

   对于“bitwise constness”非我所欲的问题,mutable是个解决办法,但并不能解决所有的难题。假设有个类,类内的operator[ ]不单只是返回一个引用指向某字符,也执行边界检验,志记访问信息,甚至可能进行数据完善性检验等...把所有这些放进const版本和非const版本的operator[ 里的问题是会导致代码膨胀以及大量代码重复:

class Text
{

public:
  char& operator[](size_t pos)
  {
      //... 边界检验
      //... 志记数据访问
      //... 检验数据完整性
     return text[pos];
  }

 const char& operator[](size_t pos)const
 {
      //... 边界检验
      //... 志记数据访问
      //... 检验数据完整性
     return text[pos];
 }


private:
   string text;

};

避免代码重复的安全做法:

class Text
{

public:
  const char& operator[](size_t pos)
  {
      //... 边界检验
      //... 志记数据访问
      //... 检验数据完整性
     return text[pos];
  }

 char& operator[](size_t pos)const
 {
    return   (char&)(((const Text &)(*this))[pos]);
 }


private:
   string text;

};
  • 说明

1. 这份代码进行了两次转型实现了了“运用const成员函数实现其non-const兄弟”,避免了代码重

复。

2.第一次转型将(*this)也就是这个对象类型强转为const Text&是为了匹配const版本调用const版

本的operator[ ],否则会陷入无限调用非const版本;第二次转型则是用来从const operator[ ]的

返回值中移除const,这其中并未有权限放大的问题,强转是可行的。

3. 反向调用也就是“令const版本调用non-const版本以避免代码重复”是一件错误的事,因为non-

const并未承诺绝不改变其对象的逻辑状态,因此这种做法可能使得对象被改动,当然编译器也不

允许const调用非const,也是一种权限的放大。


总结:

1. 将某些东西声明为const可帮助编译器侦测出错误用法,比如错误的赋值行为使得不必要的对象改动。

2.const可施加于任何作用域的对象,函数参数,函数返回值,成员函数。

3.如果在const成员函数内想改变非核心成员变量以达目的,可利用mutable解除const约束。

4.当const与非const成员函数实质有着等价的实现且代码有大量重复时,可考虑复用const版本以实现非const版本。

  • 63
    点赞
  • 53
    收藏
    觉得还不错? 一键收藏
  • 36
    评论
评论 36
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值