就是用来恐吓你的

我能想到的,最短的。且const最多的一个语句是:

int const * fun(int const*const a[],const int index)const;

而这个语句还有下面若干等价语句:

const int * fun(int const * const a[],const int index)const;
const int * fun(const int * const a[],const int index)const;
const int * fun(const int * const a[],int const index)const;
//…

看似复杂,实际上分析const的语义时,唯一的一条原则就是:

const始终修饰其左側的类型标识符(简单的或复合的),唯一例外是当其左側没有类型标识符时。则修饰其右側第一个简单类型标识符。

const与常量

好了,从最简单開始分析。

使用const最简单的使用方法就是修饰一个简单的类型标识符。

int const varInt;

依据上面的分析原则,const修饰的是int。则表示varInt是一个整型常量。其值一旦初始化,就不可改动。

而我们知道,const的位置是非常灵活的。它既能够放在类型标识符的前面,也能够放在类型标识符的后面,所以以下的写法也是正确的。

const int varInt;

这实际就是上面分析原则的例外情况,const的左側在没有类型标识符,则其修饰其右側的int,即表示varInt是一个整形常量。

const与指针

那么。略微复杂些,看看指向常量的指针

const int* pInt;

这又是上面分析原则的例外情况,那么const修饰的是int(注意分析原则的描写叙述,const修饰的是右側第一个简单类型标识符。所以说const修饰的是int,而不是int *),表示是个整型常量,然后我们看到“* ”,表示pInt是个指针。其指向的就是个const int 常量,而pInt本质上还是个变量。

接下来理所当然得有与上面的等价的写法:

int const* pInt;

这就是分析原则里首先说的情况了,const修饰的是int,所以pInt还是一个指向整型常量的指针。

假设再把const往右移一位。就发生质变了:

int * const pInt;

这时,const修饰的是int *。即整型的指针,这是一个复合类型,进一步的,const表示这个指针pInt是个常量,即一旦初始化。pInt就不能再指向别的整型变量了,这就是指针常量

当然其所指的对象还是能够被改动的。

再进一步:

const int * const pInt;
int const * const pInt;

这两个语句都是在声明一个指向整型常量的指针常量,即pInt不能被改动,其所指的对象也不能被改动。

当然,在实际应用中差点儿非常少有这样的写法。

const与成员函数

到这里。最上面的语句中,前四个const事实上已经分析清楚了,就剩下最后一个了。

这个语句的本意是想声明一个函数。仅仅是函数的參数和返回值的类型太复杂了,从而显得不值关,所以为了分析最后一个const,我们简化下这个语句:

int fun(int)const;

首先。我们区分下函数的类型和函数的名称,一般的。对于一个函数的声明:

void fun(void);

表示声明了一个名称为fun,參数为void,即无參,返回值为void,即无返回值的函数,所以这个函数的名称就是fun。类型就是參数为void,返回值为void。即void (void) .

依据上面的分析原则。const修饰的应该是int(int)。而fun是函数的名称,那么就应该理解为这是在声明一个參数为整型,返回值为整型的常量函数(听着怪怪的),实际上这样的函数叫做常量成员函数,哈,一听就明确了。这表示这样的函数仅仅能做类的成员函数,而不能做全局函数。那么它的作用也就是唯一的,保证在这个函数的函数体内。不会有改动类的成员变量的操作。否则编译就会报错。

类型标识符通过const的修饰后,就变成了复合类型。也就是一种新的类型。所以使用const类修饰成员函数,也能实现成员函数的重载,即在一个类里。通过在成员函数声明的末尾带于不带const,而从形成一个成员函数的const和非const版本号。

当使用这样的类型的const对象来调用该函数时,则调用的是该函数的const的版本号,而使用很量对象来调用该函数时,则调用的是该函数的非const版本号。

如此一来,总算搞清了const,可是,我们也看到因为在const使用的灵活,在写代码时既能够放在被修饰类型标识符(仅仅能是简单类型标识符)的前面。也能够放在被修饰类型标识符的后面。从而导致我们在读代码时会混乱,所以,建议在写代码的时候不要把const放在语句的最前面,而是仅放在被修饰的类型标识符的后面。这样在分析代码也就不用担心上述分析原则的例外情况了。这不是简单非常多了嘛!

就先总结这么多了!难免有不准确之处。欢迎讨论。     :)

躁动不安的const的更多相关文章

  1. openssl 1.1.1 reference

    openssl 1.1.1 include/openssl aes.h: # define HEADER_AES_H aes.h: # define AES_ENCRYPT 1 aes.h: # de ...

  2. const,static,extern 简介

    const,static,extern 简介 一.const与宏的区别: const简介:之前常用的字符串常量,一般是抽成宏,但是苹果不推荐我们抽成宏,推荐我们使用const常量. 执行时刻:宏是预编 ...

  3. C++中的const

    一,C++中const的基本知识 1.C++中const的基本概念 1.const是定义常量的关键字,表示只读,不可以修改. 2.const在定义常量的时候必须要初始化,否则报错,因为常量无法修改,只 ...

  4. const extern static 终极指南

    const extern static 终极指南 不管是从事哪种语言的开发工作,const extern static 这三个关键字的用法和原理都是我们必须明白的.本文将对此做出非常详细的讲解. co ...

  5. const let,console.log('a',a)跟console.log('a'+a)的区别

    const 创建一个只读的常量 let块级作用域 const let重复赋值都会报错 console.log('a',a) a console.log('a'+a) a2 逗号的值会有空格:用加号的值 ...

  6. es6之let和const

    在javascript中,我们都知道使用var来声明变量.javascript是函数级作用域,函数内可以访问函数外的变量,函数外不能访问函数内的变量. 函数级作用域会导致一些问题就是某些代码块内的变量 ...

  7. construction const parameter问题 构造函数const引用参数问题

    工程在window下编译没有任何问题, 但是在linux(CentOS6)下编译就老是报错 C++ 编译器已升级到最新版 6.1.0 错误如下: In file included /bits/stl_ ...

  8. Error:const char* 类型的实参和LPCWSTR类型的形参不兼容的解决方法。

    在C++的Windows 应用程序中经常碰到这种情况. 解决方法: 加入如下转换函数: LPCWSTR stringToLPCWSTR(std::string orig) { size_t origs ...

  9. C#基础知识七之const和readonly关键字

    前言 不知道大家对const和readonly关键字两者的区别了解多少,如果你也不是很清楚的话,那就一起来探讨吧!探讨之前我们先来了解静态常量和动态常量. 静态常量 所谓静态常量就是在编译期间会对变量 ...

随机推荐

  1. STL容器迭代过程中删除元素技巧(转)

    1.连续内存序列容器(vector,string,deque) 序列容器的erase方法返回值是指向紧接在被删除元素之后的元素的有效迭代器,可以根据这个返回值来安全删除元素. vector<in ...

  2. 学习RFT之:TestObject.find方法的了解与使用

    第一部分:了解TestObject.find 一.TestObject.find方法的作用 1.测试过程中动态的找到测试对象(控件.标签等),使我们的测试用例不再依赖RFT自带的对象地图(Object ...

  3. iOS中ARC和非ARC混用

    如果在使用第三方类库的时候,我们可能会遇到一些内存管理的问题   那么如何在一个工程中实现ARC和非ARC混用呢,例如你创建一个ARC的工程,但是你引用的第三方类库是非ARC管理内存的   首先点击工 ...

  4. LR事务、集合点

    事务(Transaction):为了衡量服务器的性能,我们需要定义事务.比如:我们在脚本中有一个数据查询操作,为了衡量服务器执行查询操作的性能,我们把这个操作定义为一个事务,这样在运行测试脚本时,Lo ...

  5. 【DVWA】【SQL Injection(Blind)】SQL盲注 Low Medium High Impossible

    1.初级篇 Low.php 加单引号提交 http://localhost/DVWA-master/vulnerabilities/sqli_blind/?id=1'&Submit=Submi ...

  6. Squid 正向代理

    实现通过特定设备对特定的网址访问加速 使用squid 正向代理 实现,区别于反向代理,两者区别的根本在于作为中转的服务器在一个完整的请求中是代表客户端还是代表服务器. 服务端设置 1.安装程序包(推荐 ...

  7. mysqlbinlog flashback 使用最佳实践

    mysqlbinlog限制 该软件利用binlog中记录了操作前的数据镜像和操作后的数据镜像.有如下限制 1)binlog_format=row 2)必须打开binlog 3)只支持insert.up ...

  8. Handling unhandled exceptions and signals

    there are two ways to catch otherwise uncaught conditions that will lead to a crash: Use the functio ...

  9. TP中U方法详解

    U方法常用于ThinkPHP里的页面跳转 官方称为url组装, 就是根据某种规则组成一个url地址,这个功能就叫组装. 在ThinkPHP里,系统提供了一个封装的函数来处理url的组装,俗称U方法. ...

  10. 重启rsyncd

    systemctl  restart  rsyncd.service