const char*, char const*, char*const的区别

本文详细解析了C++中constchar*, charconst*, char*const等指针类型的区别及其使用场景,通过实例演示了如何正确理解和运用这些指针类型。

const char*, char const*, char*const的区别问题几乎是C++面试中每次都会有的题目。

事实上这个概念谁都有,只是三种声明方式非常相似很容易记混。
Bjarne在他的The C++ Programming Language里面给出过一个助记的方法:
把一个声明从右向左读。

char * const cp; ( * 读成 pointer to )
cp is a const pointer to char

const char * p;
p is a pointer to const char;

char const * p;
同上因为C++里面没有const*的运算符,所以const只能属于前面的类型。


C++标准规定,const关键字放在类型或变量名之前等价的。

const int n=5;    //same as below
int const m=10;

 

const int *p;    //same as below  const (int) * p
int const *q;    // (int) const *p

 

char ** p1; 
//    pointer to    pointer to    char 
const char **p2;
//    pointer to    pointer to const char 
char * const * p3;
//    pointer to const pointer to    char 
const char * const * p4;
//    pointer to const pointer to const char 
char ** const p5;
// const pointer to    pointer to    char 
const char ** const p6;
// const pointer to    pointer to const char 
char * const * const p7;
// const pointer to const pointer to    char 
const char * const * const p8;
// const pointer to const pointer to const char

 

说到这里,我们可以看一道以前Google的笔试题:

[题目]const char *p="hello";
       foo(&p);//函数foo(const char **pp)
下面说法正确的是[]
A.函数foo()不能改变p指向的字符串内容
B.函数foo()不能使指针p指向malloc生成的地址
C.函数foo()可以使p指向新的字符串常量
D.函数foo()可以把p赋值为 NULL.

至于这道题的答案是众说纷纭。针对上面这道题,我们可以用下面的程序测试

 

#include <stdio.h>
#include <stdlib.h>
#include <stdio.h>


void foo(const char **pp)
{
//    *pp=NULL;
//    *pp="Hello world!";
        *pp = (char *) malloc(10);
        snprintf(*pp, 10, "hi google!");
//       (*pp)[1] = 'x';

}

int
main()
{
    const char *p="hello";
    printf("before foo %s/n",p);
    foo(&p);
    printf("after foo %s/n",p);
    p[1] = 'x';

    return;
}



结论如下:

 

  1. 在foo函数中,可以使main函数中p指向的新的字符串常量。
  2. 在foo函数中,可以使main函数中的p指向NULL。
  3. 在foo函数中,可以使main函数中的p指向由malloc生成的内存块,并可以在main中用free释放,但是会有警告。但是注意,即使在foo中让p指向了由malloc生成的内存块,但是仍旧不能用p[1]='x';这样的语句改变p指向的内容。
  4. 在foo中,不能用(*pp)[1]='x';这样的语句改变p的内容。

      所以,感觉gcc只是根据const的字面的意思对其作了限制,即对于const char*p这样的指针,不管后来p实际指向malloc的内存或者常量的内存,均不能用p[1]='x'这样的语句改变其内容。但是很奇怪,在foo里面,对p指向malloc的内存后,可以用snprintf之类的函数修改其内容。

你给出的两个变量: ```c const char *username, const char *password ``` 是两个**指向常量字符的指针**,它们通常用于表示**字符串常量**,例如用户名密码。我们来详细解释这两个变量的含义、作用以及使用场景。 --- ## ✅ 1. `const char *username` 是什么意思? 这是一个**指向常量字符的指针(pointer to const char**,表示: - `username` 是一个指针,它指向一个字符(char)。 - 由于前面加了 `const`,所以**指针所指向的内容是不可修改的**。 - 你可以让指针指向不同的地址(即指向不同的字符串),但不能通过这个指针修改字符串内容。 ### 示例: ```c const char *username = "admin"; printf("%s\n", username); // 输出:admin // username[0] = 'A'; // 错误:不能修改常量字符串 username = "newuser"; // 合法:可以改变指针指向 ``` --- ## ✅ 2. `const char *password` 同理 ```c const char *password = "123456"; printf("%s\n", password); // 输出:123456 // password[0] = 'x'; // 错误 password = "newpass"; // 合法 ``` --- ## ✅ 3. 为什么使用 `const char *`? 在函数参数或变量中使用 `const char *` 有以下优点: | 优点 | 说明 | |------|------| | **安全性** | 防止函数意外修改传入的字符串内容 | | **可读性** | 明确表示该字符串是只读的 | | **兼容性** | 可以接受字符串字面量(如 `"hello"`),而非常量指针则不行 | | **优化** | 编译器可以对常量字符串进行优化,例如放入只读内存段 | --- ## ✅ 4. 常见使用场景 ### 场景一:函数参数中使用 ```c void login(const char *username, const char *password) { printf("Username: %s\n", username); printf("Password: %s\n", password); // 不会修改用户名密码 } ``` ### 场景二:主函数中传参 ```c int main() { const char *username = "admin"; const char *password = "secret"; login(username, password); return 0; } ``` ### 场景三:字符串常量池中使用 ```c const char *greetings[] = { "Hello, admin!", "Welcome back.", "Invalid username or password." }; ``` --- ## ✅ 5. 注意事项 - `const char *` 指向的内容不可修改,但指针本身可以指向别的字符串。 - 字符串字面量如 `"admin"` 是存储在**只读内存区域**中的,试图修改会引发**未定义行为**。 - 如果你确实需要修改字符串内容,请使用字符数组: ```c char password[] = "123456"; // 可以修改 password[0] = 'x'; // 合法 ``` --- ## ✅ 示例代码:完整演示 ```c #include <stdio.h> void login(const char *username, const char *password) { printf("Logging in with:\n"); printf("Username: %s\n", username); printf("Password: %s\n", password); } int main() { const char *user = "admin"; const char *pass = "123456"; login(user, pass); // user = "newuser"; // 合法 // pass[0] = 'x'; // 非法:不能修改 const 字符串 return 0; } ``` ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值