• C++声明之CV限定符


    目录

    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会编译出错

    经典错误

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


    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
    };
  • 相关阅读:
    LeetCode15.3 Sum
    LeetCode215. Kth Largest Element in an Array
    python基础结构的时间复杂度
    顺时针打印矩阵
    合并k个有序链表
    LeetCode3. Longest Substring Without Repeating Characters
    决策树剪枝问题
    LeetCode98. Validate Binary Search Tree
    LeetCode96. Unique Binary Search Trees
    Visio软件不能使用方向键移动图形的解决办法
  • 原文地址:https://www.cnblogs.com/lightmare/p/10441046.html
Copyright © 2020-2023  润新知