const关键字

大家好:

  衷心希望各位点赞。

  您的问题请留在评论区,我会及时回答。


const含义

常类型是指使用类型修饰符const说明的类型,被const修饰的常类型的变量或对象的值是不能被更新的。

const作用

  • 可以定义常量
const int a = 100;
  • 类型检查
    const常量与#define宏定义常量的区别: const常量具有数据类型,编译器可以对其进行类型检查;#define宏定义没有数据类型,只是简单的文本替换,无法对其进行安全检查。
  • 防止修改,起保护作用,增加程序的健壮性
void f(const int i){
    i++; //报错!
}
  • 可以节省空间,避免不必要的内存分配
    const定义的常量从汇编的角度来看,只是给出了对应的内存地址,而不是像#define一样给出的是立即数,所以,const定义的常量在程序运行过程中只有一份拷贝,而#define定义的常量在内存中有若干个拷贝。🐰

const定义常量

const int b = 10;
b = 0; // error: assignment of read-only variable ‘b’
const string s = "helloworld";
const int i,j=0 // error: uninitialized const ‘i’

上述有两个错误,第一:b为常量,不可更改!第二:i为常量,必须进行初始化!(因为常量在定义后就不能被修改,所以定义时必须初始化。)

const与指针

与指针相关的const有四种:

const char * a; //指向const对象的指针或者说指向常量的指针。
char const * a; //同上
char * const a; //指向类型对象的const指针。或者说常指针、const指针。
const char * const a; //指向const对象的const指针。

小结:如果const位于类型的左侧,则const就是用来修饰指针所指向的变量,即常量指针。如果const位于指针类型的右侧,则const修饰指针本身,即指针常量。
具体使用如下:

  1. 常量指针(指针指向的值不能改变)
const int* ptr;
*ptr = 10; //error

ptr是一个指向int类型const对象的指针,const定义的是int类型,也就是ptr所指向的对象类型,而不是ptr本身,所以ptr可以不用赋初始值。但是不能通过ptr去修改所指对象的值。

  1. 指针常量(指针指向的地址不能改变)
    const指针必须进行初始化,且const指针的值不能修改。
#include<iostream>
using namespace std;
int main(){
    int num=0;
    int * const ptr=&num; //const指针必须初始化!且const指针的值不能修改
    int * t = &num;
    *t = 1;
    cout<<*ptr<<endl;
}

上述修改ptr指针所指向的值,可以通过非const指针来修改。

  1. 指向常量的常指针
    理解完前两种情况,下面这个情况就比较好理解了:
const int p = 3;
const int * const ptr = &p; 

ptr是一个const指针,然后指向了一个int 类型的const对象。

函数中使用const

const 修饰函数返回值

这个跟const修饰普通变量以及指针的含义基本相同:

  • const int
const int func1();

这个本身无意义,因为参数返回本身就是赋值给其他的变量!

  • const int*
const int* func2();

指针指向的内容不变。

  • int *const
int *const func2();

指针本身不可变。

const 修饰函数参数

  • 传递过来的参数及指针本身在函数内不可变,无意义!
void func(const int var); // 传递过来的参数不可变
void func(int* const var); // 指针本身不可变

表明参数在函数体内不能被修改,但此处没有任何意义,var本身就是形参,在函数内不会改变。包括传入的形参是指针也是一样。
输入参数采用“值传递”,由于函数将自动产生临时变量用于复制该参数,该输入参数本来就无需保护,所以不要加const 修饰。

  • 参数指针所指内容为常量不可变
void StringCopy(char* dst, const char* src);

其中src 是输入参数,dst 是输出参数。给src加上const修饰后,如果函数体内的语句试图改动src的内容,编译器将指出错误。这就是加了const的作用之一。

  • 参数为引用,为了增加效率同时防止修改。
void func(const A &a)

对于非内部数据类型的参数而言,象void func(A a) 这样声明的函数注定效率比较低。因为函数体内将产生A 类型

的临时对象用于复制参数a,而临时对象的构造、复制、析构过程都将消耗时间。

为了提高效率,可以将函数声明改为void func(A &a),因为“引用传递”仅借用一下参数的别名而已,不需要产生临

时对象。但是函数void func(A &a) 存在一个缺点:

“引用传递”有可能改变参数a,这是我们不期望的。解决这个问题很容易,加const修饰即可,因此函数最终成为

void func(const A &a)。

以此类推,是否应将void func(int x) 改写为void func(const int &x),以便提高效率?完全没有必要,因为内部数

据类型的参数不存在构造、析构的过程,而复制也非常快,“值传递”和“引用传递”的效率几乎相当。


小结:对于非内部数据类型的输入参数,应该将“值传递”的方式改为“const 引用传递”,目的是提高效率。例如将void func(A a) 改为void func(const A &a)。对于内部数据类型的输入参数,不要将“值传递”的方式改为“const 引用传递”。否则既达不到提高效率的目的,又降低了函数的可理解性。例如void func(int x) 不应该改为void func(const int &x)。
以上解决了两个面试问题:

  1. 如果函数需要传入一个指针,是否需要为该指针加上const,把const加在指针不同的位置有什么区别;
  2. 如果写的函数需要传入的参数是一个复杂类型的实例,传入值参数或者引用参数有什么区别,什么时候需要为传入的引用参数加上const。

类中使用const

在一个类中,任何不会修改数据成员的函数都应该声明为const类型。如果在编写const成员函数时,不慎修改数据成员,或者调用了其它非const成员函数,编译器将指出错误,这无疑会提高程序的健壮性。使用const关键字进行说明的成员函数,称为常成员函数。只有常成员函数才有资格操作常量或常对象,没有使用const关键字修饰的成员函数不能用来操作常对象。对于类中的const成员变量必须通过初始化列表进行初始化,如下所示:

class Apple
{
private:
    int people[100];
public:
    Apple(int i); 
    const int apple_number;
};
Apple::Apple(int i):apple_number(i)
{

}

const对象只能访问const成员函数,而非const对象可以访问任意的成员函数,包括const成员函数。例如:

//apple.cpp
class Apple
{
private:
    int people[100];
public:
    Apple(int i); 
    const int apple_number;
    void take(int num) const;
    int add(int num);
    int add(int num) const;
    int getCount() const;
};
//main.cpp
#include<iostream>
#include"apple.cpp"
using namespace std;
Apple::Apple(int i):apple_number(i)
{
}
int Apple::add(int num){
    take(num);
}
int Apple::add(int num) const{
    take(num);
}
void Apple::take(int num) const
{
    cout<<"take func "<<num<<endl;
}
int Apple::getCount() const
{
    take(1);
//    add(); //error
    return apple_number;
}
int main(){
    Apple a(2);
    cout<<a.getCount()<<endl;
    a.add(10);
    const Apple b(3);
    b.add(100);
    return 0;
}
//编译: g++ -o main main.cpp apple.cpp
//结果
take func 1
2
take func 10
take func 100

上面getCount()方法中调用了一个add方法,而add方法并非const修饰,所以运行报错。也就是说const对象只能访问const成员函数。而add方法又调用了const修饰的take方法,证明了非const对象可以访问任意的成员函数,包括const成员函数。除此之外,我们也看到add的一个重载函数,也输出了两个结果,说明const对象默认调用const成员函数。我们除了上述的初始化const常量用初始化列表方式外,也可以通过下面方法:
第一:将常量定义与static结合,也就是:

static const int apple_number

第二:在类外初始化:

const int Apple::apple_number=10;

如果你使用c++11进行编译,直接可以在定义出初始化,可以直接写成:

static const int apple_number=10;
或者
const int apple_number=10;

这里提到了static,下面简单的说一下:在C++中,static静态成员变量不能在类的内部初始化。在类的内部只是声明,定义必须在类定义体的外部,通常在类的实现文件中初始化。

在类中声明:

static int ap;

在类实现文件中使用:

int Apple::ap=666
  • 9
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值