• 黑马程序员--C语言中的指针(5)


    指针型函数

    前面我们介绍过,所谓函数类型是指函数返回值的类型。 在C语言中允许一个函数的返回值是一个指针(即地址), 这种返回指针值的函数称为指针型函数。 定义指针型函数的一般形式为: 类型说明符 *函数名(形参表) { …… /*函数体*/ } 其中函数名之前加了“*”号表明这是一个指针型函数,即返回值是一个指针。类型说明符表示了返回的指针值所指向的数据类型。 如: int *ap(int x,int y) { ...... /*函数体*/ }    表示ap是一个返回指针值的指针型函数, 它返回的指针指向一个整型变量。下例中定义了一个指针型函数 day_name,它的返回值指向一个字符串。该函数中定义了一个静态指针数组namename 数组初始化赋值为八个字符串,分别表示各个星期名及出错提示。形参n表示与星期名所对应的整数。在主函数中, 把输入的整数i作为实参, 在printf语句中调用day_name函数并把i值传送给形参 nday_name函数中的return语句包含一个条件表达式, 值若大于7或小于1则把name[0] 指针返回主函数输出出错提示字符串“Illegal day”。否则返回主函数输出对应的星期名。主函数中的第7行是个条件语句,其语义是,如输入为负数(i<0)则中止程序运行退出程序。exit是一个库函数,exit(1)表示发生错误后退出程序, exit(0)表示正常退出。

      应该特别注意的是函数指针变量和指针型函数这两者在写法和意义上的区别。如int(*p)()int *p()是两个完全不同的量。int(*p)()是一个变量说明,说明是一个指向函数入口的指针变量,该函数的返回值是整型量,(*p)的两边的括号不能少。int *p() 则不是变量说明而是函数说明,说明p是一个指针型函数,其返回值是一个指向整型量的指针,*p两边没有括号。作为函数说明, 在括号内最好写入形式参数,这样便于与变量说明区别。 对于指针型函数定义,int *p()只是函数头部分,一般还应该有函数体部分。

    main(){

    int i;

    char *day_name(int n);

    printf("input Day No:/n");

    scanf("%d",&i);

    if(i<0) exit(1);

    printf("Day No:%2d-->%s/n",i,day_name(i));

    }

    char *day_name(int n){

    static char *name[]={ "Illegal day", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"};

    return((n<1||n>7) ? name[0] : name[n]); }    

    本程序是通过指针函数,输入一个17之间的整数, 输出对应的星期名。指针数组的说明与使用一个数组的元素值为指针则是指针数组。 指针数组是一组有序的指针的集合。 指针数组的所有元素都必须是具有相同存储类型和指向相同数据类型的指针变量。    指针数组说明的一般形式为: 类型说明符*数组名[数组长度]    其中类型说明符为指针值所指向的变量的类型。例如: int *pa[3] 表示pa是一个指针数组,它有三个数组元素, 每个元素值都是一个指针,指向整型变量。通常可用一个指针数组来指向一个二维数组。 指针数组中的每个元素被赋予二维数组每一行的首地址, 因此也可理解为指向一个一维数组。图6—6表示了这种关系。

    int a[3][3]={1,2,3,4,5,6,7,8,9};

    int *pa[3]={a[0],a[1],a[2]};

    int *p=a[0];

    main(){

    int i;

    for(i=0;i<3;i++)

    printf("%d,%d,%d/n",a[i][2-i],*a[i],*(*(a+i)+i));

    for(i=0;i<3;i++)

    printf("%d,%d,%d/n",*pa[i],p[i],*(p+i)); }  

      本例程序中,pa是一个指针数组,三个元素分别指向二维数组a的各行。然后用循环语句输出指定的数组元素。其中*a[i]表示i0列元素值;*(*(a+i)+i)表示ii列的元素值;*pa[i]表示i0列元素值;由于pa[0]相同,故p[i]表示0i列的值;*(p+i)表示0i列的值。读者可仔细领会元素值的各种不同的表示方法。 应该注意指针数组和二维数组指针变量的区别。 这两者虽然都可用来表示二维数组,但是其表示方法和意义是不同的。

      二维数组指针变量是单个的变量,其一般形式中"(*指针变量名)"两边的括号不可少。而指针数组类型表示的是多个指针一组有序指针)在一般形式中"*指针数组名"两边不能有括号。例如: int (*p)[3];表示一个指向二维数组的指针变量。该二维数组的列数为3或分解为一维数组的长度为3。 int *p[3] 表示p是一个指针数组,有三个下标变量p[0]p[1]p[2]均为指针变量。

      指针数组也常用来表示一组字符串, 这时指针数组的每个元素被赋予一个字符串的首地址。 指向字符串的指针数组的初始化更为简单。例如在例6.20中即采用指针数组来表示一组字符串。 其初始化赋值为: char *name[]={"Illagal day", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"};    完成这个初始化赋值之后,name[0]即指向字符串"Illegal day"name[1]?quot;Monday"......

      指针数组也可以用作函数参数。在本例主函数中,定义了一个指针数组name,并对name 作了初始化赋值。其每个元素都指向一个字符串。然后又以name 作为实参调用指针型函数day name,在调用时把数组名 name 赋予形参变量name,输入的整数i作为第二个实参赋予形参n。在day name函数中定义了两个指针变量pp1pp2pp1被赋予name[0]的值(*name)pp2被赋予name[n]的值即*(name+ n)。由条件表达式决定返回pp1pp2指针给主函数中的指针变量ps。最后输出ips的值。

    指针数组作指针型函数的参数

    main(){

    static char *name[]={ "Illegal day", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"};

    char *ps;

    int i;

    char *day name(char *name[],int n);

    printf("input Day No:/n");

    scanf("%d",&i);

    if(i<0) 

    exit(1);

    ps=day name(name,i);

    printf("Day No:%2d-->%s/n",i,ps);

    }

    char *day name(char *name[],int n) {

    char *pp1,*pp2;

    pp1=*name;

    pp2=*(name+n);

    return((n<1||n>7)? pp1:pp2); }

    下例要求输入5个国名并按字母顺序排列后输出。在以前的例子中采用了普通的排序方法, 逐个比较之后交换字符串的位置。交换字符串的物理位置是通过字符串复制函数完成的。 反复的交换将使程序执行的速度很慢,同时由于各字符串(国名的长度不同,又增加了存储管理的负担。 用指针数组能很好地解决这些问题。把所有的字符串存放在一个数组中, 把这些字符数组的首地址放在一个指针数组中,当需要交换两个字符串时, 只须交换指针数组相应两元素的内容(地址)即可,而不必交换字符串本身。程序中定义了两个函数,一个名为sort完成排序, 其形参为指 针数组name,即为待排序的各字符串数组的指针。形参n为字符串的个数。另一个函数名为print,用于排序后字符串的输出,其形参与sort的形参相同。主函数main中,定义了指针数组name 并作了初始化赋值。然后分别调用sort函数和print函数完成排序和输出。值得说明的是在sort函数中,对两个字符串比较,采用了strcmp 函数,strcmp函数允许参与比较的串以指针方式出现。name[k]name[ j]均为指针,因此是合法的。字符串比较后需要交换时, 只交换指针数组元素的值,而不交换具体的字符串, 这样将大大减少时间的开销,提高了运行效率。 现编程如下:

    #include"string.h"

    main(){

    void sort(char *name[],int n);

    void print(char *name[],int n);

    static char *name[]={ "CHINA","AMERICA","AUSTRALIA", "FRANCE","GERMAN"};

    int n=5;

    sort(name,n);

    print(name,n); }

    void sort(char *name[],int n){

    char *pt;

    int i,j,k;

    for(i=0;i<n-1;i++)

    { k=i;

    for(j=i+1;j<n;j++)

    if(strcmp(name[k],name[j])>0)

     k=j;

    if(k!=i)

    { pt=name[i];

    name[i]=name[k];

    name[k]=pt; }

    }

    }

    void print(char *name[],int n){

    int i;

    for (i=0;i<n;i++)

     printf("%s/n",name[i]); }

    main函数的参数

      前面介绍的main函数都是不带参数的。因此main 后的括号都是空括号。实际上,main函数可以带参数,这个参数可以认为是 main函数的形式参数。C语言规定main函数的参数只能有两个, 习惯上这两个参数写为argcargv。因此,main函数的函数头可写为: main (argc,argv)C语言还规定argc(第一个形参)必须是整型变量,argv( 第二个形参)必须是指向字符串的指针数组。加上形参说明后,main函数的函数头应写为: main (argc,argv) int argv; char *argv[];或写成: main (int argc,char *argv[])    由于main函数不能被其它函数调用, 因此不可能在程序内部取得实际值。那么,在何处把实参值赋予main函数的形参呢实际上,main函数的参数值是从操作系统命令行上获得的。当我们要运行一个可执行文件时,在DOS提示符下键入文件名,再输入实际参数即可把这些实参传送到main的形参中去。

      DOS提示符下命令行的一般形式为: C:/>可执行文件名 参数 参数……; 但是应该特别注意的是,main 的两个形参和命令行中的参数在 位置上不是一一对应的。因为,main的形参只有二个,而命令行中的参数个数原则上未加限制。argc参数表示了命令行中参数的个数(注意:文件名本身也算一个参数)argc的值是在输入命令行时由系统按实际参数的个数自动赋予的。例如有命令行为: C:/>E6 24 BASIC dbase FORTRAN由于文件名E6 24本身也算一个参数,所以共有4个参数,因此argc取得的值为4argv参数是字符串指针数组,其各元素值为命令行中各字符串(参数均按字符串处理)的首地址。 指针数组的长度即为参数个数。数组元素初值由系统自动赋予。其表示如图6.8所示:

    main(int argc,char *argv){

    while(argc-->1)

    printf("%s/n",*++argv);

    }

    本例是显示命令行中输入的参数如果上例的可执行文件名为e24.exe,存放在A驱动器的盘内。 因此输入的命令行为: C:/>a:e24 BASIC dBASE FORTRAN 则运行结果为: BASIC dBASE FORTRAN    该行共有4个参数,执行main时,argc的初值即为4argv4个元素分为4个字符串的首地址。执行while语句,每循环一次 argv值减1,当argv等于1时停止循环,共循环三次, 因此共可输出三个参数。在printf函数中,由于打印项*++argv是先加1再打印, 故第一次打印的是argv[1]所指的字符串BASIC。第二、 三次循环分别打印后二个字符串。而参数e24是文件名,不必输出。

      下例的命令行中有两个参数,第二个参数20即为输入的n值。在程序中*++argv的值为字符串“20”,然后用函数"atoi"把它换为整型作为while语句中的循环控制变量,输出20个偶数。

    #include"stdlib.h"

    main(int argc,char*argv[]){

    int a=0,n;

    n=atoi(*++argv);

    while(n--)

     printf("%d ",a++*2);

    }    

    本程序是从0开始输出n个偶数。指向指针的指针变量如果一个指针变量存放的又是另一个指针变量的地址, 则称这个指针变量为指向指针的指针变量。

      在前面已经介绍过,通过指针访问变量称为间接访问, 简称间访。由于指针变量直接指向变量,所以称为单级间访。 而如果通过指向指针的指针变量来访问变量则构成了二级或多级间访。在C语言程序中,对间访的级数并未明确限制, 但是间访级数太多时不容易理解解,也容易出错,因此,一般很少超过二级间访。 指向指针的指针变量说明的一般形式为: 类型说明符** 指针变量名; 例如: int ** pp; 表示pp是一个指针变量,它指向另一个指针变量, 而这个指针变量指向一个整型量。下面举一个例子来说明这种关系。

    main(){

    int x,*p,**pp;

    x=10;

    p=&x;

    pp=&p;

    printf("x=%d/n",**pp);

    }   

     上例程序中是一个指针变量,指向整型量xpp也是一个指针变量, 它指向指针变量p。通过pp变量访问x的写法是**pp。程序最后输出x的值为10。通过上例,读者可以学习指向指针的指针变量的说明和使用方法。

      下述程序中首先定义说明了指针数组ps并作了初始化赋值。 又说明了pps是一个指向指针的指针变量。在5次循环中, pps 分别取得了ps[0]ps[1]ps[2]ps[3]ps[4]的地址值(如图6.10所示)。再通过这些地址即可找到该字符串。

    main(){

    static char *ps[]={ "BASIC","DBASE","C","FORTRAN", "PASCAL"};

    char **pps; int i;

    for(i=0;i<5;i++){

    pps=ps+i;

    printf("%s/n",*pps);

    } }

    本程序是用指向指针的指针变量编程,输出多个字符串。

  • 相关阅读:
    30天内自动登录
    本地保存cookie
    thymeleaf 基本表达式
    适配器(adapter)与fragment之间、fragment与activity之间的通信问题
    String/Stringbuilder/StringBuffer
    if else与switch for与foreach
    iOS App 签名的原理
    iOS category内部实现原理
    iOS中的事件的产生和传递
    RunLoop
  • 原文地址:https://www.cnblogs.com/bztx-zb/p/4515854.html
Copyright © 2020-2023  润新知