C-学生管理系统

目的

通过完成一个简单的学生管理系统,达到掌握文件操作,动态分配内存,单链表的作用

技术

文件操作,动态分配内存,单链表

如何使用

1.文件操作

打开文件函数:

FILE* fopen(const char* fileName,const char* openMode);
  • fileName是文件地址,openMode是打开方式
  • r:是读一个已有的文件
  • w:是写,没有文件会创建,有了会将内容清空,从头开始写
  • a:和w类似,但是在文本末尾追加写
  • r+,w+,a+:可读可写

关闭文件函数:

int fclose(FILE* pointerOnFile);

写、读数据块数据到文件函数:

fwrite(buffer,size,count,fp);

 fread(buffer,size,count,fp);
  • buffer -- 是一个指针,在fread函数中,它表示存放输入数据的首地址。在fwrite函数中,它表示存放输出数据的首地址。
  • size -- 表示数据库的字节数
  • count -- 表示要读写的数据库块数
  • fp -- 表示文件指针

获得当前游标位置的函数:

long ftell(FILE* pointerOnFile);

游标移动函数:

int fseek(FILE* pointerOnFile, long move, int origin);
  • 使游标在文件(pointerOnFile指针所指)中从位置(origin所指)开始移动一段距离(move所指)

  • move:可以是一个正整数,表明向前移动;0,不移动;负整数,表明回退

  • origin:它有三个取值,SEEK_SET:文件开始处,SEEK_CUR:游标当前所处位置,SEEK_END:文件末尾


//加载数据
void loadData(Student *pTemp){

    //打开文件
    FILE *fp = fopen(FILE_PATH,"r");

    //循环读取数据
    int SEEK_MOVE = 0;
    while (1){

        //准备一个结构体 保存数据
        Student *pStu = (Student *)malloc(1 * sizeof(Student));

        //移动游标
        fseek(fp,SEEK_MOVE,SEEK_SET);

        //读取数据
        int result = fread(pStu,sizeof(Student),1,fp);
        //如果读到文件末尾
        if (result == 0){
            break;
        }
        //printf("id:%d 姓名:%s 成绩:%f \n",pStu->s_ID,pStu->name,pStu->score);

        //链接
        pTemp->next = pStu;
        pTemp = pTemp->next;

        //为下次移动确定位置
        SEEK_MOVE = ftell(fp);
    }

    //关闭文件
    fclose(fp);

    printf("加载数据成功\n");
}

上面的函数主要用来读取存放在文件里面的一个个Student类型的结构体,并将它们依次链接到头结点之后。

2.动态分配内存

  • 需要导入<stdlib.h>头文件,才可以调用malloc,realloc,free函数

  • void *malloc(int num);函数在堆区分配一定的内存。

  • void *realloc(void *address, int newsize);在malloc分配的基础上重新分配内存。

  • void free(void *address);释放掉手动分配的内存。

  • void *类型表示未确定类型的指针,可以通过类型转换强制转换为任何其它类型的指针。

Student *pTemp = (Student *)malloc(1 * sizeof(Student));

上面的例子就是动态分配了一个Student结构体类型的变量。

3.单链表

链表的基本结构

头结点初始化情况
//初始化一个头结点
void initHeader(Student **pHeader){
    //动态分配内存
    Student *pTemp = (Student *)malloc(1 * sizeof(Student));

    //初始化
    pTemp->next = NULL;

    //改变外部的值
    *pHeader = pTemp;
}

上面函数的功能是得到一个头结点。

尾插一个结点
void insert(Student *pTemp){
    //准备一个结构体 保存数据
    Student *pStu = (Student *)malloc(1 * sizeof(Student));
    pStu->next = NULL;

    //提示用户输入
    printf("请输入姓名:");
    scanf("%s",pStu->name);
    printf("请输入成绩:");
    scanf("%f",&(pStu->score));

    //自动编号
    int count = 0;
    while (pTemp->next != NULL){
        pTemp = pTemp->next;
        count++;
    }
    pStu->s_ID = count+1;

    //插入
    pTemp->next = pS
}

上面函数的功能是在链表的最后添加一个结点。

删除一个结点



//删除数据
void deleteInfo(Student *pTemp){

    printf("请输入要删除的学生ID:");
    int id = 0;
    scanf("%d",&id);
    
    //找到要删除的那个结点的上一个结点
    while (pTemp->next->s_ID != id ){

        //推到下一个结点
        pTemp = pTemp->next;

        if (pTemp->next == NULL){
            printf("没有该学生\n");
            return;
        }
    }

    //更改链接关系
    Student *temp = pTemp->next;
    pTemp->next = temp->next;

    //释放内存
    free(temp);

    printf("删除数据成功\n");
}

上面函数的功能是根据结点数据域的值删除对应的结点。

修改某个结点的数据域的值
//修改数据
void updateInfo(Student *pTemp){
    printf("请输入要修改的学生ID:");
    int id = 0;
    scanf("%d",&id);

    //找到第一个数据结点
    pTemp = pTemp->next;

    //找到要删除的那个结点
    while (pTemp->s_ID != id ){
        //推到下一个结点
        pTemp = pTemp->next;

        if (pTemp == NULL){
            printf("没有该学生\n");
            return;
        }
    }

    printf("请输入修改后的学生名字:");
    scanf("%s",pTemp->name);
    printf("请输入修改后的学生分数:");
    scanf("%f",&pTemp->score);

    printf("修改数据成功\n");
}

上面函数的功能是根据结点数据域的值找到对应的结点,修改其数据域的值。

具体使用

#include <stdio.h>
#include <stdlib.h>

//文件路径
#define FILE_PATH "C:/Users/a2867/Desktop/S_T.txt"

//定义一个学生的结构体结点
typedef struct Node{
    char name[10];
    int s_ID;
    float score;
    struct Node *next;
}Student;

//初始化一个头结点
void initHeader(Student **pHeader);

//操作界面
void showMenu();

//获取操作序列
int getChoice();

//加载数据
void loadData(Student *pTemp);

//退出程序-保存数据
void exitS_T(int status,Student *pTemp);

//插入数据
void insert(Student *pTemp);

//查询数据
void querry(Student *pTemp);

//删除数据
void deleteInfo(Student *pTemp);

//修改数据
void updateInfo(Student *pTemp);


#include "S_T.h"

int main(){

    //定义一个指针 记录头结点
    Student *pHeader = NULL;

    //初始化头结点
    initHeader(&pHeader);

    //加载数据
    loadData(pHeader);
    
    //操作界面
    int choice = 0;
    while (1){

        //显示操作
        showMenu();

        //选择
        choice = getChoice();

        //具体操作
        switch (choice){

            case 1:
                //查询
                querry(pHeader);
                break;
            case 2:
                //删除
                deleteInfo(pHeader);
                break;
            case 3:
                //修改
                updateInfo(pHeader);
                break;
            case 4:
                //插入
                insert(pHeader);
                printf("插入成功!\n");
                break;
            default:
                exitS_T(EXIT_SUCCESS,pHeader);
                break;
        }

        //清除输入缓冲区
        fflush(stdin);
        getchar();

        //清屏
        system("CLS");

    }

    return 0;
}

//初始化一个头结点
void initHeader(Student **pHeader){
    //动态分配内存
    Student *pTemp = (Student *)malloc(1 * sizeof(Student));
    //初始化
    pTemp->next = NULL;
    //改变外部的值
    *pHeader = pTemp;
}

//操作界面
void showMenu(){
    printf("************\n");
    printf("1.查询\n");
    printf("2.删除\n");
    printf("3.更改\n");
    printf("4.插入\n");
    printf("5.退出\n");
    printf("************\n");
}

//获取操作序列
int getChoice(){

    int choice;

    printf("请选择操作:");
    scanf("%d", &choice);
    
    return choice;
}

//加载数据
void loadData(Student *pTemp){

    //打开文件
    FILE *fp = fopen(FILE_PATH,"r");

    //循环读取数据
    int SEEK_MOVE = 0;
    while (1){

        //准备一个结构体 保存数据
        Student *pStu = (Student *)malloc(1 * sizeof(Student));

        //移动游标
        fseek(fp,SEEK_MOVE,SEEK_SET);

        //读取数据
        int result = fread(pStu,sizeof(Student),1,fp);
        //如果读到文件末尾
        if (result == 0){
            break;
        }
        //printf("id:%d 姓名:%s 成绩:%f \n",pStu->s_ID,pStu->name,pStu->score);

        //链接
        pTemp->next = pStu;
        pTemp = pTemp->next;

        //为下次移动确定位置
        SEEK_MOVE = ftell(fp);
    }

    //关闭文件
    fclose(fp);

    printf("加载数据成功\n");
}

//退出程序
void exitS_T(int status,Student *pTemp){
    
    //创建、打开文件,从头写
    FILE *fp = fopen(FILE_PATH,"w");

    //保存数据
    while (pTemp->next){

        //找到下一个
        pTemp = pTemp->next;

        //保存当前节点结构体到文件中
        fwrite(pTemp,sizeof(Student),1,fp);
        //printf("id:%d 姓名:%s 成绩:%f \n",pTemp->s_ID,pTemp->name,pTemp->score);  
    }

    //关闭文件
    fclose(fp);

    printf("保存数据成功 感谢你的使用\n");
    
    //退出
    exit(status);
}

//插入数据
void insert(Student *pTemp){
    //准备一个结构体 保存数据
    Student *pStu = (Student *)malloc(1 * sizeof(Student));
    pStu->next = NULL;

    //提示用户输入
    printf("请输入姓名:");
    scanf("%s",pStu->name);
    printf("请输入成绩:");
    scanf("%f",&(pStu->score));

    //自动编号
    int count = 0;
    while (pTemp->next != NULL){
        pTemp = pTemp->next;
        count++;
    }
    pStu->s_ID = count+1;

    //插入
    pTemp->next = pStu;
}

//查询数据
void querry(Student *pTemp){

    //找到第一个数据结点
    pTemp = pTemp->next;

    //循环打印数据
    printf("         学生数据\n");
    while (pTemp != NULL){

        //输出
        printf("ID:%d ", pTemp->s_ID);
        printf("name:%s ", pTemp->name);
        printf("score:%.1f",pTemp->score);
        printf("\n");

        //下一个
        pTemp = pTemp->next;
    }
}

//删除数据
void deleteInfo(Student *pTemp){

    printf("请输入要删除的学生ID:");
    int id = 0;
    scanf("%d",&id);
    
    //找到要删除的那个结点的上一个结点
    while (pTemp->next->s_ID != id ){

        //推到下一个结点
        pTemp = pTemp->next;

        if (pTemp->next == NULL){
            printf("没有该学生\n");
            return;
        }
    }

    //更改链接关系
    Student *temp = pTemp->next;
    pTemp->next = temp->next;

    //释放内存
    free(temp);

    printf("删除数据成功\n");
}

//修改数据
void updateInfo(Student *pTemp){
    printf("请输入要修改的学生ID:");
    int id = 0;
    scanf("%d",&id);

    //找到第一个数据结点
    pTemp = pTemp->next;

    //找到要删除的那个结点
    while (pTemp->s_ID != id ){
        //推到下一个结点
        pTemp = pTemp->next;

        if (pTemp == NULL){
            printf("没有该学生\n");
            return;
        }
    }

    printf("请输入修改后的学生名字:");
    scanf("%s",pTemp->name);
    printf("请输入修改后的学生分数:");
    scanf("%f",&pTemp->score);

    printf("修改数据成功\n");
}

运行结果

1.查询数据,插入一条数据,保存数据

2.查询数据(检查上次插入的数据是否保存),删除ID为4的数据,修改ID为3的数据,保存数据

3.查询数据(检查上次的删除,上次的修改是否保存)

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容

  • 题目类型 a.C++与C差异(1-18) 1.C和C++中struct有什么区别? C没有Protection行为...
    阿面a阅读 7,657评论 0 10
  • 一些概念 数据结构就是研究数据的逻辑结构和物理结构以及它们之间相互关系,并对这种结构定义相应的运算,而且确保经过这...
    Winterfell_Z阅读 5,767评论 0 13
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,096评论 1 32
  • 内存是计算机非常关键的部件之一,是暂时存储程序以及数据的空间,CPU只有有限的寄存器可以用于 存储计算数据,而大部...
    dreamer_lk阅读 1,198评论 2 10
  • 转自:http://blog.csdn.net/oreo_go/article/details/52116214 ...
    YYT1992阅读 974评论 0 4