2022年

2022年发布的文章
  • C语言指针常量和指向常量的指针

    指针常量本质是一个常量,而用指针修饰它,那么说明这个常量的值应该是一个指针。指针常量 的值是指针,这个值因为是常量,所以不能被赋值。因为指针常量是一个常量,在声明的时候一定要给它赋初值。一旦赋值,以后这个常量再也不能指向别的地址。下面的例子展示了指针常量不同于指向常量的指针:

    int var;                        // 一个int类型的对象
    int *const c_ptr = &var;     // 一个int类型指针常量
    *c_ptr = 123;            // 合法:我们可以修改它所引用的对象
    ++c_ptr;                         // 错误:我们不能修改指针

    当一个指针指向具有 const 限定的对象,称为指向常量的指针,可以修改该指针的值。然而,只能使用这样的指针来读取所指向的对象,但不能修改所指向的对象。因此,指向常量的指针常常被称为只读指针(read-only pointer)。所引用对象本身可以是常量,也可以不是常量。例子如下:

    int var;                      //一个int类型的对象
    const int c_var = 100,   // 一个int类型的常量对象
              *ptr_to_const;         // 一个指向常量的指针:指针本身不是常量!
    ptr_to_const = &c_var;       // 合法:使得ptr_to_const指向c_var
    var = 2 * *ptr_to_const;         // 合法:等效于var = 2 * c_var
    ptr_to_const = &var;         // 合法:使得ptr_to_const指向var
    if ( c_var < *ptr_to_const )  // 合法:“只读”方式获取
      *ptr_to_const = 77;    // 错误:我们不能使用ptr_to_const修改var,
                                             // 尽管var不是常量

    类型修饰符和类型限定符可以以任何顺序排列。因此,下面的写法是合法的:

    int const c_var = 100, *ptr_to_const;

    赋值表达式 ptr_to_const=&var 必需采用隐式转换:int 指针值 &var 会自动地转换成左操作数的类型,也就是指向 const int 的指针。对于与之类似的类型操作数的运算符,编译器会隐式地将指向某一类型T的指针,转换为具有更多限定符的类型 T 的指针。

    如果想将一个指针转换为有较少限定符的类型,必须使用显式的类型转换。下面的程序代码片段使用前面例子所声明的变量:

    int *ptr = &var;             // 一个指向var的int指针
    *ptr = 77;                      // 合法:ptr 不是一个只读指针
    ptr_to_const = ptr;             // 合法:隐式地将ptr从指向int的指针
                                            // 转换为指向int常量的指针
    *ptr_to_const = 77;             // 错误:不能通过一个只读指针修改一个变量
    ptr = &c_var;               // 错误:不能隐式地将指向int常量的指针转换
                                            // 为指向int 的指针
    ptr = (int *)&c_var;        // 合法:显式的指针类型转换总是可行的
    *ptr = 200;                     // 尝试修改c_var:可能会造成运行错误

    如果编译器将常量对象 c_var 放置在内存中的只读区域,那么最后一条语句将会引发运行错误。

    也可以声明指向常量的常量指针,如下面函数原型的参数声明:

    void func( const int * const c_ptr_to_const );

    该函数的参数是只读指针,当该函数被调用时该只读指针参数会被初始化,并且在函数执行过程中保持该值不变。

更多...

加载中...