C++中const与#define的利弊分析

C++中const与#define的区别如下:

用#define MAX 255定义的常量是没有类型的,所给出的是一个立即数,编译器只是把所定义的常量值与所定义的常量的名字联系起来,define所定义的宏变量在预处理的时候进行替换,在程序中使用到该常量的地方都要进行拷贝替换;

用const float MAX = 255; 定义的常量有类型名字,存放在内存的静态区域中,在程序运行过程中const变量只有一个拷贝,而#define 所定义的宏变量却有多个拷贝,所以宏定义在程序运行过程中所消耗的内存要比const变量的大得多;

用define定义的常量是不可以用指针变量去指向的,用const定义的常量是可以用指针去指向该常量的地址的;

用define可以定义一些简单的函数,const是不可以定义函数的.

const和#define的利弊,从而推导const的意义;

const和#define都有类似的功能,那就是定义一个“常量”;

想用来替换#define定义常量这种方式。这是一种定义宏的方式。因为宏替换定义常量有一定的缺陷:不做类型检查,没有作用域限制(这样很容易被后续污染)。

#include#includeusingnamespacestd;voidmyfunc1(){#definea 10}voidmyfunc2(){printf("a=%d\n", a);}intmain(){printf("外面打印:a=%d\n", a);myfunc1();myfunc2();system("pause");return0;}

因为只做字面上的直接替换,全局都有效,所以无论定义在哪里,全局都可以访问。因为是在预编译的时候就替换好了(只要有定义,就在预编译的时候进行全程替换,所以外面里面都可以访问)。

同时,很容易受到污染。

#include#includeusingnamespacestd;#definea 10voidmyfunc1(){#definea 20printf("myfunc1里面的:a=%d\n", a);}voidmyfunc2(){printf("myfunc2里面的:a=%d\n", a);}intmain(){printf("外面打印:a=%d\n", a);myfunc1();myfunc2();system("pause");return0;}

提示有宏重定义,结果全部都改变为新的:

宏的方式相当于全局变量,无论在函数里还是函数外命名的时候都要精心雕琢(有点头痛),否则很容易在以后新的函数中不小心被替换掉,这就是为什么用它定义常量都基本上全部大写,而变量都弄成小写,这样既然不记得有多少宏名了,也不至于冲突。但是它的全局性还是没有解决。

而const因为有作用域限制,解决了污染全局变量的困扰。

下面的程序是不行的:

#include#includeusingnamespacestd;voidmyfunc1(){constinta =20;printf("myfunc1里面的:a=%d\n", a);}voidmyfunc2(){printf("myfunc2里面的:a=%d\n", a);}intmain(){printf("外面打印:a=%d\n", a);myfunc1();myfunc2();system("pause");return0;}

定义个全局的只读变量:

#include#includeusingnamespacestd;constinta =10;voidmyfunc1(){constinta =20;printf("myfunc1里面的:a=%d\n", a);}voidmyfunc2(){printf("myfunc2里面的:a=%d\n", a);}intmain(){printf("外面打印:a=%d\n", a);myfunc1();myfunc2();system("pause");return0;}

里面的既不干扰外面的,还可以有优先级之分,同时要做全局也可以做全局。

这样新做的函数中要想使用a这个名字了,不用考虑什么,直接用就是了。不会影响以前外面定义的全局变量a,是不是省事的多啊。

const是只读变量,本质上还是变量,是变量就可以传递参数,而const还做类型检查,所以好处更多,如:做形参,可以接收不同的参数,更灵活。

你不能在里面把我的变量给改了吧,可以传递不同的变量,因此就晓得更灵活了;

#include#includeusingnamespacestd;voidmyfunc1(constintk){printf("myfunc1里面的数据=%d\n", k);}intmain(){constinta =20;myfunc1(a);constintb =30;myfunc1(b);system("pause");return0;}

const的应用:

由于是只读变量,因此保护了外面的实参,外面传递实参进来,在函数体里不能修改。因此让外面的实参得到安全性考虑。

#include#includeusingnamespacestd;voidmyfunc1(constint* k){*k =3;printf("myfunc1里面的数据=%d\n", k);}intmain(){constinta =20;myfunc1(&a);system("pause");return0;}

宏替换的方式相当于弄全局变量,很容易被污染,没有作用域限制,做不了优先级区分。它是在预编译的时候就被替换了。

而const是在编译的时候才分配变量,有作用域区分,和类型一致的安全性检测,应用const来开发项目更方便灵活...

宏替换定义的是常量,必定全局有效;

const定义的是只读变量,有作用域之分,可以做全局的,也可以做局部的,还有优先级之分。既方便又安全,可以代替#define了。那为什么都存在?因为也都有好处,只是想拿各自的好处罢了:

宏替换的方式,让整个编译过程变慢(预编译时间+真正编译的时间),但是让程序运行速度变快,因为早已直接替换好了(宏展开),直接运行就得了。

const和它相反,整个编译时间少,但是程序运行速度慢点了,因为要找内存空间开辟变量...

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

时间: 2018-04-30

c++ 尽量不要使用#define 而是用const、enum、inline替换。

例如:这里程序文件开头有如下#define语句 复制代码 代码如下: #define N 10 #define PI 3.14 #define MAX 10000 #define Heigth 6.65 ... ... 假设这里程序运行出错误,而且就是在我们使用这些常量有错误,此时编辑器应该会抛出错误信息.如果该信息提示6.65这里有错误,Ok如果你运气好你正好记得或者程序简单一眼能找到6.65表示什么,如果程序很复杂,而且报出6.65的文件是引用该文件,不记得,那么你会困惑这是什么?或许会花大

c++ const引用与非const引用介绍

const引用是指向const对象的引用. 复制代码 代码如下: const int i = 10; const int &ref = i; 可以读取ref,但不能修改.这样做是有意义的,因为i本身就不可修改,当然也不能通过ref来修改了.所以也就有将const变量赋值给非const引用是非法的. 复制代码 代码如下: int &ref1 = i; // error: nonconst reference to a const object 非const引用是指向非const类型变量的引用

C/C++中宏定义(#define)

#define是C语言中提供的宏定义命令,其主要目的是为程序员在编程时提供一定的方便,并能在一定程度上提高程序的运行效率,但学生在学习时往往不能 理解该命令的本质,总是在此处产生一些困惑,在编程时误用该命令,使得程序的运行与预期的目的不一致,或者在读别人写的程序时,把运行结果理解错误,这对 C语言的学习很不利. 宏的定义在程序中是非常有用的,但是使用不当,就会给自身造成很大的困扰.通常这种困扰为:宏使用在计算方面. 本例子主要是在宏的计算方面,很多时候,大家都知道定义一个计算的宏,对于编译和编程

C语言编程技巧 关于const和#define的区别心得

#define ASPECT_RATIO 1.653 编译器会永远也看不到ASPECT_RATIO这个符号名,因为在源码进入编译器之前,它会被预处理程序去掉,于是ASPECT_RATIO不会加入到符号列表中.如果涉及到这个常量的代码在编译时报错,就会很令人费解,因为报错信息指的是1.653,而不是ASPECT_RATIO.如果ASPECT_RATIO不是在你自己写的头文件中定义的,你就会奇怪1.653是从哪里来的,甚至会花时间跟踪下去.这个问题也会出现在符号调试器中,因为同样地,你所写的符号名不

C++中的类型转换static_cast、dynamic_cast、const_cast和reinterpret_cast总结

前言 这篇文章总结的是C++中的类型转换,这些小的知识点,有的时候,自己不是很注意,但是在实际开发中确实经常使用的.俗话说的好,不懂自己写的代码的程序员,不是好的程序员:如果一个程序员对于自己写的代码都不懂,只是知道一昧的的去使用,终有一天,你会迷失你自己的. C++中的类型转换分为两种: 1.隐式类型转换: 2.显式类型转换. 而对于隐式变换,就是标准的转换,在很多时候,不经意间就发生了,比如int类型和float类型相加时,int类型就会被隐式的转换位float类型,然后再进行相加运算.而关

C++ 中pragma once 与 #ifndef _XXX_H_ #define _XXX_H_的区别

C++ 中pragma once 与 #ifndef _XXX_H_ #define _XXX_H_的区别 pragma once 之前一直用的很好,今天和同事的配合中发现自己没有真正理解pragma once. 原因:同事喜欢把公共的头文件通过生成后事件复制到一个公共的include文件夹中. 摘抄: #ifndef方式: #ifndef __SOMEFILE_H__ #define __SOMEFILE_H__ 1 ... ... // 一些声明语句 #endif #ifndef的方式依赖于

C++中const的实现机制深入分析

问题 C语言以及C++语言中的const究竟表示什么?其具体的实现机制又是如何实现的呢? 本文将对这两个问题进行一些分析,简单解释const的含义以及实现机制. 问题分析 简单的说const在C语言中表示只读的变量,而在C++语言中表示常量.关于const在C与C++语言中的使用以及更多的区别,以后有时间另开一贴说明. 那么const究竟是如何实现的呢? 对于声明为const的内置类型,例如int,short,long等等,编译器会如何实现const的本意?那么对于非内置类型是否也是与内置数据类

C/C++中的typedef和#define详解

C/C++中的typedef和#define 前言: 在C/C++中,我们平时写程序可能经常会用到typedef关键字和#define宏定义命令,在某些情况下使用它们会达到相同的效果,但是它们是有实质性的区别,一个是C/C++的关键字,一个是C/C++的宏定义命令,typedef用来为一个已有的数据类型起一个别名,而#define是用来定义一个宏定义常量.下面谈谈两者在实际使用中应当注意的地方. 1.typedef关键字 typedef是用来声明类型别名的,在实际编写代码过程使用typedef往

C/C++中static,const,inline三种关键字详细总结

一.关于staticstatic 是C++中很常用的修饰符,它被用来控制变量的存储方式和可见性,下面我将从 static 修饰符的产生原因.作用谈起,全面分析static 修饰符的实质. static 的两大作用: 一.控制存储方式 static被引入以告知编译器,将变量存储在程序的静态存储区而非栈上空间. 引出原因:函数内部定义的变量,在程序执行到它的定义处时,编译器为它在栈上分配空间,大家知道,函数在栈上分配的空间在此函数执行结束时会释放掉,这样就产生了一个问题: 如果想将函数中此变量的值保

java向多线程中传递参数的三种方法详细介绍

在传统的同步开发模式下,当我们调用一个函数时,通过这个函数的参数将数据传入,并通过这个函数的返回值来返回最终的计算结果.但在多线程的异步开发模式下,数据的传递和返回和同步开发模式有很大的区别.由于线程的运行和结束是不可预料的,因此,在传递和返回数据时就无法象函数一样通过函数参数和return语句来返回数据.本文就以上原因介绍了几种用于向线程传递数据的方法,在下一篇文章中将介绍从线程中返回数据的方法. 欲先取之,必先予之.一般在使用线程时都需要有一些初始化数据,然后线程利用这些数据进行加工处理,并

C语言中函数指针的三种使用方法总结

 C语言中函数指针的三种使用方法总结 在这里分享一下自己的心得,希望和大家一起分享技术,如果有什么不足,还请大家指正.写出这篇目的,就是希望大家一起成长,我也相信技术之间没有高低,只有互补,只有分享,才能使彼此更加成长. 定义方式:int (*p)(int x, int y); 实现代码: #include <stdio.h> int sum(int x, int y){ return x + y; } int reduce(int x, int y){ return x - y; } int

关于C++中定义比较函数的三种方法小结

C++编程优与Pascal的原因之一是C++中存在STL(标准模板库).STL存在很多有用的方法. C++模板库中的许多方法都需要相关参数有序,例如Sort().显然,如果你想对一个集合进行排序,你必须要知道集合中的对象,那个在前那个在后.因此,学会如何定义比较方法是非常重要的. C++模板库的许多容器需要相关类型有序,例如set<T> 和priority_queue<T>. 这篇文章旨在告诉大家如何为一个类定义一个排序方法,以便在STL容器或者方法中使用. 作为一个C++程序员,

Java中实现线程的三种方式及对比_动力节点Java学院整理

Java中创建线程主要有三种方式: 一.继承Thread类创建线程类 (1)定义Thread类的子类,并重写该类的run方法,该run方法的方法体就代表了线程要完成的任务.因此把run()方法称为执行体. (2)创建Thread子类的实例,即创建了线程对象. (3)调用线程对象的start()方法来启动该线程. package com.thread; public class FirstThreadTest extends Thread{ int i = 0; //重写run方法,run方法的方

oracle中if/else的三种实现方式详解

1.标准sql规范 1.单个IF IF v=... THEN END IF; 2.IF ... ELSE IF v=... THEN ELSE t....; END IF; 3.多个IF IF v=... THEN ELSIF v=... THEN t...; END IFL 注意: 多个IF的是'ELSIF' 不是 ' ELSE IF' 2.decode函数 DECODE(VALUE,IF1,THEN1,IF2,THEN2,IF2,THEN2,..,ELSE) 表示如果value等于if1时,

Android中使用定时器的三种方法

本文实例为大家分享了Android中使用定时器的三种方法,供大家参考,具体内容如下 图示: 因为都比较简单,所以就直接贴代码(虑去再次点击停止的操作),有个全局的Handler负责接收消息更新UI 第一种方法:Thread.sleep();方法 Runnable runnable = new Runnable() { @Override public void run() { while (true) { mHandler.sendEmptyMessage(0); try { Thread.sl

vue中锚点的三种方法

第一种: router.js中添加 mode: 'history', srcollBehavior(to,from,savedPosition){ if(to.hash){ return { selector:to.hash } } } 组件: <template> <div> <ul class="list"> <li><a href="#1" rel="external nofollow"

Struts 2中实现Ajax的三种方式

Ajax本质上和普通的HTTP请求是一样的,只不过普通的HTTP请求是给人看的,而Ajax请求是给JS代码去用的. 所以Ajax请求的页面一般比普通的HTTP请求的响应内容还要简单,可能是一个页面的一部分,也可能是xml.json等结构化的数据,还可能是一个简单的字符串. 所以,在Struts 2中使用Ajax,Action一般就不会调用一个jsp视图来显示了(如果Ajax请求内容是页面的一部分也可能调用jsp视图),而是通过一些其他的方式. 下面介绍了三种方法,用于Action实现Ajax请求

jQuery中each遍历的三种方法实例分析

本文实例讲述了jQuery中each遍历的三种方法.分享给大家供大家参考,具体如下: 1.选择器+遍历 $('div').each(function (i){ //i就是索引值 //this 表示获取遍历每一个dom对象 }); 2.选择器+遍历 $('div').each(function (index,domEle){ //index就是索引值 //domEle 表示获取遍历每一个dom对象 }); 3.更适用的遍历方法 1)先获取某个集合对象 2)遍历集合对象的每一个元素 var d=$(