C++声明之CV限定符

时间:2022-01-17 15:05:44

目录

1.const

1.1 const obj 如果调用 non-const member fun会编译出错

经典错误

1.2 例子:STD里的操作符重载

1.3 例子:《cpp primer》15节 -基类的定义


1.const

class complex
{
public:
  complex (double r = 0, double i = 0): re (r), im (i) { }
  complex& operator += (const complex&);
  complex& operator -= (const complex&);
  complex& operator *= (const complex&);
  complex& operator /= (const complex&);
  double real () const { return re; }
  double imag () const { return im; }
private:
  double re, im;

  friend complex& __doapl (complex *, const complex&);
  friend complex& __doami (complex *, const complex&);
  friend complex& __doaml (complex *, const complex&);
};

只有member fun后面可以用 const定义。成员变量后面不可以写const,写在前面。


1.1 const obj 如果调用 non-const member fun会编译出错

C++声明之CV限定符

经典错误

如果定义时没有写成void print() const{}; 就会报错。因为str是const-obj,不能用non-const member fun调用。 C++声明之CV限定符


1.2 例子:STD里的操作符重载

来自 TEMPLATE CLASS basic_string的定义。string类的operator[]。

  • 用户可能拿[]来改变字符串。如String str[5]="a";由于string的实现机制是共享模式,不可以直接改变内部内容。只能拷贝一份再改变。所以要做cow(copy on write)设计。
  • 如果const String str[2];str是const obj,其内部不可以动。所以不必考虑cow。const obj只会调用const fun,所以const fun不必写cow。
reference operator[](size_type _Off)
{
    // 下标可变序列
    // 必须考虑cow
  
    return (this->_Myptr()[_Off]);
}

const_reference operator[](size_type _Off) const
{
	// 下标不可变序列
    // 不必考虑cow

    return (this->_Myptr()[_Off]);
}
  • 可见const属于签名的一部分,这两个函数可以重载。 
  • C++规定:如果类中有成员函数的const fun 和nonconst fun同时存在,则const obj默认调用const fun,non-const obj默认调用non-const fun。
  • C++规定:non-const fun(obj可变)可以调用const fun(obj不可变)。 反过来报错。

1.3 例子:《cpp primer》15节 -基类的定义

class Quote {
	/* 
	** istream&:必须是非常量,因为本操作就是向流读出数据,其状态会改变;
	** Quote& :必须是非常量,因为本操作就是向此对象写入数据,此对象值会改变;
	*/  
	friend istream& operator>>(istream&, Quote&);
 
	
	/*
	**ostream& :必须是非常量,因为本操作就是向流写入数据,其状态会改变;
	**const Quote& :引用是因为避免赋值实参;常量是因为通常打印对象不会改变对象本身的值;
	*/
	friend ostream& operator<<(ostream&, const Quote&);  

public:

	/*这种传递中的代码使用以下变量来控制编译;
	**变量:IN_CLASS_INITS/DEFAULT_FCNS;对应C++作用:类初始状态设置/default(默认);
	*/
#if defined(IN_CLASS_INITS) && defined(DEFAULT_FCNS)
	//若定义类初始值设置及默认值,则定义默认构造函数,不接受任何实参;
	Quote() = default;  
#else
	/*若未定义类初始值及默认值,则定义构造函数,函数体空;
	**构造函数初始列表为新创建的数据成员price初始化,对应初始化值为 0.0;
	*/
	Quote() : price(0.0) { }
#endif // !defined(IN_CLASS_INITS) && defined(DEFAULT_FCNS)


	//初始化构造函数
	Quote(const string &book,double sales_price):
		bookNo(book),price(sales_price){ }

	//虚析构函数,动态绑定
#ifdef DEFAULT_FCNS
	virtual ~Quote() = default;
#else
	virtual ~Quote() {	}
#endif // DEFAULT_FCNS

	//const fun。 因为bookNo被期望是const obj,所以只能被const fun调用,必须定义为const fun,否则报错。
	string isbn() const 
	{ return bookNo; }

	//虚函数。将在派生类中重写,根据书的数量,采取不同的折扣算法。
	//因为price(定价)也是const obj,不会被改。所以定义为const fun。
	virtual double net_price(size_t n) const 
	{ return n*price; }//虚函数返回动态分配的自身副本


private:
	const string bookNo;//书号,被期望是const obj

protected:

#ifndef IN_CLASS_INITS
	const double price = 0.0;
#else
	const double price;
#endif // !IN_CLASS_INITS
};