C++的const

既要使数据能在一定范围内共享,又要保证它不被任意修改,可使用const

·        const修饰的声明数据成员称为常数据成员

·        const修饰的声明成员函数称为常成员函数

·        const修饰的定义对象称为常对象

变量或对象被const修饰后其值不能被更新。因此被const修饰的变量或对象必须要进行初始化。
(一)用const修饰的声明数据成员称为常数据成员 

   有两种声明形式:

const int  cctwl;
int const  cctwl;
不能省略数据类型,可以添加 public private等访问控制符。 

说明:

1. 任何函数都不能对常数据成员赋值。

2. 构造函数对常数据成员进行初始化时只能通过初始化列表进行。(只有此一种方法)

3. 常数据成员在初始化时必须赋值或称其必须初始化.

4. 如果类有多个默认构造函数必须都初始化常数据成员。

通过下面例子来理解常数据成员以上4点。

这样也可以给常数据成员num赋值啊,而且可以不同的对象赋不同的值。

#include<iostream>
using namespace std;

class Student
{
private:
    const int num;
    int score;
public:
    Student(int a, int b):num(a)
    {
        score = b;
    }
    void display();
};

void Student::display()
{
    cout << num << endl << score << endl;
    cout << "sizeof(int)" << sizeof(int) << endl;
    cout << "sizeof(Student)" << sizeof(Student) << endl;
}

int main()
{
    Student stu(5,8);
    stu.display();
    return 0 ;
}

运行结果如图:

A、请指出下面程序的错误

class A
{
private:
    int w, h;
    const int ci = 5;    //错误一
public:
};

void main()
{
    A a;                //错误二
    cout << "sss";
    return 0;
}

错误一:不能对常数据成员在类中初始化、要通过类的构造函数。

错误二:没有合适的默认构造函数可用。因为有常量ci没有初始化,必须初始化所有常数据成员。

更正后结果如下:

class A
{
private:
    int w, h;
    const int ci;
public:
    const int ci_2;         //可以是公有私有访问权限
    A():ci(1),ci_2(2){}     //通过构造函数初始化列表初始化常数据成员
};

B、多个构造函数下的常数据成员

请指出下面程序的错误:

class A
{
private:
    int w, h;
    const int ci;
public:
    const int ci_2;         
    A():ci(1),ci_2(2){}     
    A(int x, int y)         //错误一
    {
        w = x;
        h = y;
    }
};

void main()
{
    A a;
    A b(3,5);
    cout << "sss";
    return 0;
}

错误一:每个构造函数都要初始化常数据成员,应改为

A(int x, int y):ci(7),ci_2(8)      
    {
        w = x;
        h = y;
    }


(二)用const修饰的声明声明成员函数称为常成员函数

声明:<类型标志符>函数名(参数表)const

说明: 

1. const是函数类型的一部分,在实现部分也要带该关键字。

2. const关键字可以用于对重载函数的区分。

3. 常成员函数不能用来更新类的成员变量,也不能调用类中未用const修饰的成员函数,只能调用常成员函数。即常成员函数不能更改类中的成员状态,这与const语义相符。

A、通过例子来理解const是函数类型的一部分,在实现部分也要带该关键字。

class A
{
private:
    int w,h;
public:
    int getValue() const;
    int getValue();
    A(int x,int y)
    {
        w=x;
        h=y;
    }
    A(){}
};
int A::getValue() const         //实现部分也带该关键字
{
    return w*h;
}

void main()
{
    A const a(3,4);
    A c(2,6);
    cout << a.getvalue()<<c.getValue() <<"test";
    return 0;
}

B、通过例子来理解const关键字的重载

class A
{
private:
    int w,h;
public:
    int getValue() const;
    int getValue();
    A(int x,int y)
    {
        w=x;
        h=y;
    }
    A(){}
};
int A::getValue() const         //实现部分也带该关键字
{
    return w*h;
}
int A::getValue()
{
    return w+h;
}

int main()
{
    A const a(3,4);
    A c(2,6);
    cout << a.getvalue()<< c.getValue() << "test";
    return 0;
}

C、通过例子来理解常成员函数不能更新任何数据成员

class A
{
private:
    int w,h;
public:
    int getValue() const;
    int getValue();
    A(int x,int y)
    {
        w=x;
        h=y;
    }
    A(){}
};
int A::getValue() const         //实现部分也带该关键字
{
//    w = 10; h = 10;      // 错误,常成员函数不能更新非常数据成员
    return w*h;
}
int A::getValue()
{
    w = 10; h = 10;
    return w+h;
}

int main()
{
    A const a(3,4);
    A c(2,6);
    cout << a.getValue() << endl << c.getValue() << endl << "test";
    return 0;
}

D、通过例子来理解

1、常成员函数可以被其他成员函数调用。
2、但是不能调用其他非常成员函数。
3、只可以调用其他常成员函数。

class A
{
private:
    int w,h;
public:
    int getValue() const
    {
//        return w*h + getValue2();       // 错误,常成员函数不能调用非常成员函数
        return w*h;
    }
    int getValue2()
    {
        return w+h+getValue();
    }
    A(int x,int y)
    {
        w=x;
        h=y;
    }
    A(){}
};


int main()
{
    A const a(3,4);
    A c(2,6);
    cout << a.getValue() << endl << c.getValue2() << endl << "test";
    return 0;
}

(三)用const修饰的定义对象称为常对象

常对象是指对象的数据成员的值在对象被调用时不能被改变。常对象必须进行初始化,且不能被更新。不能通过常对象调用普通成员函数,但是可以通过普通对象调用常成员函数。常对象只能调用常成员函数。

常对象的声明如下: 

     const  <类名>  <对象名>

     <类名>  const  <对象名>

两种声明完全一样没有任何区别。

1、通过下面例子来理解常对象:

A、请指出下面程序的错误

class A
{
private:
    int w,h;
public:
    int getArea() const
    {
        return w*h;
    }
    int getW()
    {
        return w;
    }
    void setWH(int x,int y)
    {
        w = x;
        h = y;
    }
    A(int x,int y)
    {
        w=x;
        h=y;
    }
    A(){}
};


int main()
{
    A a;            //普通对象不初始化不会报错
    a.setWH(3,9);
    A const b(3,6);//常对象必须声明的同时初始化
//    b.setWH(3,7);   //错误:因为b是常对象,不能调用非常成员函数。
    cout << a.getArea() << endl << b.getArea() << endl << "test";
    return 0;
}


PS:

  既要使数据能在一定范围内共享,又要保证它不被任意修改,这时可以使用const。对于const  int *pa,指的是不能通过改变*pa的值来改变pa指向的变量的值,但可以通过改变pa指向的变量的值来改变*pa的值。设置为常引用,还是可以改变实际的值。(但不可用*pa或b来改变a的值)

指向对象的常指针,其指向始终不变。

指向常变量的指针变量:const 类型名 *指针变量名;(把变量换成对象,即指向常对象的指针变量同下)

1、如果一个变量已被声明为常变量,只能用指向常变量的指针变量指向它,而不能用一般的(指向非const型变量的)指针变量指向它。

2、指向常变量的指针变量除了可以指向常变量外,还可以指向未被声明为const的变量。此时不能通过此指针变量改变该变量的值

3、如果函数的形参是指向非const型变量的指针,实参只能用指向非const变量的指针,而不能用指向const变量的指针

C++面向对象程序设计中,经常用常指针和常引用作函数参数。这样既能保证数据安全,使数据不能被随意修改,在调用函数时又不必建立实参的拷贝。用常指针和常引用作函数参数,可以提高程序运行效率。

猜你喜欢

转载自blog.csdn.net/buknow/article/details/80275843