C 语言中const与指针:三种常见写法辨析

摘要: const int *pint const *p 含义相同,表示「通过指针不能修改所指对象」;int * const p 表示「指针本身不可再指向别处」。本文用经典与拓展示例说明差异,并给出工程实践建议。

关键词: C 语言、const、指针、只读视图、常量指针

一、核心结论(速查)

写法 一句话含义
const int *grape 指向只读 int 的指针 :不能通过 grape 修改那个 int,但 grape 可以指向别的地址。
int const *grape 与上一行完全相同 ,仅 const 与类型名的书写顺序不同。
int * const grape_jelly 常量指针,指向可改 intgrape_jelly 不能再指向别处,但可以通过它修改指向的 int

记忆口诀(看 *const 的相对位置):

  • const* 左边 → 数据只读;const* 右边 → 指针只读

二、const int *int const *(等价)

2.1 语义

指针的类型是「指向 const int」,即 通过该指针不能修改所指内存中的 int

这不代表「内存里一定是常量」------对象本身可能是普通变量,只是 当前这个指针通道不允许写

2.2 经典例子:只读函数参数

标准库中大量接口用此约定表达「只读缓冲区」:

c 复制代码
size_t strlen(const char *s);  /* 实现中承诺不通过 s 修改字符串 */

void print_values(const int *p, size_t n)
{
    size_t i = 0;

    for (i = 0; i < n; i++) {
        printf("%d ", p[i]);   /* 可以读 */
        /* p[i] = 0; */         /* 编译错误:不能通过 p 写 */
    }
}

2.3 经典例子:指针可改指向,不可改目标

c 复制代码
void example_read_only_view(void)
{
    int a = 1;
    int b = 2;
    const int *grape = &a;

    /* *grape = 10; */   /* 错误:不能通过 grape 修改所指 int */
    grape = &b;          /* 正确:grape 本身不是 const 指针,可以改指向 */
}

2.4 拓展:const int * 指向非 const 变量

c 复制代码
void example_alias_read_only(void)
{
    int x = 5;
    const int *p = &x;   /* 合法:用「只读视图」观察 x */

    /* *p = 1; */        /* 错误:不能通过 p 写 */
    x = 1;               /* 合法:仍可直接改 x(不经过 p) */
}

2.5 拓展:两种写法等价

c 复制代码
const int *p1;
int const *p2;   /* 与 p1 类型相同 */

团队可统一一种风格(例如始终写 const int *),避免无谓争论。

三、int * const(常量指针)

3.1 语义

指针变量本身是常量 :定义后不能再让该指针指向别的地址(必须在定义时给出合法初值,除非在特殊上下文如函数参数中另有规则)。
仍可通过该指针修改所指的 int

3.2 经典例子:固定指向某缓冲区

c 复制代码
void example_const_pointer(void)
{
    int buffer[4] = {1, 2, 3, 4};
    int * const grape_jelly = buffer;  /* 始终指向 buffer 首元素 */

    grape_jelly[0] = 99;   /* 正确:修改的是 int 对象 */
    *grape_jelly = 100;    /* 同上 */

    /* grape_jelly++; */              /* 错误:不能移动指针 */
    /* grape_jelly = &buffer[1]; */   /* 错误:不能改指针的值 */
}

3.3 经典例子:函数内锁定指针形参(实现细节)

c 复制代码
void bump_first(int * const p)
{
    if (p == NULL) {
        return;
    }
    (*p)++;            /* 可以修改指向的 int */
    /* p++; */         /* 错误:不能修改 p 本身 */
}

3.4 拓展:与数组名的类比

在多数表达式中,数组名会退化为指向首元素的指针,但数组名不是可赋值的左值:

c 复制代码
void example_array_name(void)
{
    int arr[3] = {0, 0, 0};

    /* arr = NULL; */   /* 错误:不能对数组名整体赋值 */
}

思路上与「指针不可改指向」有相似之处,但数组名与 int * const 在标准中的规则并不完全相同,类比有助于直觉理解。

四、三种组合对照表

声明 能否改指针指向的地址 能否通过指针改所指 int
const int *p 不能
int * const p 不能
const int * const p 不能 不能

拓展:const int * const

c 复制代码
void example_both_const(void)
{
    const int x = 42;
    const int * const grape = &x;

    /* grape = &x; */     /* 错误:指针常量 */
    /* *grape = 0; */     /* 错误:指向 const int */
}

五、如何读复杂声明

  1. 螺旋法则 / 先右后左 :从标识符出发,向右读,遇到 ) 或边界再向左读,交替进行。
  2. 简化规则const 默认修饰其左侧紧邻的类型;若左侧没有可修饰成分,则修饰右侧。

掌握后可以继续分析 const int **int * const * 等多级指针,原则是 逐层、逐个 const* 分析

相关推荐
SuperByteMaster14 分钟前
keil 工程 .gitignore配置文件
c语言
老花眼猫4 小时前
编制椭圆旋转绘图函数
c语言·经验分享·青少年编程·课程设计
iCxhust7 小时前
微机原理实践教程(C语言篇)---A002流水灯
c语言·开发语言·单片机·嵌入式硬件·51单片机·课程设计·微机原理
qeen877 小时前
【数据结构】建堆的时间复杂度讨论与TOP-K问题
c语言·数据结构·c++·学习·
handler018 小时前
Linux 内核剖析:进程优先级、上下文切换与 O(1) 调度算法
linux·运维·c语言·开发语言·c++·笔记·算法
热心网友俣先生8 小时前
2026年第二十三届五一数学建模竞赛C题超详细解题思路+各问题可用模型推荐+部分模型结果展示
c语言·开发语言·数学建模
li1670902709 小时前
第二十七章:智能指针
c语言·数据结构·c++·visual studio
Aurorar0rua11 小时前
CS50 x 2024 Notes C - 07
c语言·学习方法
爱编码的小八嘎11 小时前
C语言完美演绎9-15
c语言
weixin_4217252612 小时前
C语言常用字符串函数:长度、比较、拼接和查找
c语言·字符串函数·查找·比较·长度