C++中const使用注意要点(一)
最近看《C++编程思想》发现自己的基础确实不牢固,也想起了以前写代码时也因为const的事情浪费过时间,这里总结下几个要点。
首先说下内部链接和外部链接。
当一个cpp文件在编译时,预处理器首先递归包含头文件,形成一个含有所有必要信息的单个源文件,这个源文件就是一个编译单元。编译器对每个编译单元(.cpp文件)进行编译生成相应.obj文件。PS:.c文件对应.o文件
接下来关键一步:链接器将所有不相关的.obj文件进行链接,生成最终可执行文件(.exe文件)
// a.cpp:
#include "stdafx.h"
void show(); int main()
{
show();
return 0;
}
// b.cpp
#include "stdafx.h"
#include <iostream>
void show()
{
std::cout << "Hello World" << std::endl;
}
看上述代码,函数的声明和定义可以在不同的cpp文件中,并且声明可以有很多个,定义只能有一个!
在某个编译单元中调用show函数,那么必须进行声明。来看下外部链接和内部链接的定义:
外部链接:一个名称对编译单元来说不是局部的,链接的时候其他编译单元可以访问它。
内部链接:一个名词对编译单元来说是局部的,链接的时候其他编译单元无法链接到它且不会与其他编译单元的同样称冲突。
可以看出,show函数是外部链接,a.cpp中的void show();等价于extern void show();
而static和全局的变量或函数都是外部链接,而由于函数的声明跟定义明显有区别(一个后接分号一个后接大括号),所以不用特地加extern区分。
回到主题,C++的const默认是内部链接的!也就是说const仅在const被定义过的文件里才是可见的。除非用extern进行声明,才会使用外部链接。
默认内部链接时,C++编译器不会给const创建存储空间,而是把它的定义放在符号表中。这称之为常量折叠,起到了跟#宏一样的效果。当然,在运行时还是会创建的,见下述例子。
#include <iostream> int main()
{
const int i = 10;
int* p = const_cast<int*>(&i);
*p = 8; std::cout << *p << " " << i << std::endl; return 0;
}
结果是8 10
说明运行时可以取得const常量的地址,并且可以改变存放的值。但是对于const常量来说,实际只是从符号表中查找值。因此虽然内容发生了变化,但是i输出还是10。
再来看看const的常见用法(下面的int均可换成其他基本数据类型或类名)
1、和指针一起使用
const int*或int const *——不能通过指针改变int的值,但是指针指向的地址可以改变,一般使用前者
int* const——不能改变指针指向的地址,但是可以通过指针改变int的值
const int* const或int const* const——不能通过指针改变int的值,也不能改变指针指向的地址
2、一般要把const int*转换成int*需要强制转换,而字符数组是没有强调const特性的,见下面代码
#include <iostream> int main()
{
char* pc = "hello world!";
std::cout << pc << std::endl; // 可以输出
pc[2] = 's'; // 运行时错误!
return 0;
}
编译通过,也能输出hello world!,但是运行时出错。这里"hello world!"被编译器当作常量字符数组建立的,所以得到的只是数组在内存里的首地址,对字符数组内任何字符进行修改会导致运行时错误。
要想修改字符串需要把它放在数组中,即把char* pc改成char pc[]
3、作为函数参数和返回值
作为参数:比如void f(const int i){}假如调用f(5);相当于在函数体中const int i = 5;然后再对i进行操作,所以不能修改i。
作为返回值:比如const int f(){ return 1; }假如调用int i = f();相当于const int tmp = f(); int i = tmp;所以没有任何意义,返回值进行值传递,还是能改变。
但是返回的是对象(假如是类A的对象)时,比如const A f(); 返回值不能作为左值!(左值右值以后再详述)
虽然A g();中g()的返回值可以作为左值,但是也只是临时对象,表达式被编译过后临时对象也会被清除
C++中const使用注意要点(一)的更多相关文章
- C++中const使用注意要点(二)
当const修饰类的成员变量 1.const修饰类的非静态成员时必须在构造函数初始化列表上初始化: 在构造函数内会提示表达式必须是可修改的左值,因为在构造函数内并不是初始化,仅仅是赋值,而const类 ...
- C、C++中const的区别
C语言中: 被const修饰的变量,仍然是变量.虽然不能用C语法给这个变量改变值,但他本质上还是变量. C编译器会给它分配空间. C中,const默认使用的是外部链接. C++中: 被const修饰的 ...
- JavaScript中const、var和let区别浅析
在JavaScript中有三种声明变量的方式:var.let.const.下文给大家介绍js中三种定义变量的方式const, var, let的区别. 1.const定义的变量不可以修改,而且必须初始 ...
- C/C++中const的用法 分类: C/C++ 2015-07-05 00:43 85人阅读 评论(0) 收藏
const是C语言的关键字,经C++进行扩充,变得功能强大,用法复杂.const用于定义一个常变量(只读变量),当const与指针,引用,函数等结合起来使用时,情况会变得复杂的多.下面将从五个方面总结 ...
- C++中const 的各种用法
C++中const 关键字的用法 const修饰变量 const 主要用于把一个对象转换成一个常量,例如: ; size = ; // error: assignment of read-only v ...
- (转) C/C++中const关键字详解
文章转自 http://www.cnblogs.com/yc_sunniwell/archive/2010/07/14/1777416.html 为什么使用const?采用符号常量写出的代码更容易维 ...
- 实例讲述PHP面向对象的特性;;;php中const与define的使用区别
php中const与define的使用区别 1.const:类成员变量定义,一旦定义且不能改变其值. define:定义全局常量,在任何地方都可以访问.2.define:不能在类中定义,而const可 ...
- C++中const简介及用法
1.const简介 C++中的const关键字的用法非常灵活,而使用const将大大改善程序的健壮性,本人根据各方面查到的资料进行总结如下,期望对朋友们有所帮助. Const 是C++中常用的类型修饰 ...
- C/C++ 中 const 修饰符用法总结
C/C++ 中 const 修饰符用法总结 在这篇文章中,我总结了一些C/C++语言中的 const 修饰符的常见用法,供大家参考. const 的用法,也是技术性面试中常见的基础问题,希望能够帮大家 ...
随机推荐
- NumberFormat
package com.NumberFormat; import java.text.NumberFormat; public class Study01 { public static void m ...
- 初次使用VCS仿真软件
由于刚开始接触VCS,对于VCS不是太了解,在网上找了很多的资料终于遇到了一个相对比较初级的入门资料,这个资料是以一个简单的4位加法器为例来介绍vcs的用法的,比较好入门,这个文章的地址如下: htt ...
- Linux内存中Swap机制(转)
在做监控时,发现内存中有一项Swap space,不是很理解,这里查了一些资料: http://blog.sina.com.cn/s/blog_502d765f0100krph.html 在linux ...
- 利用 squid 反向代理提高网站性能(转载)
本文在介绍 squid 反向代理的工作原理的基础上,指出反向代理技术在提高网站访问速度,增强网站可用性.安全性方面有很好的用途.作者在具体的实验环境下,利用 DNS 轮询和 Squid 反向代理技术, ...
- C# 如何判断字符串中是否包含另一个字符串?
如 字符串1(str1)为:“你好怎么解决呢!” 字符串2(str2)为:“你好” 如果str1里面包str2 则 Response.Write("成功");否则 Resp ...
- [置顶]
kubernetes1.7新特性:日志审计变化
背景概念 出于安全方面的考虑,Kubernetes提供了日志审计记录,用来记录不同普通用户.管理员和系统中各个组件的日志信息. Kubernetes日志审计是Kube-apiserver组件的一部分功 ...
- 学习三部曲:WHAT、HOW、WHY
一个人学习的过程要经历以下三步,才可以说得上"学会"两字: 第一步:WHAT 所谓的"WHAT",就是搞清楚某个东东是什么?有什么用?有什么语法?有什么功能特性 ...
- 细说并发4:Java 阻塞队列源码分析(上)
上篇文章 趣谈并发3:线程池的使用与执行流程 中我们了解到,线程池中需要使用阻塞队列来保存待执行的任务.这篇文章我们来详细了解下 Java 中的阻塞队列究竟是什么. 读完你将了解: 什么是阻塞队列 七 ...
- KAFKA 0.11 RHEL6.5安装
KAFKA简介 KAFKA是一款分布式消息发布和订阅的系统. 官网:http://kafka.apache.org/ 1.下载KAFKA及JDK KAFKA下载地址: http://kafka.apa ...
- IE11降级到IE8