zoukankan      html  css  js  c++  java
  • c++ const用法小结

    const用法

    1,定义全局变量的内存分配问题

    #define  Pi_1  3.14       //使用#define宏

    const double Pi_2 = 3.14    //使用const,这时候Pi并没有放入内存中

    double  a = Pi_2;  //这时候才为Pi分配内存,不过后面再有这样的定义也不会再分配内存

    double  b = Pi_1;  //编译时分配内存

    double  c = Pi_2;  //不会再分配内存,

    double  d = Pi_1;  //编译时再分配内存

    const定义的变量,系统只为它分配一次内存,而使用#define定义的常量宏,能分配好多次,这样const就很节约空间

    2,const修饰变量

    1、限定符声明变量只能被读

       const int i=5;

       int j=0;

       ……

       i=j;  //非法,导致编译错误

       j=i;  //合法

    2、 必须初始化

       const int i=5;    //合法

       const int j;     //非法,导致编译错误

    3、在另一连接文件中引用const常量

       extern const int i;    //合法

       extern const int j=10;   //非法,常量不可以被再次赋值

    3,指向常量的指针变量

    int a = 10;

    int b = 20;

    const int* p = &a;    // assignment of read-only location ‘* p’, 即*p为只读的,不能对*p进行写操作。

    int const* p1 = &a;   //同上

    *p = 11;            // error

    p = &b;            // ok, 此时 *p = 20;

    4,常量指针

    int * const p2 = &a;  //assignment of read-only variable ‘p2’, 即p2始终指向&a,不可被改变,但是可对*p2 做读写操作。

    p2 = &b;           // error

    *p2 = b;            // ok , 此时 *p2 = 20,由于p2始终指向&a,所以此时 a = 20;

    5,指向常量的常量指针

    const int * const p3 = &a; //此时p3始终指向 &a,并且不能对*p3做“写”操作。但是可以通过 a = 12 去改变*p3的值。

    6,const限定类的成员函数

    这种用法只在C++中有用(C语言中没有成员函数).如:

           class classname

           {

               public:

                   int func() const

                   {... ...;}

           }

    采用这种const后置的形式一种规定,为了不引起混淆,在此函数的声明中和定义中均要使用const,因为const已经成为类型信息的一部分.但要注意,这种情况下,该函数不能修改类的非静态(static)数据成员,也不能在函数中调用其它非const的函数.比如:

           class base

           {

               int x;

               static int y;

               public:

                  void foo() { cout << “foo” << endl;}

    void foo() const { cout << “foo const” << endl;}

                  void foo1() { cout << “foo1” << endl;}

    void foo2() const { cout << “foo2 const” << endl;}

    static void foo3() { cout << “foo3” << endl;}

                  void test() const

                  {

                      int c;

                      x = 5;   //错误

                      y = 10;  //正确

                      foo()    //正确

                      foo1();  //错误

                      foo2();  //正确

                      foo3()   //正确

                      c = 10;  //正确

                  }

           }

    这种情况下,在函数内部无法改变成员数据x的值,因此编译器会报错:"l-value secifies const object".

    尽管函数名和参数列表都相同,void foo( ) const成员函数是可以与void foo( )并存的,可以形成重载! 我们假设调用语句为obj.foo(),如果obj为non-const对象,则调用foo()。如果obj为const对象,则调用foo()const。另外要注意,假如没有提供foo()const,则const obj调用foo()将会报错。但假如是没有提供foo(),则non-const obj调用foo()const是完全没有问题的。也就是说,non-const对象可以调用const函数(当然也可以调用non-const函数),但const对象不能调用non-const函数。

    const关键字所起作用的本质,就是把隐藏着的默认的this指针参数,改成const类型。也就是说:假如void foo( )函数被编译器改写为 void foo(T* pThis),则void foo( ) const将会被改写为void foo(const T* pThis) 。i.e. 在函数末尾添加一个const,就相当于在隐藏的this参数类型前加一个const.

    这样做有两个效果,第一:编译器将不允许foo()const修改pThis指向的对象的成员。第二、const对象只能调用const成员函数,否则就会报错说把const T* 转化为T* 会丢失qualifier

  • 相关阅读:
    vue.js 第二课
    vue.js学习(第一课)
    2016-11-14看张大神的微博总结
    这几天的工作总结:
    调了一天的兼容总结下
    鸭式辩论
    prototype 原型
    前端ps常用的小技巧
    Android的开始之相对布局
    Android的开始之线性布局
  • 原文地址:https://www.cnblogs.com/lxd2502/p/4390044.html
Copyright © 2011-2022 走看看