“指向字符数据类型的指针如何能够保存字符串数据类型?”好吧,在 C 语言中,键入 'pointer to char' is 是字符串类型,这在一定程度上是正确的。任何对字符串(包括printf)进行操作的函数都会通过char * 类型的参数来接受这些字符串。
"printf() 如何在不使用明显引号的情况下打印字符串?"没有规则说你需要引号才能有一个字符串!带引号的那个东西是 string constant 或 string literal,这是将字符串放入程序的一种方法,但它根本不是唯一的方法。有很多方法可以构造(以及操作和修改)根本不涉及任何引号的字符串。
让我们画一些代表你的代码的图片:
char *p;
p 是指向char 的指针,但正如您正确指出的那样,它还没有指向任何地方。我们可以像这样用图形表示它:
+-----------+
p: | ??? |
+-----------+
接下来将p 设置为指向某处:
p = "%d";
这会将字符串 "%d" 分配到某处(无论在哪里),并设置 p 指向它:
+---+---+---+
| % | d |\0 |
+---+---+---+
^
|
\
\
\
|
+-----|-----+
p: | * |
+-----------+
接下来,您开始递增p:
p++;
正如你所说,这使得p 指向过去的位置,指向字符串的第二个字符:
+---+---+---+
| % | d |\0 |
+---+---+---+
^
|
|
|
|
|
+-----|-----+
p: | * |
+-----------+
接下来,
p++;
现在我们有了:
+---+---+---+
| % | d |\0 |
+---+---+---+
^
|
/
/
/
|
+-----|-----+
p: | * |
+-----------+
接下来你打电话给printf,但有点奇怪:
printf(p-2,23);
关键是表达式p-2。如果p指向字符串中的第三个字符,那么p-2指向字符串中的第一个字符:
+---+---+---+
| % | d |\0 |
+---+---+---+
^ ^
+----|----+ |
p-2: | * | /
+---------+/
/
|
+-----|-----+
p: | * |
+-----------+
而那个指针p-2 或多或少与printf 收到的指针相同,如果您更习惯地称为printf("%d", 23)。
现在,如果您认为 printf 收到了一个字符串,那么您可能会惊讶地听到 printf 很高兴收到一个 char * 而实际上它总是收到一个char *。如果这令人惊讶,问问你自己,如果不是指向char 的指针,你让printf 收到了什么?
严格来说,C 中的字符串是一个字符数组(以'\0' 字符结尾)。但是有一个关于 C 的超级重要的秘密事实,如果你还没有遇到过,你很快就会发现(因为它根本不是秘密):
在 C 中你不能用数组做很多事情。每当你在 C 中的表达式中提到一个数组,每当你看起来想对数组的值做一些事情时,你得到的是一个指向数组的第一个元素。
该指针几乎就是数组的“值”。由于指针算法的工作方式,您可以使用指针非常透明地访问数组(几乎就像指针 是 数组一样,但当然不是)。这一切都非常适用于字符数组(和指针)。
所以由于 C 中的字符串是一个字符数组,所以当你编写时
"%d"
这是一个由三个字符组成的数组。但是当你在表达式中使用它时,你得到的是一个指向数组第一个元素的指针。例如,如果你写
printf("%d", 23);
你有一个字符数组,你在一个表达式中提到它,所以你得到的是一个指向数组第一个元素的指针,这就是传递给printf的东西。
如果我们说
char *p = "%d";
printf(p, 23);
我们做了同样的事情,只是更明确一点:再一次,我们在表达式中提到了数组"%d",所以我们得到的值是指向它的第一个元素的指针,所以这就是用于初始化指针变量p 的指针,这是作为第一个参数传递给printf 的指针,所以printf 很高兴。
在上面,我说过“在 C 语言中,'pointer to char' 类型是字符串类型,这在一定程度上是正确的”。后来我说“C 中的字符串是字符数组”。那么它是哪一个?数组还是指针?严格来说,字符串是一个字符数组。但是像所有数组一样,我们不能对字符数组做很多事情,当我们尝试时,我们得到的是指向第一个元素的指针。所以大多数时候,C 中的字符串是通过指向字符的指针来访问、操作和修改的。所有对字符串进行操作的函数(包括printf)实际上都会接收指向char 的指针,指向它们将要操作的字符串。