【C语言】学籍管理系统

功能简介

  • 链表实现基本的创建和增删改查。

  • 成绩分析,链表的冒泡排序。

  • 分管理员,老师,学生三端管理,密码的修改,加密。

  • 分班级管理老师学生。

  • 方框界面,↑↓键控制,ENTER键确认。

界面功能展示

主界面
这里写图片描述
管理员登录
这里写图片描述

管理老师信息
这里写图片描述

管理学生信息
这里写图片描述

添加学生
这里写图片描述

成绩查询
这里写图片描述

具体实现的方法

1.界面

界面实现由↑↓控制选择,ENTER键确认。

这里用到了c语言光标移动函数gotoxy()。

gotoxy()函数定义
void gotoxy(int x, int y)
{ 
      COORD coord = {x, y};
     SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), coord);
}

命令行的列为x轴,行为y轴。
例如使用 gotoxy(10,5); 语句表示将光标移动到第十列,第五行。

先在第一个选项上打印出白框。
这里写图片描述
然后用getch();无缓冲读取用户的键盘输入。

用if判断,如果读取值的为↓键的ASCII码,则分别调用gotoxy(int x, int y)让光标跳到该白框的上面部分,中间部分和下面的部分的开头,打印一串空格覆盖掉白框,然后再调用gotoxy(int x, int y)让光标移动到下一个选项,分别打印出白框。除此之外还需再定义一个开关值,int key = 1, 然后每读取一次↓键,则key++,记录选中的选项,最后根据key的值再调用不同的函数。

如果判断为↑键,则同理。

还可以利用key值做特殊的判断,如果key的值为1且读入↑键,则让光标跳到最下面的选项打印白框,可以让白框循环移动。或者可以用取余数的办法, 例如一共有5个选项,则可以将每次key的值余6,便可使选框框循环移动。

附上一小段界面代码
按自己的思路写的,有点乱。仅供参考。

void zhujie() {
    char c;                                     // 用来读取键盘的输入
	int key = 1, x = 38 , y = 7;                // key值记录选项, x,y为初始的坐标,可根据自己系统具体而确定。
                                        	/*   打印界面  */
	printf("     --------------------------------------------------------------------------------------------------------------\n");
	printf("     |                                           学生信息管理系统                                                 |\n");
	printf("     --------------------------------------------------------------------------------------------------------------\n");
	printf("     |           请选择身份:                                                                                     |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                     ============================                                           |\n");
	printf("     |                                     |         1.管理员         |                                           |\n");
	printf("     |                                     ============================                                           |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                               2.老师                                                       |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                               3.学生                                                       |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                               4.退出程序                                                   |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                                                            |\n");
	printf("     |                                                                   ↑↓控制       ENTER  确认               |\n");
	printf("     --------------------------------------------------------------------------------------------------------------\n");
	c = getch();          // 先从键盘读取一个指令                    
	while( c != 13) {     //  如果不是回车循环进行,如果是回车的话直接跳出循环进行下一步判断 。
		if (c == 80 && key <= 4) {       // 判断如果读取的是↓键
			if(key == 4)                 // 如果已经是最后一个选项,又按了↓键,则跳到最上面
			key = 0;                     // 让key = 0 之后key会自增,就循环记录了第一个选项。
			gotoxy(43, y);               // 下面是打印空格替换先前的白框,
    		puts("                             ");
    		gotoxy(43, y+1);
    		puts(" ");
    		gotoxy(70, y+1);
    		puts(" ");
    		gotoxy(43, y+2);
    		puts("                             ");
		    if(key == 0)                 // 这里如果key等于0 则说明要从最后一行跳到第一行,让y等于4跳到第一行。
				y = 4;               
    		gotoxy(43, y+3);             //打印白框
    		puts("============================");
    		gotoxy(43, y+4);
    		puts("|");
    		gotoxy(70, y+4);
    		puts("|");
    		gotoxy(43, y+5);
    		puts("============================");
			key++;                      // 因为判断为↓键,所以开关值加一,记录到下一个选项。
			y += 3;                     // 坐标跳到下一个白框位置
			gotoxy(0,28);
		}
		if (c == 72 && key >= 1){       //判断如果是↑键, 与上面同理。
			if(key == 1)
				key = 5;
			gotoxy(43, y);
    		puts("                             ");
    		gotoxy(43, y+1);
    		puts(" ");
    		gotoxy(70, y+1);
    		puts(" ");
    		gotoxy(43, y+2);
    		puts("                             ");
    		if(key == 5)
				y = 19;
    		gotoxy(43, y-1);
    		puts("============================");
    		gotoxy(43, y-2);
    		puts("|");
    		gotoxy(70, y-2);
    		puts("|");
    		gotoxy(43, y-3);
    		puts("============================");
			key--;
			y -= 3;
			gotoxy(0,28);
		}
		c = getch();           //最后再读取一次键盘,循环判断,直到是回车(ASCII码为13)为止。
	}
	if (key == 1) denglu1();  // 这里跳出while 根据开关量决定进入哪个函数。
	if (key == 2) denglu2();
	if (key == 3) denglu3();
	if (key == 4) {
	gotoxy(0, 80);
	exit(0) ; 
	}
}

其他界面的实现可参考这个,都用的是gotoxy()函数。

2.多文件管理

可建立一个工程,分多文件来写管理系统,每个模块写一个文件,并添加一个头文件,方便管理查询。

分多文件管理

这里写图片描述

头文件

这里写图片描述
这里写图片描述

3.链表

数据结构
/*学生链表*/
struct ms {                                 // 学生信息和成绩 (数据域)
	char name[100];
	char sex[10];
	char number[10];
	int lisan;
	int math;
	int english;
	int c;
	int sum;
	int ban;
	char m[10]; 
	float avrg;
};
typedef struct lian {                       // 链表结构
	struct ms item ;                        // 数据域  这里分开两个域便于值的交换
	struct lian * next ;                    // 指针域
} lian;

/* 老师链表*/
struct tems {                               // 老师信息
	char name[100];							
	char z[21];
	char m[21];
	int ban;
};

typedef struct telian {						// 老师链表 
	struct tems teitem;
	struct telian * next;
}telian;
创建链表
/*返回结构指针的函数,返回创建的结构指针。也可用结构指针的指针当做函数的参数来传递值 */
lian * creat ()                             
 {                                          //创建学生链表 带头节点
	struct lian   * current =  NULL;
	current = (struct lian *)malloc(sizeof(struct lian));
	current->next = NULL;
	return current;
}



telian * creat2 () {                         //创建老师链表 带头结点
	struct telian   * current =  NULL;
	current = (struct telian *)malloc(sizeof(struct telian));
	current->next = NULL;
	return current;
}
按成绩排序查询

这里用的是优化后的链表的冒泡排序。这里按照sum(总分)进行排序,更换sum即可以按其他科目排序。

void sortsum() {	                          //冒泡排序链表
		struct lian  * current =  head->next; // 带头结点的链表,head->next 为第一个节点
		 /* 这里头指针head是在所有函数外声明的,具有文件作用域,可以再任意函数里直接使用 */
		struct lian t;                        // 用于赋值交换
		int ch = 1 , y=7;                     // ch为优化后的冒泡的开关量。
		if ( current == NULL )                // 判断链表是否为空
		{
				/* 这里打印提示信息, 说明链表没有成员。*/
		 }

		while (ch) {                         //冒泡排序,ch判断有无交换。
			ch = 0;
			while (current->next != NULL) {  // 节点遍历比较
				if ( current->item.sum < current->next->item.sum)
				{
					t.item = current->item;  // 结构可直接赋值, 交换
					current->item = current->next->item;
					current->next->item = t.item ;
					ch = 1;                 // 标记交换,还需继续循环
				}
				current = current->next ;
			}
		}
}

4. 文件存储

这里是用fwrite和fread以二进制一次性存储链表和读取链表,其他的增删改查操作都是用链表实现的。

void cun() {
	struct lian * current =  head->next;
	FILE * fp;
	fp = fopen("lian.txt","w");  
	if (fp == NULL) {

	}
	while (current != NULL) {
		fwrite(current,sizeof(struct ms),1,fp);    //二进制依次写入
		current = current->next;
	} 

	fclose(fp);
}


}

void cun2() {
	struct telian * current =  head2->next;
	FILE * fp;
	fp = fopen("laolian.txt","w");
	if (fp == NULL) {

	}
	while (current != NULL) {
		fwrite(current,sizeof(struct tems),1,fp);
		current = current->next;
	} 

	fclose(fp);
}


void du() {
	struct lian  * prev = head, * current =  NULL;
	int shu = 0;
	FILE * fp;
	fp = fopen("shu.txt","r");  
	/* 这个文件存的是链表节点,也就是成员的数量,因为fread读取时要输入数量,所以单独存一下。*/
		if (fp == NULL) {
		/*这里写打开失败的提示*/
	}
	fscanf(fp,"%d",&shu);
	fclose(fp); 
	fp = fopen("lian.txt","r");    // 打开存链表的文件
	
	if (fp == NULL) {
		/* 打开失败的提示 */
	}
	while(shu--){                                            //  读的次数。
		current = (struct lian *)malloc(sizeof(struct lian));//  添加节点。
	    fread(current,sizeof(struct ms),1,fp);               //  读取,一次读取一个节点。
	    prev->next = current; 
	    current->next = NULL;
	    prev = current;
	}
	fclose(fp);
}


5. 加密

这里的管理员不能从程序里更改,需要自行更改文件目录里的文件。
一个简单的根据ASCII码加密解密的函数

注: 自己实现的并不是很规范,可以去网上了解MD5加密算法。

void jiami(char x[]) {
	for (int i = 0; i < strlen(x); i++) {
		x[i] = x[i] + 10 + i; 
	}
	
}
void jiemi(char x[]) {
	for (int i = 0; i < strlen(x); i++) {
		x[i] = x[i] - 10 - i; 
	}
}

单独写一个程序跑一下加密的函数,然后把结果复制到文件夹里的对应文件里就行了。

账号密码用空格隔开

这里管理员默认账号密码都为admin

附录

github-项目代码

一、学籍管理系统 1.问题提出 为了分析教学效果并进行相应的学籍处理,各学校每到期末都对综合成绩进行分类统计、上报成绩汇总结果等,这给每位老师和学籍管理人员带来很大工作量。使用学籍管理系统可以减少工作者的工作负担。 2.功能要求 循环显示如图7(a)所示的主菜单。 ………………………………………………….        .………………………………………………………        …………………………………………………………. 请输入选项编号(0 ~4):.        . 请输入选项编号(0 ~4):.        .请输入选项编号(0 ~4):. ………………………………………………….        .………………………………………………………        …………………………………………………………. . 1——录入成绩.                . 1——按学号排序.               . 1——生成并打印补考通知单 . . 2——统计成绩.                . 2——计算平均分并排序 .         . 2——生成并打印退学通知单 . . 3——处理学籍.                . 3——统计分数段.               . 3——生成并打印新名册. . 4——创新功能.                . 4——返回.                    . 4——返回. . 0——退出系统.                . 0——退出系统.                 . 0——退出系统. ……………………………………………………         ……………………………………………………….        …………………………………………………………. 图7(a) 学籍管理主菜单            (b)成绩统计子菜单               (c)学籍处理子菜单 在主菜单中选择1:录入成绩。假设某班的原始成绩形式如下: 学号        姓名        高数    英语    物理    编程    马哲 0909339105 Huangying  89      92      85     88     82 0909339102 Zhangchen  72      68      83     90     78 0909339108 Linan      91      84      90     79     81 …… 其中,原始数据的排列是无序的。系统应能够保留原始成绩单。 在主菜单中选择2:进入如图7(b)所示的子菜单,并统计成绩。在此可以计算平均分、统计各分数段、按学号排序、按平均分排序。在子菜单选择1时,将该班学生的成绩按学号升序排序后的顺序存入std.dat文件中,以方便打印。在子菜单选择2时,求出每位学生的平均分,并按平均分从高到低的顺序进行排序后,写入文件sort.dat中。在子菜单选择3时,统计出各门课、各分数段学生的人数,并如表1所示的形式输出。在子菜单选择4时,返回主菜单;选择0,退出整个系统。 在主菜单选择3:进入如图7(c)所示的子菜单,并处理学籍。为不及格的学生打印重考通知单时,应在通知单上写明重考的课程、时间和地点(由键盘输入)。为了参加多门课重考的学生完成补考,应避免时间重复。为3门以上不及格的学生打印退学通知单。通知单上应写明学生的学号、姓名、退学原因(几门课不及格,成绩多少)。删除已退学的学生信息,将该班中升入高年级的学生信息写入文件next.dat中。在子菜单选择4,返回主菜单;选择0,退出整个系统。 在主菜单中选择4:此项功能学生自由发挥,根据本组爱好增加与本题止相关的新功能。 在主菜单中选择0:显示结束信息(如“感谢使用本软件!已正常退出,按任意键结束。”),按任意键后,退出本功能。 表1 各门课、各分数段学生的人数 ……………………………………………………………………………………………………………….   . 90以上. 80~89分.70~79分.60~69分.60分以下. 高数.      .       .       .       .       . 英语.      .       .       .       .       . 物理.      .       .       .       .       . 编程.      .       .       .       .       . 马哲.      .       .       .       .       . ……………………………………………………………………………………………………………….
评论 20
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值