递归算法就是直接或者间接的调用自己的算法。一般是通过函数或者子过程完成,在函数或者子过程的内部完成自身的调用。
实质是循环调用函数
学生信息管理系统:
需求分析
系统设计
1、系统界面简易、易操作2、能够从磁盘中加载数据3、可以将信息保存到磁盘中4、系统具有增加、查询、删除、显示等功能
功能:插入(输入学生信息,输入的信息保存在内存的链表中)、保存(将存储在链表中的信息保存在磁盘中)、加载(将磁盘中存储的信息加载到链表中)、显示(显示链表中所有学生的信息)、查找(将通讯录中所有学生的信息按照姓名的字母顺序进行排列)、删除学生(可以在链表中删除指定学号的学生信息)
主功能菜单设计由menu_select()实现
数据结构的设计:
将学生设计为一个结构体
struct student{
int num;//学号
char name[15];//姓名
char sex[2];//性别
int age;//年龄
double score[3];//三门课程的分数
double sum;//总分
double ave;//平均分
};
动态链表:
定义链表节点:链表结构,节点之间通过指针连接
typedef struct node
{
struct student data;
struct node *next;
}node;*link;
主函数设计
动态内存分配函数:malloc()函数,在头文件stdlib.h中定义
1、调用方式
void*malloc(unsigned size)
功能:在内存中分配size个内存空间,并返回指向被分配存储区起始地址的指针,如果不能获取需要的存储空间,返回空指针
注意:当把返回值赋给具有某一特定数据类型的指针变量时,应该对指针进行强制类型转换
例子:为1000个字符动态的分配内存
char*prt;
ptr=(char*)malloc(1000);
分配一个新的链表节点:
l=(Node*)malloc(sizeof(Node));
calloc()函数
调用方式:void*calloc(usignde n,unsigned size )
分配n个具有size个字节的存储空间,并返回一个指向一个被分配内存起始地址的指针,如果没有足够的内存可供分配,则返回一个空指针
free()函数
调用方式:void free(void *ptr)
释放由ptr指向的内存空间,并将它返回给堆。free函数无返回值
注意:ptr是最近一次调用malloc或者calloc函数的返回值
动态分配的存储单元用完之后一定要释放,否则内存中因为申请空间过多引起资源不足而出现故障
调用menu_select()显示主功能菜单,并在switch()分支选择结构中调用各个主函数实现对学生信息的插入等功能
#include
#include
#include
#include
#include
int menu_select()
{
int i;
printf("\n\n\t********************STUDENT LIST********************\n");
printf("\t}* 1.input record *|\n");
printf("\t}* 2.delete record *|\n");
printf("\t}* 3.list record *|\n");
printf("\t}* 4.search record *|\n");
printf("\t}* 5.save record *|\n");
printf("\t}* 6.load record *|\n");
printf("\t}* 7.quit *|\n");
printf("\n\n\t****************************************************\n");
do
{
printf("\n\tenter your choice:");
scanf("%d",&i);
}while(i<=0||i>7);
return i;
}
main()
{
link l;
l=(Node*)malloc(sizeof(Node));
if(!l)
{
printf("\n allocate memory failure!");
return ;
}
l->next=NULL;
while(1)
{
system("cls");
switch(menu_select())
{
case 1:input(l);break;
case 2:del(l);break;
case 3:list(l);break;
case 4:search(l);break;
case 5:save(l);break;
case 6:load(l);break;
case 7:exit(0);
}
}
}
插入学生信息模块
结构体变量之间可以进行运算、赋值等操作
比如:定义了结构体变量,student1、student2
student1=student2,但是不能进行运算操作。student.age++,变量名成员,可以进行运算操作
scanf("%d%s%f",&student1.num,student.name,&student.score);
num为int score为int name为数组型,数组名本来就代表地址,不需要画蛇添足
结构体例子:与学生系统无关
#include
#include
struct person{
char name[20];
int count;
}leader[3]={"li",0,"zhang",0,"xia",0};//定义三位领导,初始化票数为0;
int main()
{
int i,j;
char leader_name[20];
for(i=0;i<10;i++)
{
printf("请输入投票领导的姓名:");
scanf("%s",leader_name);
system("cls");//清屏操作
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);//左对齐5个位置
return 0;
}
结构体数据冒泡排序:
#include
struct student
{
int num;
char name[20];
float score;
};
int main()
{
struct student stu[5]={{10101,"zhao",80},{10102,"qian",81},{10103,"sun",82},{10104,"li",83},{10105,"xia",84}};
struct student temp;
const int n=5;
int i,j;
printf("the precious order:\n");
for(i=0;i<5;i++)
printf("%lf\t",stu[i].score);
printf("\n");
for(i=0;i<4;i++)
{
for(j=i+1;j<5;j++)
{
if(stu[i].score<stu[j].score)
{
temp=stu[j];
stu[j]=stu[i];
stu[i]=temp;
}
}
}
printf("after order is:\n");
for(i=0;i<5;i++)
{
printf("%lf\t",stu[i].score);
}
return 0;
}
结构体指针:
指向结构体对象的指针变量既可以指向结构体变量,也可以指向结构体数组中的元素
#include
#include
int main()
{
struct student{
long num;
char name[20];
char sex;
float score;
};
struct student stu_1;
struct student *p;//定义指向struct student类型数据的指针变量p
p=&stu_1;
stu_1.num=10101;
strcpy(stu_1.name,"LiLin");
stu_1.sex='m';
stu_1.score=89.5;
printf("NO.:%ld\n,name:%s\n,sex:%c\n,score:%lf\n",stu_1.num,stu_1.name,stu_1.sex,stu_1.score);
printf("\n\nNO.:%ld\n,name:%s\n,sex:%c\n,score:%lf\n",(*p).num,(*p).name,(*p).sex,(*p).score);
printf("\n\nNO.:%ld\n,name:%s\n,sex:%c\n,score:%lf\n",p->num,p->name,p->sex,p->score);
//这三种输出结果一致,最后一种使用指针指向的方式struct student *p;先定义,p=&stu_1;后指向
p指向结构体变量名,因此可以指向结构体中的元素
return 0;
}
指向结构体数组的指针:
#include
int main()
{
struct student{
long num;
char name[20];
char sex;
int age;
};
struct student *p;
struct student stu[3]={{10101,"liming",'m',18},{10102,"wangming",'m',20},{10103,"chengming",'f',16}
};
for(p=stu;p<stu+3;p++)
printf("NO.:%ld,name:%s,sex:%c,age:%d\n",p->num,p->name,p->sex,p->age);
return 0;
}