C语言字符串指针

来源:互联网 发布:国税务网络学校 编辑:程序博客网 时间:2024/05/21 19:26

原文地址:http://c.biancheng.net/cpp/html/80.html


在C语言中,可以通过字符数组存放一个字符串,也可以用字符指针指向一个字符串。


【示例】用字符数组存放一个字符串,然后输出该字符串。
#include <stdio.h>int main(){    char string[] = "I love China!";    printf("%s\n", string);    return 0;}
和前面介绍的数组属性一样,string是数组名,它代表字符数组的首地址。

【示例】用字符串指针指向一个字符串。
#include <stdio.h>int main(){    char *string = "I love China!";    printf("%s\n", string);    return 0;}
对指向字符变量的指针变量应赋予该字符变量的地址。如:
char c, *p=&c;
表示p是一个指向字符变量c的指针变量。而:
char *s="C Language";
则表示s是一个指向字符串的指针变量。把字符串的首地址赋予s。

上例中,首先定义string是一个字符指针变量,然后把字符串的首地址赋予string(应写出整个字符串,以便编译器把它装入一块连续的内存单元)。程序中的:
char *string = "I love China!";
等效于:
char *string;string = "I love China!";

【示例】输出字符串中n个字符后的所有字符。
#include <stdio.h>int main(){    char *ps = "This is a good tutorial!";    int n=10;    ps=ps+n;    printf("%s\n", ps);    return 0;}
运行结果为:
good tutorial!

在程序中对ps初始化时,把字符串首地址赋予ps,当ps= ps+10之后,ps指向字符'b',因此输出为 "good tutorial!"。

【示例】在输入的字符串中查找有无‘k’字符。
#include <stdio.h>int main(){    char str[20], *ps;    int i;    printf("Input a string: ");    ps = str;    scanf("%s", ps);    for(i=0; ps[i]!='\0'; i++){        if(ps[i]=='k'){            printf("There is a 'k' in the string!\n");            break;        }    }       if(ps[i]=='\0') printf("There is no 'k' in the string\n");    return 0;}
运行结果:
Input a string: thank you
There is a 'k' in the string!

【示例】不使用 strcpy 函数实现字符串的复制。
#include <stdio.h>void cpystr(char *pss, char *pds){    while((*pds=*pss)!='\0'){        pds++;        pss++;    }}int main(){    char *pa = "Apple, Samsung, Xiaomi, Smartisan", b[100], *pb;    pb=b;    cpystr(pa, pb);    printf("string a=%s\nstring b=%s\n",pa, pb);    return 0;}
运行结果:
string a=Apple, Samsung, Xiaomi, Smartisan
string b=Apple, Samsung, Xiaomi, Smartisan

函数 cprstr 的形参为两个字符指针变量,pss 指向源字符串,pds 指向目标字符串。注意表达式(*pds=*pss)!=`\0'的用法。

本例中程序完成了两项工作:一是把pss指向的源字符串复制到pds所指向的目标字符串中,二是判断所复制的字符是否为`\0',若是则表明源字符串结束,不再循环。否则,pds和pss都加1,指向下一字符。

在主函数中,以指针变量pa、pb为实参,分别取得确定值后调用cprstr 函数。由于采用的指针变量pa和pss、pb和pds均指向同一字符串,因此在主函数和cprstr函数中均可使用这些字符串。也可以把cprstr函数简化为以下形式:
void cprstr(char *pss,char*pds){    while( (*pds++=*pss++) != '\0' );}
即把指针的移动和赋值合并在一个语句中。 进一步分析还可发现 `\0' 的ASCII码为0,对于while语句只看表达式的值为非0就循环,为0则结束循环,因此也可省去!='\0'这一判断部分,而写为以下形式:
void cprstr(char *pss,char*pds){    while(*pds++ = *pss++);}

表达式的意义可解释为,源字符向目标字符赋值,移动指针,若所赋值为非 0 则循环,否则结束循环。这样使程序更加简洁。



补充:

 char **  与char  * a[ ] ;

            先看 char  *a [ ] ;

            由于[ ] 的优先级高于* 所以a先和 [ ]结合,他还是一个数组,数组中的元素才是char * ,前面讲到char * 是一个变量,保存的地址。。

            所以 char *a[ ] = {"China","French","America","German"};

            同过这句可以看到, 数组中的元素是字符串,那么sizeof(a) 是多少呢,有人会想到是五个单词的占内存中的全部字节数 6+7+8+7 = 28;

            但是其实sizeof(a) = 16;

            为什么,前面已经说到, 字符串常量的本质是地址,a 数组中的元素为char * 指针,指针变量占四个字节,那么四个元素就是16个字节了

            看一下实例:

            #include <stdio.h>

   int main()
   {
    char *a [ ] = {"China","French","America","German"};
    printf("%p %p %p %p\n",a[0],a[1],a[2],a[3]);

    return 0;
   }

    

      可以看到数组中的四个元素保存了四个内存地址,这四个地址中就代表了四个字符串的首地址,而不是字符串本身。。。

      因此sizeof(a)当然是16了。。

      注意这四个地址是不连续的,它是编译器为"China","French","America","German" 分配的内存空间的地址, 所以,四个地址没有关联。



0 0
原创粉丝点击