程序支持增加、查询、删除、存盘和读取操作

一 程序定义和函数声明

  头文件studentsys.h定义如下

/*  student management system by list  */
#ifndef _StudentSys_H
#define _StudentSys_H #define MAX 25 /* lenth of name string */
typedef struct _Grade Grade;;
typedef struct StdNode Std;;
typedef struct ListNode *List;

  全局结构体定义

/* 6 courses */
struct _Grade{
int Chinese;
int Math;
int English;
int Programming;
int Physic;
int Geology;
}; /* student info */
struct StdNode {
int Id; /* student id */
char name[MAX];/* student name */
Grade grades; /* student grades */
}; struct ListNode {
Std info;
List next;
};

  函数声明

  /* 1->Add   2->Find   3->Del   4->Brows   5->Store
     6->Read    7->DelAll   8->Cls    9->Quit  */    功能选择键

int AddStd( List L, int Id, char *name, Grade grades );
/* flag:0 by id, 1 by name */
int FindStd(List L, int flag, int Id ,char *name);
int DeleteStd( List L, int Id );
void PrintInfo( List L );
void PrintAll( List L );
int StoreInfo( List L );
List ReadInfo( );
void DestoryInfo( List L );
int IsRepeat( List L, int Id );
void selectInfo();

二 源文件studentsys.c

  主体部分,由while循环下的switch分支进行功能选择

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "studentsys.h" int main(void)
{
List L; /* Head Node */
L = (List)malloc(sizeof( struct ListNode ));
L->next = NULL; int i, flag; /* flag做为查找方式标志, 0由Id, 1由姓名进行搜索 */
int option, Id = ; /* option选择功能, init info */
char name[MAX]= " ";
Grade grades; printf(" Starting \n");
printf(" Student Management system, ver:0.101 \n");
printf("******************************************************\n");
printf(" 学生管理系统\n");
printf("$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$\n");
selectInfo();
while ( ) {
scanf("%d", &option);
switch ( option ) {
case : printf("Please in put id, name, six grades:\n");
scanf("%d %s", &Id, name);
scanf("%d %d %d %d %d %d", &grades.Chinese, &grades.Math,
&grades.English, &grades.Programming,&grades.Physic,
&grades.Geology );
if ( AddStd( L, Id, name, grades ) == ) {
printf("Added student.\n");
} else {
printf("Sorry, can't add student\n");
}
break; case : printf("select method: 0 by Id, 1 by name\n");
scanf("%d", &flag);
printf("Please in put id, name\n");
scanf("%d %s", &Id, name); /* notice & */
if ( FindStd( L, flag, Id, name ) == ) {
printf("found student\n");
} else {
printf("sorry, can't found student\n");
}
break; case : printf("Which id you want to del: ");
scanf("%d", &Id);
if ( DeleteStd( L, Id ) ) {
printf("Successfully Delete!\n");
} else {
printf(" Not found!\n");
}
break; case : PrintAll( L );
break; case : if ( StoreInfo( L ) ) {
printf("Write successfulluy!\n");
} else {
printf("Failed to read!\n");
}
break; case : L = ReadInfo( );
break;
case : DestoryInfo( L );
break;
case : system("cls");
selectInfo();
break;
case : exit();
break;
default: printf("error!请重新输入!\n");
}
}
return ;
}

  

  函数主体,由简单的链表的插入、删除和遍历

int AddStd( List L, int Id, char *name, Grade grades )
{
List Head = L;
List Cur;
if( IsRepeat( L, Id) == ) {
printf(" Id repeat!!!\n");
return ;
}
/* 链表的插入 */
Cur = (List)malloc(sizeof( struct ListNode ));
if ( Cur != NULL ) {
Cur->info.Id = Id;
strcpy( Cur->info.name, name);
Cur->info.grades.Chinese = grades.Chinese;
Cur->info.grades.Math = grades.Math;
Cur->info.grades.English = grades.English;
Cur->info.grades.Programming = grades.Programming;
Cur->info.grades.Physic = grades.Physic;
Cur->info.grades.Geology = grades.Geology; Cur->next = Head->next;
Head->next = Cur; return ;
} else {
printf("Failed to malloc!\n");
return ;
}
} int FindStd( List L, int flag, int Id , char *name )
{
List Cur = L->next;
while ( Cur != NULL ) {
if( flag == ) {
if ( strcmp( name, Cur->info.name ) == ) {
PrintInfo(Cur);
break;
}
} else if ( flag == ) {
if ( Cur->info.Id == Id ) {
PrintInfo(Cur);
break;
}
}
Cur = Cur->next;
}
if ( Cur == NULL ){
return ;
} else {
return ;
}
} int DeleteStd( List L, int Id )
{
List Pre = L;
List Cur = Pre->next; if( L->next == NULL ) {
printf("Empty Node!\n");
return ;
} else {
while ( Cur != NULL ) {
if ( Cur->info.Id == Id ) {
/* 找到对应编号 */
Pre->next = Cur->next;
free(Cur);
break;
}
Pre = Cur;
Cur = Cur->next;
}
if ( Cur == NULL ) {
return ;
} else {
return ;
}
} /* 1 else */
} void PrintInfo( List L )
{
List Cur = L;
if ( Cur != NULL ) {
printf("ID: %d ", Cur->info.Id);
printf("Name: %s\n", Cur->info.name);
printf("Chinese :%d ", Cur->info.grades.Chinese);
printf("Math :%d ", Cur->info.grades.Math);
printf("English :%d\n", Cur->info.grades.English);
printf("Programming:%d ", Cur->info.grades.Programming);
printf("Physic :%d ", Cur->info.grades.Physic);
printf("Geology :%d\n", Cur->info.grades.Geology);
}
} void PrintAll( List L )
{
List Cur = L->next;
int i = ;
while ( Cur != NULL ) {
i++;
printf("%d-->Student ", i);
PrintInfo( Cur );
Cur = Cur->next;
}
}
void DestoryInfo( List L )
{
List Cur, Tmp;
Cur = L->next;
while ( Cur != NULL ) {
Tmp = Cur->next;
free( Cur );
Cur = Tmp;
}
L->next = NULL;
printf("删除所有学生信息成功!\n");
} int IsRepeat( List L, int Id )
{
List Cur = L->next;
int flag = ;
while ( Cur != NULL ) {
if ( Cur->info.Id == Id ) {
       flag = ;
       break;    
     }
     Cur = Cur->next;
}
  return flag;
} void selectInfo()
{
printf("Stumansys 请选择:\n");
printf("(1)---增加学生 (2)---查询学生 (3)---删除学生 \n");
printf("(4)---查看全部 (5)---记录存盘 (6)---读取信息\n");
printf("(7)---删除所有 (8)---清除屏幕 (9)---退出系统\n");
}

  文件读写操作

    1. 链表写入文件,每次写一个结构体,直到链表尾。

    2. 文件读取,使用feof(fp)函数判断是否达到文件流尾,否则读取一个结构体,并尾插入链表。

int StoreInfo( List L )
{
FILE *fp = fopen("student.data", "w");
List H = L->next;
const int size = ;
int ret = -; /* default */ if ( fp == NULL ) {
printf("Failed to open file!\n");
} else if( H == NULL ) {
printf("Empty student info!\n");
} else {
/* write list into fp */
while ( H != NULL ) {
ret = fwrite( H, sizeof(H->info), size, fp );
H = H->next;
}
ret = ;
}
fclose( fp );
return ret;
} List ReadInfo( )
{
int i = ; /* count */
List L, P, tmp;
P = L = (List)malloc( sizeof(struct ListNode) );
FILE *fp = fopen("student.data", "r"); if ( fp == NULL ) {
printf("Failed to read file!\n");
return NULL;
}
/* 假如文件为空,检查文件结束符 feof在到达文件流尾返回非零值 */
if ( feof(fp) != ) {
tmp = (List)malloc( sizeof(struct ListNode) );
fread( tmp, sizeof(tmp->info), , fp);
if( feof(fp) ) {
printf("读取失败!文件为空!\n");
free(tmp);
}
return(NULL);
} while ( feof(fp) == ) {
tmp = (List)malloc( sizeof(struct ListNode) );
fread( tmp, sizeof(tmp->info), , fp);
printf("正在读取第%d个数据\n", ++i);
if ( feof(fp) != ) {
/* reached end */
free(tmp); break;
}
/* List tail insert */
P->next = tmp;
P = tmp;
P->next = NULL;
}
fclose(fp);
printf("Read successfulluy!\n");
return L;
}

C语言小程序-基于链表的学生信息管理的更多相关文章

  1. 通过反汇编C语言小程序学习Liunx汇编语言

    大家好!    我是来自山东师范大学的吴乐.    今天在<Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 ...

  2. Linux下简单C语言小程序的反汇编分析

    韩洋原创作品转载请注明出处<Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 写在开始,本文为因为参加MOO ...

  3. 小程序基于疼讯qcloud的nodejs开发服务器部署

        腾讯,疼讯,很疼. 请慎重看腾讯给出的文档,最好做一个笔记. 我只能说我能力有限,在腾讯云小程序的文档中跳了n天. 最后还是觉得记录下来,以防止我的cpu过载给烧了. 此文档是对<小程序 ...

  4. c语言小程序以及java生成注释文档方法

    c语言小程序:sizeof和strlen() sizeof运算符以字节为单位给出数据的大小,strlen()函数以字符为单位给出字符串的长度,字符和字节不是一回事. char类型用于存储字母和标点符号 ...

  5. C语言小程序——推箱子(窄字符和宽字符)

    C语言小程序——推箱子(窄字符Version) 推箱子.c #include <stdio.h> #include <conio.h> #include <stdlib. ...

  6. 微信小程序-基于canvas画画涂鸦

    代码地址如下:http://www.demodashi.com/demo/14461.html 一.前期准备工作 软件环境:微信开发者工具 官方下载地址:https://mp.weixin.qq.co ...

  7. 微信小程序基于swiper组件的tab切换

    代码地址如下:http://www.demodashi.com/demo/14010.html 一.前期准备工作 软件环境:微信开发者工具 官方下载地址:https://mp.weixin.qq.co ...

  8. 微信小程序基于scroll-view实现锚点定位

    代码地址如下:http://www.demodashi.com/demo/14009.html 一.前期准备工作 软件环境:微信开发者工具 官方下载地址:https://mp.weixin.qq.co ...

  9. Linux C语言小程序

    Linux C语言小程序 #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include & ...

随机推荐

  1. npm install warning: no description; no repository field

    npm install 报错:warning no description; no repository field 开始以为必须npm init,npm init在git bash(win7)里,还 ...

  2. Oracle数据库设计实例-实时生产效率系统数据库设计

    Oracle数据库设计实例-实时生产效率系统数据库设计 引言 1.1 设计前提 某部门经理要求IT部门设计一个流水线实时生产效率系统,用来统计实时的生产量和效率.流水线有数百条,实时间隔为1min. ...

  3. vsphere的P2V工具做的物理机迁移到虚拟机报错out of memory

     vsphere的P2V工具做的物理机迁移到虚拟机  迁移成功,但是启动报错  进入rescue模式后发现是sysctl.conf文件的参数设大了因为虚拟机的内存没有物理机内存大 kernel.shm ...

  4. Ace admin 如何实现类似于freamset加载页面

    如上标题所述,ace admin做后台页面的时候,可以实现类似于用freamset的功能,但是ace admin做的比freamset更好,他可以用异步加载的形式展示,而加载的页面的内容可以尽可能的少 ...

  5. WC2018 滚粗记

    虽然又考炸了但还是总结一下.. $day0$:没有什么很重要的事.. $day1$:除了听(dong)课(mian)以外没有什么很重要的事.. $day2$:除了听(dong)课(mian)以外没有什 ...

  6. SpringBoot学习(三)IDEA

    一.什么是JPA JPA(Java Persistence API)定义了一系列对象持久化的标准,目前实现这一规范的产品有Hibernate.TopLink等. 二.Mysql数据库示例 1.在app ...

  7. 新一代的IT实验室长啥样?

    这个时代 “互联网”江湖门派众多 “互联网+”现代农业.“互联网+”制造业. “互联网+”信息技术服务.“互联网+”文化创意服务. “互联网+”社会服务 各种“互联网+” 在不断的改变着我们的生活 这 ...

  8. 图片验证码——base64编码的使用

    一.介绍: 1.base64编码简介: Base64就是一种编码格式.Base64要求把每三个8Bit的字节转换为四个6Bit的字节(3*8 = 4*6 = 24),然后把6Bit再添两位高位0,组成 ...

  9. VC++读写*.ini配置文件

    ini文件(即Initialization file),这种类型的文件中通常存放的是一个程序的初始化信息.ini文件由若干个节(Section)组成,每个Section由若干键(Key)组成,每个Ke ...

  10. 【题解】洛谷P1351 [NOIP2014TG] 联合权值(树形结构+DFS)

    题目来源:洛谷P1351 思路 由题意可得图为一棵树 在一棵树上距离为2的两个点有两种情况 当前点与其爷爷 当前点的两个儿子 当情况为当前点与其爷爷时比较好操作 只需要在传递时不仅传递父亲 还传递爷爷 ...