C语言中结构体指针的定义和引用

    xiaoxiao2025-05-20  11

    这篇文章是九九年写的,这篇文章适合中级程序员。有不明白之处不要紧,多看几遍,然后花些时间上机操作及认真思考每个问题。遇到难题要研究、解决,难题出现于哪里?该用什么方式来解决?为什么要选择这个方式解决?有其它的解决方法吗?这样的解决方案完美吗?其实做个程序员这些基本的思考心得是要掌握的。记住;遇问题不要逃避,要面对现实、勇于挑战,仔细研究难题的所在,这样相信你会成功的!

      指针结构与指针的关系亦有两重:其一是在定义结构时,将指针作为结构中的一个成员;其二是指向结构的指针(称为结构指针)。前者同一般的结构成员一样可直接进行访问,后者是本节讨论的重点。 结构指针说明的一般形式是: struct 结构类型名称 * 结构指针变量名; 例如:struct date * pdate, today; 说明了两个变量,一个是指向结构date的结构指针pdate,today是一个date结构变量。语句: pdate = &today;

    pdate today (struct date)

      year   month   day   

      通过结构变量today访问其成员的操作,也可以用等价的指针形式表示: today.year = 2001;  等价于  (*pdate).year = 2001; 由于运算符"*"的优先级比运算符"."的优先级低,所以必须有"( )"将*pdate括起来。若省去括号,则含义就变成了"*(pdate.year)"。 在C语言中,通过结构指针访问成员可以采用运算符"->"进行操作,对于指向结构的指针,为了访问其成员可以采用下列语句形式: 结构指针->成员名; 这样,上面通过结构指针pdate访问成员year的操作就可以写成: pdate->year = 2001; 如果结构指针p指向一个结构数组,那么对指针p的操作就等价于对数组下标的操作。   结构指针是指向一种结构类型的指针变量,它是结构在内存中的首地址,结构指针具有一般指针的特性,如在一定条件下两个指针可以进行比较,也可以与整数进行加减。但在指针操作时应注意:进行地址运算时的放大因子由所指向的结构的实际大小决定。 例11-7:用结构指针改写加密程序。 #include "stdio.h" struct table { char input, output; } ; struct table translate[ ]= { 'a', 'd', 'b', 'w', 'c', 'k', 'd', ';' , 'e', 'i', 'i', 'a', 'k', 'b', ';', 'c', 'w', 'e' };       /* 建立加密对照表 */ main( ) { char ch; struct table *p, *pend; /* p和pend为指向结构table的指针 */ pend = & translate[ sizeof(translate)/sizeof(struct table)-1 ]; /* pend指向结构数组translate的最后一个元素 */ while ( (ch=getchar( )) != '/n') { for ( p=translate ; p->input!=ch && p!=pend; p++ ) ; if ( p->input==ch ) putchar( p->output); else putchar (ch); } }

    读者可以将两个程序对照阅读,体会结构指针特点。程序中用pend指向数组的最后一个元素。 由于结构指针和在结构中将指针作为成员,使得对于结构变量的运算和对成员的操作变得较为复杂。由于取内容的"*"与"."和"->"运算符的优先级与结合性不同,使得对成员的访问和操作又增加了一层难度,再因为"++"和"--"运算所具有的"先操作"与"后操作"的特性,以及"++"和"--"运算的结合性,使得"++"和--"运算与结构操作混合在一起时,实际操作会更为复杂。

    例11-8:请分析程序的运算结果。 #include "stdio.h" struct s { int x, *y; /* y: 结构中的成员是指向整型的指针 */ } *p; /* p: 指向结构的指针 */ int data[5]={10, 20, 30, 40, 50,}; /* data: 整型数组 */ struct s array[5]= { 100, &data[0], 200, &data[1], 300, &data[2], 400, &data[3], 500, &data[4] }; /* array: 结构数组 */ main ( ) { p=array; /* 指针p指向结构数组的首地址 */ printf ("For printer:/n"); printf ("%d/n", p->x); printf ("%d/n", (*p).x);  printf ("%d/n", *p->y); printf ("%d/n", *(*p).y); printf ("%d/n", ++p->x); printf ("%d/n", (++p)->x); printf ("%d/n", p->x++); printf ("%d/n", p->x);  printf ("%d/n", ++ (*p->y)); printf ("%d/n", ++ * p->y);  printf ("%d/n", * ++ p->y); printf ("%d/n", p->x);  printf ("%d/n", * (++p)->y); printf ("%d/n", p->x);  printf ("%d/n", * p->y ++); printf ("%d/n", p->x);  printf ("%d/n", * (p->y) ++); printf ("%d/n", p->x);  printf ("%d/n", * p ++ ->y); printf ("%d/n", p->x); }

    结构数组array的初始化后的状态如图11.4所示。程序中指针操作的含义如下: p->x /* 取结构指针p指向的结构的成员x的值,输出 100 */ (*p).x /* 取结构指针p的内容的成员x的值,功能同上,输出 100 */ *p->y /* 取结构指针p的指针成员y的内容,输出 10 */ *(*p).y /* 取结构指针p的内容的指针成员y的内容,功能同上,输出10 */ ++p->x /* p所指的x加1,x先加1后再输出 101 ,p不加1 */ (++p)->x /* p先加1后再取x的值,x不加1,输出 200 */ p->x++ /* 先取x的值后x再加1,输出 200 */ p->x /* 输出 201 */ ++(*p->y) /* p所指的y的内容先加1,输出 21 ,p不加1,y也不加1 */ ++ *p->y /* 同上,由运算的结合性隐含了括号,输出 22 */ * ++p->y /* y先加1后再取y的内容,输出30,p不加1,y的内容不加1 */ p->x /* 输出 201 */ *(++p)->y /* p先加1后取所指y的内容,输出 30 */ p->x /* 输出 300 */ *p->y ++ /* 取p所指的y的内容,输出 30,然后p所指的y加1 */ p->x /* 输出 300 */ *(p->y)++ /* 取p所指的y的内容,输出 40,然后p所指的y加1 */ p->x /* 输出 300 */ *p++->y /* 取p所指的y的内容,输出 50,然后p加1 */ p->x /* 输出 400 */

    程序运行结束时,指针与结构数组array的状态如图11-7所示。

    例11-9:可用一个结构表示学生的学号和成绩,编写程序,对班中30名学生按成绩进行排序,并输出排序后的学号、成绩和全班平均分。 #include <stdio.h> #define STNUM 30 /* 全班同学人数 */ struct stuinf { int stid; /* 学生学号 */ int score; /* 学生成绩 */ } stu[STNUM]; /* stu: 结构数组 */ main ( ) { struct stuinf *ptemp, /* ptemp:指向结构的指针, 临时变量 */ *p[STNUM]; /* p:指向结构的指针构成的指针数组 */ int i, j, k, sum=0; /* i,j,k:临时变量;sum:分数累计 */ for (i=0; i<=STNUM-1; i++) /* 输入学生的学号和成绩 */ { scanf ("%d%d", &stu[i].stid, &stu[i].score); /* 输入学生的学号和成绩 */ p[i] = &stu[i]; /* 指针数组p[i]的第i个指针(元素)指向结构数组的第i个元素 */ sum += stu[i].score;      /* 累计学生的分数 */ } for ( i=0; i<=STNUM-2; i++ ) /* 排序操作 */ { k = i; /* k:在第i次循环中,指向当前最高分的指针在指针数组p中的下标 */ for (j=i; j<=STNUM-1; j++) if (p[k]->score < p[j]->score) k=j; /* 查找当前最大值, k中存放最大值对应的指针在指针数组p中的下标 */ if ( k!=i ) /* 当k不等于i时,交换两个指向结构的指针 */ { ptemp = p[i]; p[i] = p[k]; p[k] = ptemp; } } for (i=0; i<=STNUM-1; i++) /* 按排序顺序输出学号和成绩 */ printf("%d,%d/n", (*p[i]).stid, p[i]->score); printf ("average score = %d/n", sum/STNUM); /* 输出平均分 */ }

      程序中使用了较为复杂的数据结构,包括:结构数组stu,指向结构的指针ptemp,由指向结构的指针构成的指针数组p。 程序在结构数组stu和指针数组p之间建立了对应的指针关系,从而为简化后续处理打下了良好的基础。在排序过程中,程序使用选择排序的思想,先查找确定当前的最大值,再进行一次有实效的数据交换。进行数据交换时,也没有交换结构数据本身,而是交换了指向结构数据的指针。在输出时,按照排序后指针的顺序,输出排序后的数据。

     
    转载请注明原文地址: https://ju.6miu.com/read-1299076.html
    最新回复(0)