结构体

注:结构体类型的名字是由一个关键字struct和结构体名组合而成的(例如struct Student)。结构体名是由用户指定的,又称“结构体标记”,以区别于其他结构体类型。上面的结构体声明中Student就是结构体名(结构体标记)

说明
struct 结构体名
{成员列表}
类型名 成员名;
“成员列表”也称为“域表”

struct Student
{int num;char name[20];char sex;int age;float score;
}

注:

  1. 如果成员本身又属于一个结构体类型,则要用若干成员运算符,一级一级找到最低级的成员。只能对最低级的成员进行赋值或存取以及运算。
  2. 同类的结构体变量可以相互赋值
    student1 = student2
  3. 可以引用结构体变量成员的地址,也可以引用结构体变量的地址
scanf("%d",&student1.num);
printf("%o",&student1);

声明和定义结构体类型变量

1. 声明和定义分开

声明结构体类型
sturct Student student1,student2;
定义结构体

student1.num = 10001;
student1.name = "zhangxin"; //此处如此赋值是不正确的,请自行百度如何正确赋值char数组。
student1.sex = "M";
student1.age = 19;
student1.score = 90.5; student2.num = 10002;
student2.name = "wangli"; //此处如此赋值是不正确的,请自行百度如何正确赋值char数组。
student2.sex = "F";
student2.age = 20;
student2.score = 98;

2. 声明和定义同时进行

struct Student
{int num;char name[20];char sex;int age;float score;
}student1,student2;

注:该定义方法的一般形式为

struct 结构体名
{成员列表
}变量名列表;

3. 不指定类型名而直接定义

struct
{成员表列
}变量名表列;

说明:

  • 只能对结构体变量赋值而不能对结构体类型赋值。在编译时,结构体类是不分配空间的,只对变量分配空间。
  • 结构体类型中的成员可以与程序中的变量名相同,但二者不代表同一对象。例如,程序中可以另定义一个变量num,它与struct Student中的num是两回事,相互不干扰。

结构体变量的初始化和引用

【例9.1】把一个学生的信息(包括学号、姓名、性别)放到同一个结构体变量中,并输出这个学生的信息

#include<stdio.h>
int main()
{struct Student{int num;char name[20];char sex;int age;// float score;}a={10101,"Li Lin",'M'};printf("学号:%d,姓名:%d,性别:%d。\n",a.num,a.name,a.sex,a.age);return 0;
}

定义结构体数组

  1. sturct 结构体名
    {成员列表} 数组名【数组长度】;

    struct Person
    {char name[20];int count;
    }leader[3];
    
  2. 结构体类型 数组名【数组长度】;
    struct Person leader[3];
    

定义结构体数组
【例9.3】有三个候选人,每个选民只能投票选一人,要求编一个统计选票的程序,先输入被选人的名字,最后输出各人得票结果。

#include<string.h>
#include<stdio.h>
struct Person()
{char name[30];int count;
}leader[3]={"Li",0,"zhang",0,"sun",0};int main()
{int i,j;char leader_name[20];for(i=1;i<=10;i++){scanf("%s",leader_name);for(j=0;j<3;j++){if(strcmp(leader_name,leader[j].name==0))leader[j].count++;}}printf("\nResult:\n");for(i=0;i<3;i++)printf("%5s:%d\n",leader[i].name,leader[i].count);return 0;
}

指向结构体变量的指针

struct Student *pt

【例9.5】通过指向结构体变量的指针变量输出结构体变量中的成员信息

#include<stdio.h>
#include<string.h>
int main()
{struct Student{long num;char name[20];char sex;float score;};struct Student stu_1;return 0;
}

注:以下三种等价

  1. stu成员名(如stu.num)
  2. (*p).成员名(如( *p).num)
  3. p->成员名(如p->num)

指向结构体的指针

【例9.6】有3个学生的信息,放在结构体数组中,要求输出全部学生的信息

#include<stdio.h>
struct Student
{int num;char name[20];char sex;int age;
};
struct Student stu[3]={{10101,"Li",'M',18},
{10102,"zhangfang",'M',19},{10104,"wangmin",'F',20}};
int main()
{struct Student *p;printf("No.     Name        sex age\n");for(p=stu;p<stu+3;p++>){printf("%5d %-20s %2c %4d\n",p->num,p->name,p->sex,p->age);}return 0;
}

注:

  1. 如果p的初值为stu,即指向stu的序号为0的元素,p加1后,p就指向下一个元素。例如:
    (++p)->num //先使p自加1,然后得到p指向的元素中num成员值(即10102)
    (p++)->num//先求得p->num的值(即10101),然后再使得p自加1,指向stu[]
  2. p指针用来指向一个struct Student类型对象,不应用指向stu数组元素中的某一成员。
    p = stu[1].name;//不合法,stu[1].name是stu[1]元素中成员name首字符的地址。编译时将会给出“警告”信息,表示地址的类型不匹配。
    
  3. 如果将某一个成员地址赋值给p,例如p = (struct Student*) stu[0].name; 此时,p的值是stu[0].name成员的起始地址。但是p仍然保留原来的类型。如果执行printf("%s",p+1);,则会输出stu[1]中name的值。执行p++时,p的值的增量是结构体struct Student的长度。

用结构体变量和结构体变量的指针做函数参数

struct Student
{int num;char name[20];float score[3];float aver;
}

1. 结构体变量做函数参数

void input(struct Student stu);

2. 结构体变量的指针/数组做函数参数

void input(struct Student stu[]);

指针处理链表

注:C程序设计 谭浩强版此处有9.4节“用指针处理链表”本人将其归类到了第八章指针中

共用体

格式:
union 共用体名
{成员表列}变量表列;
例如

union Data
{int i;char ch;float f;
}a,b,c;

注:结构体变量所占内存长度是各成员占的内存长度之和。每个成员分别占有其自己的内存单元。而共用体变量所占的内存长度等于最长成员的长度。

引用共用体变量

a.i
a.ch
a.f

特点

  1. 同一个内存段可以用来存放几种不同类型的成员,但在每一瞬时只能存放其中一个成员,而不是同时存放几个。

    union Date
    {int i;char ch;float f;
    }a;
    a.i = 97;
    printf("%d",a.i);   //输出97
    printf("%c",a.ch);  //输出a
    printf("%f",a.f);   //输出实数0.00000
    

    注:整型无法用浮点型的格式输出,故printf("%f",a.f);输出为0

  2. 对共用体量化,但是初始化表只能有一个常量

    union Date{int i;char ch;float f;}a = {1,‘a’,1.5};   //此处错误
    
  3. 共用体变量中起作用的成员是最后一次被赋值的成员,在对共用体变量中的一个成员赋值后,原有的变量存储单元中的值就被取代了。

  4. 共用体变量的地址和它的成员的地址是相同的。

  5. 不能对共用体变量名赋值,也不能企图引用变量名来得到一个值。

    a = 1;
    m = a;
    
  6. C99之前不允许共用体变量作为函数参数,只能通过指针的形式做函数参数,C99之后允许了。

注:共用体类型一般用在两个组数据的成员大部分相同的情况之下。例如,有学生和老师两组数据,学生的成员包括:姓名、号码、性别、职业、班级。教师的成员包括:姓名、号码、性别、职业、职务。这两组数据只有班级和职务不同,因此可以定义成如下的共用体:

struct
{char name[20];char sex;char job;union{int class;char position[10];}category;
}person[2];

枚举类型

注:枚举类型放在了第3章“顺序程序设计”

typedef声明新类型名

注:typedef声明新类型名放在了第3章“顺序程序设计”

习题

1.定义一个结构体变量(包括年、月、日)。计算该日在本年中是第几天,注意闰年问题。

代码思路:

闰年:

①:可以被4整除但是不能被100整除

②:可以被400整除

#include <stdio.h>int main() {int year, month, day, d;int months[] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};printf("请输入年、月和日:");fflush(stdout);scanf("%d %d %d", &year, &month, &day);d = 0;for (int i = 1; i < month; ++i) {d += months[i];}d += day;int flag = (year%4 == 0 && year%100 != 0) || (year%400 == 0);if(flag && month>= 3) {d++;}printf("这是%d年的第%d天\n", year, d);return 0;
}

2.写一个函数days,实现第1 题的计算。由主函数将年、月、日传递给days函数,计算后将日子数传回主函数输出。

#include <stdio.h>int main() {int days(int, int, int);int year, month, day;printf("请输入年、月和日:");fflush(stdout);scanf("%d %d %d", &year, &month, &day);printf("这是%d年的第%d天\n", year, days(year, month, day));return 0;
}int days(int year, int month, int day) {int d;int months[] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};d = 0;for (int i = 1; i < month; ++i) {d += months[i];}d += day;int flag = (year%4 == 0 && year%100 != 0) || (year%400 == 0);if(flag && month>= 3) {d++;}return d;
}

3.编写一个函数print,打印一个学生的成绩数组,该数组中有5个学生的数据记录,每个记录包括num,name,score[3],用主函数输人这些记录,用print函数输出这些记录。

#include <stdio.h>typedef struct Student {int num;char name[30];float score[3];
}Student;int main() {void print(Student *stu);Student stu[5];printf("请输入5组学生的数据\n");fflush(stdout);for (int i = 0; i < 5; ++i) {printf("请输入编号:");fflush(stdout);scanf("%d", &stu[i].num);printf("请输入名字:");fflush(stdout);scanf("%s", stu[i].name);printf("请输入分数:");fflush(stdout);for (int j = 0; j < 3; ++j) {scanf("%f", &stu[i].score[j]);}fflush(stdin);}print(&stu);return 0;
}void print(Student *stu) {for (int i = 0; i < 5; ++i) {printf("编号:%d,姓名:%s,分数1:%f,分数2:%f,分数3:%f\n", (stu+i)->num, (stu+i)->name, (stu+i)->score[0], (stu+i)->score[1], (stu+i)->score[2]);}
}

4.在第3题的基础上,编写一个函数input,用来输人5个学生的数据记录。

#include <stdio.h>typedef struct Student {int num;char name[30];float score[3];
}Student;int main() {void print(Student *stu);void input(Student *stu);Student stu[5];input(&stu);print(&stu);return 0;
}void input(Student *stu) {printf("请输入5组学生的数据\n");fflush(stdout);for (int i = 0; i < 5; ++i) {printf("请输入编号:");fflush(stdout);scanf("%d", &stu[i].num);printf("请输入名字:");fflush(stdout);scanf("%s", stu[i].name);printf("请输入分数:");fflush(stdout);for (int j = 0; j < 3; ++j) {scanf("%f", &stu[i].score[j]);}fflush(stdin);}
}void print(Student *stu) {for (int i = 0; i < 5; ++i) {printf("编号:%d,姓名:%s,分数1:%f,分数2:%f,分数3:%f\n", (stu+i)->num, (stu+i)->name, (stu+i)->score[0], (stu+i)->score[1], (stu+i)->score[2]);}
}

5.有10个学生,每个学生的数据包括学号、姓名、3门课程的成绩,从键盘输人10个学生数据,要求输出3门课程总平均成绩,以及最高分的学生的数据(包括学号、姓名、3门课程成绩、平均分数)。


6.13个人围成一圈,从第1个人开始顺序报号1,2,3。凡报到3者退出圈子。找出最后留在圈子中的人原来的序号。要求用链表实现。

7.在第9章例9.9和例9.10的基础上,写一个函数del,用来删除动态链表中指定的节点

#include <stdio.h>
#include <stdlib.h>typedef struct Student {int num;float score;struct Student *next;
}Student, *SPoint;int main() {//    一定要有返回!!!stu传入函数后,在函数内分配的地址会被回收。SPoint createStuList(SPoint stu, int n);void printStuList(SPoint stu);int deleteStu(SPoint sp, int num);SPoint stu;int n;printf("请输入需要创建的学生数量 n = 3\n");fflush(stdout);stu = createStuList(stu, n);
//    printStuList(stu);int num;printf("请输入想要删除的编号:");fflush(stdout);scanf("%d", &num);int result = deleteStu(stu, num);if (result == 1) printf("删除成功\n");else printf("未找到相关的编号\n");printf("=====================\n");printStuList(stu);return 0;
}SPoint createStuList(SPoint stu, int n) {Student *r;stu = (Student *) malloc(sizeof(Student));stu->next = NULL;   // 记得跟上NULLr = stu;for (int i = 0; i < 3; ++i) {Student *s = (Student *) malloc(sizeof(Student));s->num = i + 1;printf("请输入编号:%d,", s->num);fflush(stdout);s->score = 90 - 10 * i;printf("请输入分数:%f\n", s->score);fflush(stdout);r->next = s;r = s;  // 这边不要写成了s = r;}r->next = NULL;return stu;
}void printStuList(SPoint stu) {SPoint p;p = stu->next;while (p) {printf("编号:%d,成绩:%f\n", p->num, p->score);p = p->next;}
}int deleteStu(SPoint sp, int num) {//找到并删除成功返回1,未找到返回0;Student *p, *pre;p = sp->next;pre = sp;while (p) {if(p->num == num) {pre->next = p->next;free(p);return 1;}p = p->next;pre = pre->next;}return 0;
}

8.写一个函数insert,用来向一个动态链表插入结点

#include <stdio.h>
#include <stdlib.h>typedef struct Student {int num;float score;struct Student *next;
} Student, *SPoint;int main() {//    一定要有返回!!!stu传入函数后,在函数内分配的地址会被回收。SPoint createStuList(SPoint stu, int n);void printStuList(SPoint stu);int deleteStu(SPoint sp, int num);void insertStu(SPoint sp, Student *s);SPoint stu;int n;printf("请输入需要创建的学生数量 n = 3\n");fflush(stdout);stu = createStuList(stu, n);printf("=====================\n");Student newStu;printf("请输入一个新的num: ");fflush(stdout);scanf("%d", &newStu.num);printf("请输入学生的分数:");fflush(stdout);scanf("%f", &newStu.score);insertStu(stu, &newStu);printStuList(stu);return 0;
}SPoint createStuList(SPoint stu, int n) {Student *r;stu = (Student *) malloc(sizeof(Student));stu->next = NULL;   // 记得跟上NULLr = stu;for (int i = 0; i < 3; ++i) {Student *s = (Student *) malloc(sizeof(Student));s->num = i + 1;printf("请输入编号:%d,", s->num);fflush(stdout);s->score = 90 - 10 * i;printf("请输入分数:%f\n", s->score);fflush(stdout);r->next = s;r = s;  // 这边不要写成了s = r;}r->next = NULL;return stu;
}void printStuList(SPoint stu) {SPoint p;p = stu->next;while (p) {printf("编号:%d,成绩:%f\n", p->num, p->score);p = p->next;}
}int deleteStu(SPoint sp, int num) {//找到并删除成功返回1,未找到返回0;Student *p, *pre;p = sp->next;pre = sp;while (p) {if (p->num == num) {pre->next = p->next;free(p);return 1;}p = p->next;pre = pre->next;}return 0;
}void insertStu(SPoint sp, Student *s) {Student *pre, *p;pre = sp;p = sp->next;while (p != NULL && s->num >= p->num) {p = p->next;pre = pre->next;}s->next = p;pre->next = s;
}

9.综合本章例9.9(建立链表的函数creat)、例9.10(输出链表的函数print)和本章习题第7题(删除链表中结点的函数del)、第8题(插入结点的函数insert),再编写一个主函数,先后调用这些函数。用以上5个函数组成一个程序,实现链表的建立、输出、删除和插入,在主函数中指定需要删除和插人的结点的数据。

#include <stdio.h>
#include <stdlib.h>typedef struct Student {int num;float score;struct Student *next;
} Student, *SPoint;int main() {//    一定要有返回!!!stu传入函数后,在函数内分配的地址会被回收。SPoint createStuList(SPoint stu, int n);void printStuList(SPoint stu);int deleteStu(SPoint sp, int num);void insertStu(SPoint sp, Student *s);SPoint stu;int n;printf("请输入需要创建的学生数量 n = 3\n");fflush(stdout);stu = createStuList(stu, n);printf("=====================\n");Student newStu;printf("请输入一个新的num: ");fflush(stdout);scanf("%d", &newStu.num);printf("请输入学生的分数:");fflush(stdout);scanf("%f", &newStu.score);insertStu(stu, &newStu);printStuList(stu);return 0;
}SPoint createStuList(SPoint stu, int n) {Student *r;stu = (Student *) malloc(sizeof(Student));stu->next = NULL;   // 记得跟上NULLr = stu;for (int i = 0; i < 3; ++i) {Student *s = (Student *) malloc(sizeof(Student));s->num = i + 1;printf("请输入编号:%d,", s->num);fflush(stdout);s->score = 90 - 10 * i;printf("请输入分数:%f\n", s->score);fflush(stdout);r->next = s;r = s;  // 这边不要写成了s = r;}r->next = NULL;return stu;
}void printStuList(SPoint stu) {SPoint p;p = stu->next;while (p) {printf("编号:%d,成绩:%f\n", p->num, p->score);p = p->next;}
}int deleteStu(SPoint sp, int num) {//找到并删除成功返回1,未找到返回0;Student *p, *pre;p = sp->next;pre = sp;while (p) {if (p->num == num) {pre->next = p->next;free(p);return 1;}p = p->next;pre = pre->next;}return 0;
}void insertStu(SPoint sp, Student *s) {Student *pre, *p;pre = sp;p = sp->next;while (p != NULL && s->num >= p->num) {p = p->next;pre = pre->next;}s->next = p;pre->next = s;
}

10.已有a,b两个链表,每个链表中的结点包括学号、成绩。要求把两个链表合并, 按学号升序排列。

#include <stdio.h>
#include <stdlib.h>
#include <time.h>typedef struct Student {int num;float score;struct Student *next;
} Student, *StudentList;int main() {//    一定要有返回!!!stu传入函数后,在函数内分配的地址会被回收。StudentList createStuList(StudentList stu, int n);void printStuList(StudentList stu);void mergeList(StudentList Sa, StudentList Sb);void sortList(StudentList S);StudentList Sa, Sb;int na, nb;printf("请输入需要创建的学生数量 na和nb: ");fflush(stdout);scanf("%d,%d", &na, &nb);fflush(stdin);Sa = createStuList(Sa, na);Sb = createStuList(Sb, nb);printStuList(Sa);printf("*****************\n");printStuList(Sb);printf("=====================\n");//    对每个链表排序sortList(Sa);sortList(Sb);printStuList(Sa);printf("*****************\n");printStuList(Sb);printf("=====================\n");mergeList(Sa, Sb);printStuList(Sa);return 0;
}StudentList createStuList(StudentList stu, int n) {Student *r;stu = (Student *) malloc(sizeof(Student));stu->next = NULL;   // 记得跟上NULLr = stu;unsigned int second = time(NULL);srand(second);//    延迟函数while (1) {unsigned int nextSecond = time(NULL);if(second + 2 >= nextSecond) break;}for (int i = 0; i < n; ++i) {Student *s = (Student *) malloc(sizeof(Student));printf("请输入编号:");fflush(stdout);scanf("%d", &s->num);printf("请输入分数:");fflush(stdout);scanf("%f", &s->score);r->next = s;r = s;  // 这边不要写成了s = r;}r->next = NULL;return stu;
}void printStuList(StudentList stu) {StudentList p;p = stu->next;while (p) {printf("编号:%d,成绩:%f\n", p->num, p->score);p = p->next;}
}void mergeList(StudentList Sa, StudentList Sb) {Student *head, *ph, *pa, *pb;head = Sa;ph = Sa;pa = Sa->next;pb = Sb->next;while (pa && pb) {if(pa->num <= pb->num) {ph->next = pa;ph = pa;pa = pa->next;}else {ph->next = pb;ph = pb;pb = pb->next;}}if (pa) {ph->next = pa;}if (pb) {ph->next = pb;}
}void sortList(StudentList S) {int compare(const void*, const void*);int size = 0;Student *hp, *sp;hp = S->next;while (hp) {hp = hp->next;size++;}hp = S->next;sp = (Student*) malloc(sizeof(Student)*size);for (int i = 0; i < size; ++i) {*(sp + i) = *hp;hp = hp->next;}hp = S->next;qsort(sp, size, sizeof(Student), compare);for (int i = 0; i < size; ++i) {*hp = *(sp +i);hp = hp->next;}
}int compare(const void *a, const void *b) {Student sa = *(Student*)a;Student sb = *(Student*)b;return (sa.num - sb.num);
}

11.有两个链表a和b,设结点中包含学号、姓名。从a链表中删去与b链表中有相同学号的那些结点。

#include <stdio.h>
#include <stdlib.h>typedef struct Student {int num;float score;struct Student *next;
} Student, *StudentList;int StudentSize(StudentList S);     // 求列表的元素int main() {//    一定要有返回!!!stu传入函数后,在函数内分配的地址会被回收。StudentList createStuList(StudentList stu, int n);void printStuList(StudentList stu);void sortList(StudentList S);void deleteSameElem(StudentList Sa, StudentList Sb);StudentList Sa, Sb;int na, nb;printf("请输入需要创建的学生数量 na和nb: ");fflush(stdout);scanf("%d,%d", &na, &nb);fflush(stdin);Sa = createStuList(Sa, na);Sb = createStuList(Sb, nb);deleteSameElem(Sa,Sb);printStuList(Sa);return 0;
}StudentList createStuList(StudentList stu, int n) {Student *r;stu = (Student *) malloc(sizeof(Student));stu->next = NULL;   // 记得跟上NULLr = stu;for (int i = 0; i < n; ++i) {Student *s = (Student *) malloc(sizeof(Student));printf("请输入编号:");fflush(stdout);scanf("%d", &s->num);printf("请输入分数:");fflush(stdout);scanf("%f", &s->score);r->next = s;r = s;  // 这边不要写成了s = r;}r->next = NULL;return stu;
}void printStuList(StudentList stu) {StudentList p;p = stu->next;while (p) {printf("编号:%d,成绩:%f\n", p->num, p->score);p = p->next;}
}void sortList(StudentList S) {int compare(const void*, const void*);int size = StudentSize(S);  //链表长度Student *hp, *sp;hp = S->next;sp = (Student *) malloc(sizeof(Student)*size);  //sp指向链表S的数组形式for (int i = 0; i < size; ++i) {*(sp + i) = *hp;hp = hp->next;}hp = S->next;qsort(sp, size, sizeof(Student), compare);for (int i = 0; i < size; ++i) {hp->num = (sp +i)->num;hp->score = (sp +i)->score;hp = hp->next;}
}int compare(const void *a, const void *b) {Student sa = *(Student*)a;Student sb = *(Student*)b;return (sa.num - sb.num);
}void deleteSameElem(StudentList Sa, StudentList Sb) {//    两个链表必须是排序过的int sizeA, sizeB, indexA;Student *pa, *pb, *pre;pa = Sa->next;pb = Sb->next;sizeA = StudentSize(Sa);sizeB = StudentSize(Sb);Student *arrA = (Student*) malloc(sizeof(sizeA));Student *arrB = (Student*) malloc(sizeof(sizeB));for (int i = 0; i < sizeB; ++i) {*(arrB + i) = *pb;pb = pb->next;}//    删除B链表中num相同的结构pre = Sa;pa = Sa->next;while (pa) {for (int i = 0; i < sizeB; ++i) {if(pa->num == (arrB+i)->num) {pre->next = pa->next;pa = pa->next;}else {pre = pre->next;pa = pa->next;}}}
}int StudentSize(StudentList S) {int size = 0;Student  *p = S->next;while (p) {p = p->next;size++;}return size;
}

12.建立一个链表,每个结点包括:学号、姓名、性别、年龄。输入一个年龄,如果链表中的结点所包含的年龄等于此年龄,则将此结点删去。

#include <stdio.h>
#include <stdlib.h>typedef struct Student {int no;char name[30];char sex;   //年龄男性为M,女性为Wint age;struct Student *next;
} Student, *StudentList;// 求列表的元素
int StudentSize(StudentList S) {int size = 0;Student  *p = S->next;while (p) {p = p->next;size++;}return size;
}int main() {//    一定要有返回!!!stu传入函数后,在函数内分配的地址会被回收。StudentList createStudentList(StudentList stuHead, int n);
//    如果年龄相同,则把这个元素在链表中删除void deleteStudent(StudentList stuHead, int age);
//    打印链表void printStudentList(StudentList stu);int n, age;Student *stuHead;//    创建一组链表printf("请输入个数,这个数将用来创建一组链表。\n");printf("n = ");fflush(stdout);scanf("%d", &n);stuHead = createStudentList(stuHead, n);//    如果链表中有值和输入的年龄相同则把链表中删除的元素删去。printf("请输入一个年龄\n");printf("age = ");fflush(stdout);scanf("%d", &age);deleteStudent(stuHead, age);printStudentList(stuHead);return 0;
}StudentList createStudentList(StudentList stuHead, int n) {Student *r;stuHead = (Student *) malloc(sizeof(Student));stuHead->next = NULL;   // 记得跟上NULLr = stuHead;for (int i = 0; i < n; ++i) {Student *s = (Student *) malloc(sizeof(Student));printf("请输入学号:");fflush(stdout);scanf("%d", &s->no);printf("请输入姓名:");fflush(stdout);scanf("%s", s->name);fflush(stdin);printf("请输入性别:");fflush(stdout);scanf("%c", &s->sex);printf("请输入年龄:");fflush(stdout);scanf("%d", &s->age);r->next = s;r = s;  // 这边不要写成了s = r;}r->next = NULL;return stuHead;
}void deleteStudent(StudentList stuHead, int age) {Student *sp, *pre, *m;pre = stuHead;sp = stuHead->next;while (sp) {if(sp->age == age) {pre->next = sp->next;m = sp;sp = sp->next;free(m);}else {pre = pre->next;sp = sp->next;}}
}void printStudentList(StudentList stuHead) {StudentList p;p = stuHead->next;while (p) {printf("学号:%d,姓名:%s,性别:%c,年龄:%d\n", p->no, p->name, p->sex, p->age);p = p->next;}
}

C程序设计 谭浩强 第九章相关推荐

  1. C++面向对象的程序设计谭浩强 第六章课后题

    以往章节 C++面向对象的程序设计谭浩强 第二章课后题 C++面向对象的程序设计谭浩强 第三章课后题 C++面向对象的程序设计谭浩强 第四章课后题 C++面向对象的程序设计谭浩强 第五章课后题 C++ ...

  2. C程序设计谭浩强第五版课后答案 第三章习题答案

    C语言程序设计谭浩强第五版课后答案第三章 1.假如我国国民生产总值的年增长率为7%, 计算10年后我国国民生产总值与现在相比增长多少百分比.计算公式为p=(1+r)np = (1+r)^np=(1+r ...

  3. C程序设计谭浩强版总结笔记

    结构体 注:结构体类型的名字是由一个关键字struct和结构体名组合而成的(例如struct Student).结构体名是由用户指定的,又称"结构体标记",以区别于其他结构体类型. ...

  4. C程序设计(谭浩强第五版)总结

    C程序设计(谭浩强第五版)总结 本篇文章主要是总结谭浩强第五版C语言书上的重点和易漏点的知识点,其目的主要是给高校期末考试的同学们点参考.本文所参考的书籍是谭浩强的<C程序设计(第五版)> ...

  5. C程序设计-谭浩强 第三版-学习笔记第1章 C语言概述

    第一章 C语言概述 1.C语言历史背景 C语言是在B语言的基础上发展起来的,兼具一般高级语言和低级语言的优点,可用来编写系统软件或应用软件. 1972-1973年,贝尔实验室在B语言基础上设计出C语言 ...

  6. C程序设计-谭浩强 第三版-学习笔记 第2章 程序的灵魂 算法

    第 2 章 算法 --程序的灵魂 程序 = 算法 + 数据结构 (沃思,计算机科学家) 一个程序应该包括两方面: 对数据的描述:在程序中要指定数据的类型和数据的组织形式,即数据结构(data stru ...

  7. C程序设计 谭浩强 第三章

    变量类型 常量 注:一般变量初始化不是在编译阶段完成的(只有静态存储变量和外部变量的初始化时在编译阶段完成的),而是在程序运行时执行本函数时赋予初值的,相当于执行一个赋值语句. 整型常量 实型常量:十 ...

  8. C语言程序设计谭浩强(第四版)部分课后习题作答——第七章——7.8,7.10,7.16

    7.8:输入一个4位数,输出这4个数字字符 7.10:输入一行字符,输出最长的单词 7.16:十六进制转十进制 #include<stdio.h> #include<stdlib.h ...

  9. C语言程序设计谭浩强(第四版)部分课后习题作答——第六章-输出杨辉三角,输出魔方阵

    代码: 杨辉三角题目较为简单,魔方阵题目实现了输出任意整数即可输出对应的魔方阵 #include<stdio.h> #include<stdlib.h> #include< ...

最新文章

  1. git---远程仓库版本回滚
  2. 【免费福利】零AI基础,如何搭建聊天机器人:技术架构剖析
  3. 数字图像处理中常用图像分割算法有哪些?
  4. 洛谷——P2118 比例简化
  5. 【干货】迅雷产品经理:浅析用户成长体系
  6. Github标星8.3k+,Tensorflow 2.0的代码实现及教学材料(“龙书”)公布下载了!
  7. Java Web专题(一)
  8. mysql 5.7主从延迟 相关参数配置
  9. MFC 蜂鸣声或播放音频
  10. [转]Xshell连接win10 Linux子系统
  11. linux下各种颜色文件的意义
  12. Spring Boot Initilizr Web界面
  13. XML 文档对象模型 (DOM)细说
  14. StructureMap极速上手指南(翻译)
  15. 使用 SDK (Nodejs)操作阿里 OSS (对象存储服务)学习笔记
  16. 自媒体全套教程+全套工具(带教程)+原创实操教程
  17. 如何使用 FFMpeg 在 Node.js 中将音频从 Wav 转换为 MP3
  18. Android移动应用技术打地鼠小游戏(简单App实现)——学习成果
  19. 什么是DAOstack
  20. Perl中的单行注释和多行注释

热门文章

  1. 将一个DataFrame中的一列(行),插入到另一个DataFrame中
  2. 【Python】字符串 - 集大成篇
  3. censo7安装mysql_centos7 安装MySQL
  4. 电脑小写字母怎么切换_macOS amp; Windows 通吃,一套键鼠就能控制多台电脑
  5. 一级计算机快捷键大全,计算机快捷键大全(最全篇)
  6. Visio里Mathtype公式变形问题解决
  7. android 锯齿
  8. 12.pandas 读取与写入文件
  9. html文档中strokestyle,HTML5简明教程-1.1.2.HTML5Canvas参考手册 之 笔触strokeStyle
  10. Apache Pulsar的数据写入和读取流程,及读写异常处理流程