《C prime plus (第五版)》 ---第11章 字符串和字符串函数

 11-1:字符串表示和字符串I/O

  1.首先先通过一个整体的例子来初步了解建立,读入和输出字符串的几种方式。

#include<stdio.h>
#define MSG "你一定有很多智慧,告诉我一些吧!"
#define LIM 5
#define LINELEN 81
int main(){
    char name[LINELEN];
    char talents[LINELEN];
    int i ;
    const char m1[40] = "Limit yourself to one line's worth. ";
    const char m2[] = "If you can't think of anything,fake it. ";
    const char *m3 = "\nEnough about me - what's your name? ";
    const char *mytal[LIM] = {
    "Adding numbers swiftly",
    "MUltiplying accurately ",
    "Stashing data",
    "Following instructions to the letter",
    "Understanding the C language"
    };
    printf("你好阿,我是电脑机器人,我有一些智慧\n");
    printf("他们是什么呢?哦,对了,这里有部分清单:");
    for(i = 0 ; i < LIM ; i++){
    puts(mytal[i]);
    }
    puts(m3);
    gets(name);
    printf("哦,%s,%s\n",name,MSG);
    printf("%s\n%s",m1,m2);
    gets(talents);
    puts("Let's see if I've got that list: ");
    puts(talents);
    printf("谢谢你的信息,%s\n",name);
    return 0;
}

  2.定义字符串:基本的方法有字符串常量,char数组,char指针和字符串数组

    1)字符串常量:属于静态存储,是指如果在一个函数中使用字符串常量,即使多次调用这个函数,这个字符串在程序的整个运行过程中只存储一份。整个引号的内容作为指向该字符串存储位置的指针

    例如:

#include<stdio.h>
int main(){
    printf("%s,%p,%c\n","We","are",*"World");
}

  显示结果为:We 0x2321342 W

    2)字符串数组及其初始化

     第一种方法是定义数组的大小,然后给出字符串。

     第二种方法是不给出数组的大小,然后给出字符串,大小由编译器计算。

     第三种方法是如果定义的数组需要在程序运行时才放入字符,那么必须要指明数组的大小。

     第四种方法任何数组名都是数组首元素的地址。m1 == &m1[0]

     第五种方法是用指针符号来建立字符串:例如 const char *m3 = "Hello"   它和char m3[] = "Hello" 作用几乎相同。

     3. 但是也存在区别:数组初始化是从静态存储区把一个字符串复制给数组,而指针初始化只是复制字符串的地址。

#include<stdio.h>
int main(){
    char heart[] = "I love Tillie!";
    char *head = "I love Millie!";
    int i ;
    for(i = 0 ; i < 6 ; i++){
        putchar(heart[i]);
    }
    printf("\n");
    for( i = 0 ; i < 6 ; i++){
        putchar(head[i]);
    }
}

    4.在这个程序中,我们学到的知识: 1)*head =head[0] 2)如果head = heart。那么head就指向了heart字符串,那么原来的Millie还是存在的,但是如果没有保存这个字符串的地址的话,就没有办法再访问这个字符串了,石沉大海了。很难遇到。

    这里需要引入一个问题,char *word = "frame" , word[1] = 'l' => 这样做允许吗?  解答:可能编译器允许这么做,但是会造成内存访问错误。如果编译器使用这种单个拷贝表示法允许把第一个字符改变,那将影响到所有对这个字符串的使用,建议的做法是初始化一个指向字符串文字的指针使用const修饰符, const char *p1 = "Klingon" ;

    5.字符串数组:像最上面的程序中定义了一个指针字符数组。

    这样的一个字符数组在解释上类似于一个二维数组。

    区别是前一个是存放地址的数组,后者是放着完整的字符数组。

 

    最后要建立一种认识,绝大多数字符串使用的都是指针。观察这个程序

#include<stdio.h>
int main(){
    char  *mesg = "Don't be a fool!";
    char  *copy;
    copy = mesg;
    printf("%s",copy);
    printf("mesg = %s,&mesg = %p;value = %p\n",mesg,&mesg,mesg);
    printf("copy = %s,&copy = %p;value = %p\n",copy,&copy,copy

  这个运行结果表明:两个指针变量指向的都是同一个字符串。

11-2:字符串的输入: gets()函数 scanf() 函数  fgets()函数

  注意:在输入字符串的时候,要提前申明一块足够大的区域给它,计算机是不会计算字符串的长度的,后果是会导致输入的字符覆盖程序中其他的数据和代码,导致程序异常终止。

  1.gets()函数:从系统的标准输入设备获得一个字符串,当读到\n的时候,停止输入,舍去\n,补充\0。gets()函数如果成功读取字符串,会返回这个字符串的地址,例如char *ptr , ptr = gets(name);如果出错或者遇到文件结尾,它返回一个空(或0)地址。称为空指针。在<stdio.h>中用NULL来表示。gets(name)!=NULL 

  2.fgets()函数:fgets()函数要求指定最大的字符数,fgets(name,MAX,stdin) ,如果遇到换行符,也会把它读取,而不会扔掉。如果成功读取了字符串,会返回这个字符串的地址。如果出错或者遇到文件结尾,它会返回一个空(或0)地址。

  3.scanf()函数:比较的约束,%s,遇到空白符(空格,制表符,换行符)等都会停止。如果%10s,那么函数就会读取10个或以下停止。如果成功读取了字符,它会返回读取字符的个数,否则返回EOF。

 

  最后总结:gets()函数读取文本更快更好,scanf()函数主要以某种标准输入的混合类型数据的读取和转换。

            

  

posted @ 2015-12-15 14:52  易天曦小盆友  阅读(309)  评论(0编辑  收藏  举报