问题描述
对于任意给定的单词小助手程序(现有功能可以实现查单词的增删改查、中英、英中测试和分数显示功能),完善其单词查询、错误单词重复记忆和排行功能。
问题分析
首先附上给出的初始代码(已经可以实现基本功能)
#include <stdio.h> #include <string.h> #include <stdlib.h> #define MAX_CHAR 20 // 最大字符 #define MAX_NUM 200 // 单词的最大个数 struct word //单词的结构体 { char en[MAX_CHAR]; // 英文形式 char ch[MAX_CHAR]; //中文形式 } s[MAX_NUM]; //单词数组 int num; //单词个数 int select=1;//select 为是否退出系统的标记 int d=0,c=0; //帮助 void help() { printf("\n本系统主要实现英语单词学习的功能。用户可对词典文件中的单词进行预览,增删改查。"); printf("\n同时还可进行中英、英中测试。本系统还提供了测试成绩的显示功能。"); } //从文件中读取单词的信息 void readfile() { FILE *fp; int i=0; fp=fopen("data.txt","r"); if(!fp) { printf("\n打开文件data.txt失败!"); } while(fscanf(fp,"%s %s ",s[i].en,s[i].ch)==2) { i++; } num=i; if(0==i) printf("\n文件为空,请选择词典维护增加词条!"); else printf("\n"); fclose(fp); } //从文件中读取单词的信息 void writefile() { FILE *fp; int i=0; fp=fopen("data.txt","w"); if(!fp) { printf("\n打开文件data.txt失败!"); } for(i=0;i<num;i++) { fprintf(fp,"\n%s %s ",s[i].en,s[i].ch); } printf("\n"); fclose(fp); } void sort()/*按字典排序*/ { int i,j; char temp[MAX_CHAR]; for(i=0;i<num-1;i++) { for(j=num-1;j>i;j--) if(strcmp(s[j-1].en,s[j].en)>0) { strcpy(temp,s[j-1].en); strcpy(s[j-1].en,s[j].en); strcpy(s[j].en,temp); strcpy(temp,s[j-1].ch); strcpy(s[j-1].ch,s[j].ch); strcpy(s[j].ch,temp); } } } //添加单词信息 void add() { int i=num,j,flag=1; while(flag) { flag=0; printf("\n请输入单词的英文形式:"); scanf("%s",s[i].en); for(j=0;j<i;j++) if(strcmp(s[i].en,s[j].en)==0) { printf("已有该单词,请检查后重新录入!\n"); flag=1; break; /*如有重复立即退出该层循环,提高判断速度*/ } } printf("\n请输入单词的中文形式:"); scanf("%s",s[i].ch); num++; printf("\n您输入的信息为: 英文: %s 中文: %s ",s[i].en,s[i].ch); sort(); } //删除单词信息 void del() { int i=0,j=0; char en[MAX_CHAR]; //英文形式 printf("\n请输入你要删除的单词英文形式:"); scanf("%s",en); for(i=0;i<num;i++)//先找到该英文形式对应的序号 if(strcmp(s[i].en,en)==0) { for(j=i;j<num-1;j++) s[j]=s[j+1]; num--; //数量减少 1 return; } printf("\n没有这个单词!"); } //修改单词信息 void modify() { int i=0,choose=0,flag=1;//chooses代表选项标识,flag代表是否找到单词 char en[MAX_CHAR]; //英文形式 while(flag||choose) { printf("\n请输入你要修改的单词英文形式:"); scanf("%s",en); for(i=0;i<num;i++)//先找到该英文形式对应的序号 if(strcmp(s[i].en,en)==0) { printf("\n请输入单词正确的英文形式:"); scanf("%s",s[i].en); printf("\n请输入此单词正确的的中文形式:"); scanf("%s",s[i].ch); printf("\n继续修改请选1,返回上一级请选0:"); scanf("%d",&choose); if(choose==0) return; } flag=0; } if(!flag) printf("\n没有这个单词!"); } //单词预览 void show() { int i=0; printf("\n单词: 英文 中文 "); for(i=0;i<num;i++) printf("\n %-12s%-12s",s[i].en,s[i].ch); } //查询单词 void search() { int i=0,choose=0,flag=1; char ch[MAX_CHAR]; //中文形式 while(choose||flag) { printf("\n请输入你要查询的单词中文形式:"); scanf("%s",ch); for(i=0;i<num;i++)//先找到该中文形式对应的序号 if(strcmp(s[i].ch,ch)==0) { printf("\n英文形式 中文形式 "); printf("\n %-12s%12s",s[i].en,s[i].ch); printf("\n继续查询请选1,返回上一级请选0:"); scanf("%d",&choose); if(choose==0) return; } flag=0; } if(!flag) printf("\n没有这个单词!"); } //中译英测试 void zytest() { char b1[20]; int z; int choose=1; int i; while(choose) { i = rand()%num; printf("\n【%s】请输入英文单词:",s[i].ch); scanf("%s",b1); for(z=0;strcmp(b1,s[i].en)!=0;z=z) { printf("\n输入错误!!请重新输入:");scanf("%s",b1);c=c+1;} printf("\n恭喜你,回答正确,加10分!\n\n");d=d+1; printf("\n继续测试请选1,返回上一级请选0:"); scanf("%d",&choose); if(choose==0) return; } } //英译中测试 void yztest() { char b1[20]; int z,x=41; int choose=1; int i; i = rand()%num; while(choose) { printf("【%s】请输入中文意思:",s[i].en); scanf("%s",b1); for(z=0;strcmp(b1,s[i].ch)!=0;z=z) { printf("输入错误!!请重新输入:");scanf("%s",b1);c=c+1;} printf("\n恭喜你,回答正确,加10分!\n\n");d=d+1; printf("\n继续测试请选1,返回上一级请选0:"); scanf("%d",&choose); if(choose==0) return; } } //成绩列表 void list() { printf("\n 共计输入错误:%d次 **每次扣10分**\n",c); printf(" 共计输入正确:%d次 **每次加10分**\n",d); printf(" 你的总得分为:%d分\n\n",10*d-10*c); } //词典维护 void maintain() { int choose;//维护功能选择 printf(" ------------------\n"); printf(" 1.增加单词\n"); printf(" 2.修改单词\n"); printf(" 3.删除单词\n"); printf(" 4.查询单词\n"); printf(" 5.退出本菜单\n"); printf(" ------------------\n"); while(1) { printf(" \n请输入维护功能编号:"); scanf("%d",&choose); switch(choose) { case 1: add();writefile();break; case 2: modify();writefile();break; case 3: del();writefile();break; case 4: search();break; case 5: return; default: printf("\n请在1-5之间选择"); } } } //用户界面 void menu() { int item; printf("\n"); printf(" *********************************************************\n"); printf("# #\n"); printf("# 英语单词小助手 #\n"); printf("# #\n"); printf("# 版本 : v1.0 #\n"); printf("# #\n"); printf(" *********************************************************\n"); printf("# #\n"); printf("# 0.词库维护 1.单词预览 #\n"); printf("# #\n"); printf("# 2.单词背诵(中英) 3.单词背诵(英中) #\n"); printf("# #\n"); printf("# 4.查询成绩 5.帮助 #\n"); printf("# #\n"); printf("# 6.退出系统 #\n"); printf("# #\n"); printf(" **********************************************************\n"); printf("\n"); printf(" 请选择您需要的操作序号(0-5)按回车确认:"); scanf("%d",&item); printf("\n"); readfile(); switch(item) { case 0: maintain();break; case 1: show();break; case 2: zytest();break; case 3: yztest(); break; case 4: list();break; case 5: help();break; case 6: select =0;break; default: printf("请在0-6之间选择\n"); } } int main() { while(select) { menu(); } system("pause"); return 0; }
感兴趣的用户可以自己运行测试,在这里仅附上一张主界面图,其余暂不赘述~
关键算法构造
在这里参考了一些学长学姐写过的博客...
1>单词查询
由于这里要求在原有程序基础上添加显示同样中文意义,不同的英文单词的功能,因此我将查询函数分设立两个子模块:汉语查询、英语查询以解决显示同样中文意义,不同的英文单词和一词多义的情况。具体思想为:设置一个记录遍历单词数组的下标变量k初值为-1,在每次遍历之前置位-1,遍历的过程中如果遇到一次多译或一意多词的情况就更新下标变量k的值,以此录入多条单词意思,如果遍历结束k仍未-1就没有找到单词。
这里我以中文查询为例:
//查中文 void ChSearch() { char ch[MAX_CHAR]; //中文形式 int choose=1; int flag; int k=-1;//查询到当前单词的中文下标 while(choose==1) { //查找下一个单词时重置标志变量 k 和 flag k=-1; flag = 1; printf("输入要查询的单词的中文形式:"); scanf("%s",ch); for(int i=0;i<num;i++)//遍历单词中文形式 { if(strcmp(s[i].ch,ch)==0 && k==-1) { printf("\n 英文 中文 "); printf("\n %-12s%12s",s[i].en,s[i].ch); k=i;//记录当前下标 } else if(strcmp(s[i].ch,ch)==0)//遍历到一译多词 { printf("\n %-12s%12s",s[i].en,s[i].ch); k=i;//更新下标 } } if(k==-1)//没有找到单词的情况 { flag = 0; printf("啊哦,没有找到当前单词哦~"); } printf("\n1.继续查询2.返回上一级:"); scanf("%d",&choose); } }
2>错词重记
错词结构体定义:
//错误单词的结构体 struct wrongWord { char en[MAX_CHAR]; char ch[MAX_CHAR]; } w[MAX_NUM];//错误单词数组
依照个人理解,这里应该和单词背诵模块大同小异~建立一个错词结构体,把单词背诵过程中的错词记录下来,依次存到错词数组里就好~
3>用户排行
用户信息结构体:
//用户信息结构体 typedef struct user { char name[MAX_CHAR]; int score; }user; user users[MAX_USER];//用户结构体数组 user nowuser;
再定义一个用户信息的结构体,规定用户上限,针对不同的用户存储不同的分数,在排行榜里展示~(时间有限,这里还没有实现)
运行结果
程序源代码
这里仅供参考,因为还有部分功能没有完善......
#include <stdio.h> #include <string.h> #include <stdlib.h> #include<time.h> #define MAX_CHAR 20 // 最大字符 #define MAX_NUM 200 // 单词的最大个数 #define MAX_USER 5 //用户的最大个数 struct word //单词的结构体 { char en[MAX_CHAR]; // 英文形式 char ch[MAX_CHAR]; //中文形式 } s[MAX_NUM]; //单词数组 //错误单词的结构体 struct wrongWord { char en[MAX_CHAR]; char ch[MAX_CHAR]; } w[MAX_NUM];//错误单词数组 //用户信息结构体 typedef struct user { char name[MAX_CHAR]; int score; }user; user users[MAX_USER];//用户结构体数组 user nowuser; int num;//单词个数 int select=1;//select 为是否退出系统的标记 int d=0,c=0;//c统计输入错误次数 d统计输入正确次数 //函数声明 void ReciteAllVocabulary(); void ReciteWrongVocabulary(); //读文件(单词词库) void ReadVocabularyFile() { FILE *fp; int i=0; fp=fopen("vocabulary.txt","r"); if(!fp) { printf("\n打开文件vocabulary.txt失败!"); } while(fscanf(fp,"%s %s ",s[i].en,s[i].ch)==2)//? { i++; } num=i; if(i==0) printf("\n哎呀,词库空空如也~,请先选择词典维护增加词条哦!\n"); else printf("\n"); fclose(fp); } //读文件(错词词库) void ReadWrongVocabularyFile() { FILE *fp; int i = 0; int item= 0; fp = fopen("wrongvocabulary.txt","r"); if(!fp) { printf("\n打开文件wrongVocabulary.txt失败!"); } while(fscanf(fp,"%s %s ",w[i].en,w[i].ch)==2)//? { i++; } num=i; if(i==0) { printf("\n你的错词库当前没有单词~请先选择单词测试查漏补缺哦!\n"); printf("1.单词测试 2.返回上一级 3.退出系统\n"); scanf("%d",&item); switch(item) { case 1: ReciteAllVocabulary(); break; case 2: return; case 3: exit(0); } } else { printf("\n"); } fclose(fp); } //写文件(单词词库) void WriteVocabularyFile() { FILE *fp; int i=0;//? fp=fopen("vocabulary.txt","w"); if(!fp) { printf("\n打开文件vocabulary.txt失败!"); } for(i=0;i<num;i++) { fprintf(fp,"\n%s %s ",s[i].en,s[i].ch); } printf("\n"); fclose(fp); } //写文件(错词库) void WriteWrongVocabularyFile() { FILE *fp; int i=0; fp=fopen("wrongvocabulary.txt","w"); if(!fp) { printf("\n打开文件wrongvocabulary.txt失败!"); } for(i=0;i<num;i++) { fprintf(fp,"\n%s %s ",w[i].en,w[i].ch); } fclose(fp); printf("\n"); } //词库单词测试(中英+英中) void ReciteAllVocabulary() { int right = 0;//标志变量 答对right记为1 答错right置为0 char b1[20]; int choose=1; srand((unsigned int)time(NULL)); int i=rand()%num; int j=0; int n;//由n产生一个1或2的随机数 1对应英译中测试 2对应中译英测试 while(choose) { right=0; n = rand()%2+1; i = rand()%num; if(right==0) { if(n==1)//英译中 { printf("\n【%s】请输入中文意思:",s[i].en); scanf("%s",b1); if(strcmp(b1,s[i].ch)!=0)//输入的单词和词库的单词不相同 { printf("\n你记错啦,要加油哦!"); right = 0; printf("\n正确形式:%s%5s",s[i].en,s[i].ch); strcpy(w[j].ch,s[j].ch);//将记错的单词(中文形式)录入错词库 strcpy(w[j].en,s[j].en);//将记错的单词(英文形式)录入错词库 } else right =1; } else if(n==2)//中译英 { printf("\n【%s】请输入英文单词:",s[i].ch); scanf("%s",b1); if(strcmp(b1,s[i].en)!=0) { printf("\n你记错啦,要加油哦!\n"); printf("\n正确形式:%s%5s",s[i].en,s[i].ch); right =0; strcpy(w[j].ch,s[j].ch); strcpy(w[j].en,s[j].en); } else right=1; } j++; } if(right==1) { printf("\n恭喜你,回答正确,加10分!\n\n"); right = 1; } printf("\n继续测试请选1,返回上一级请选0:"); scanf("%d",&choose); } if(choose==0) return; WriteWrongVocabularyFile(); } void Sort()/*按字典排序*/ { int i,j; char temp[MAX_CHAR]; for(i=0;i<num-1;i++) { for(j=num-1;j>i;j--) if(strcmp(s[j-1].en,s[j].en)>0) { strcpy(temp,s[j-1].en); strcpy(s[j-1].en,s[j].en); strcpy(s[j].en,temp); strcpy(temp,s[j-1].ch); strcpy(s[j-1].ch,s[j].ch); strcpy(s[j].ch,temp); } } } //添加单词信息 void Add() { int i=num,j,flag=1;//num,i为单词总个数 while(flag) { flag=0; printf("\n请输入单词的英文形式:"); scanf("%s",s[i].en); for(j=0;j<i;j++) if(strcmp(s[i].en,s[j].en)==0)//比较当前录入的单词在现有词库中是否已经存在 { printf("已有该单词,请检查后重新录入!\n"); flag=1; break; //如有重复立即退出该层循环,提高判断速度 } } printf("\n请输入单词的中文形式:"); scanf("%s",s[i].ch); num++; printf("\n您输入的信息为: 英文: %s 中文: %s ",s[i].en,s[i].ch); Sort(); } //删除单词信息 void del() { int i=0,j=0; char en[MAX_CHAR]; //英文形式 printf("\n请输入你要删除的单词英文形式:"); scanf("%s",en); for(i=0;i<num;i++)//先找到该英文形式对应的序号 if(strcmp(s[i].en,en)==0) { for(j=i;j<num-1;j++) s[j]=s[j+1]; num--; //数量减少 1 return; } printf("\n没有这个单词!"); } //修改单词信息 void Modify() { int i=0,choose=0,flag=1;//chooses代表选项标识,flag代表是否找到单词 char en[MAX_CHAR]; //英文形式 while(flag||choose) { printf("\n请输入你要修改的单词英文形式:"); scanf("%s",en); for(i=0;i<num;i++)//先找到该英文形式对应的序号 { if(strcmp(s[i].en,en)==0) { printf("\n请输入单词正确的英文形式:"); scanf("%s",s[i].en); printf("\n请输入此单词正确的的中文形式:"); scanf("%s",s[i].ch); printf("\n继续修改请选1,返回上一级请选0:"); scanf("%d",&choose); if(choose==0) return; } } flag=0; } if(!flag) printf("\n没有这个单词!"); } //单词预览 void ShowAllVocabulary() { int i=0; printf("\n 英文 中文 "); for(i=0;i<num;i++) { printf("\n %-12s%-12s",s[i].en,s[i].ch); } } //错词预览 void ShowWrongVocabulary() { int i=0; printf("\n 英文 中文 "); for(i=0;i<num;i++) { printf("\n %-12s%-12s",w[i].en,w[i].ch); } } //查中文 void ChSearch() { char ch[MAX_CHAR]; //中文形式 int choose=1; int flag; int k=-1;//查询到当前单词的中文下标 while(choose==1) { //查找下一个单词时重置标志变量 k 和 flag k=-1; flag = 1; printf("输入要查询的单词的中文形式:"); scanf("%s",ch); for(int i=0;i<num;i++)//遍历单词中文形式 { if(strcmp(s[i].ch,ch)==0 && k==-1) { printf("\n 英文 中文 "); printf("\n %-12s%12s",s[i].en,s[i].ch); k=i;//记录当前下标 } else if(strcmp(s[i].ch,ch)==0)//遍历到一译多词 { printf("\n %-12s%12s",s[i].en,s[i].ch); k=i;//更新下标 } } if(k==-1)//没有找到单词的情况 { flag = 0; printf("啊哦,没有找到当前单词哦~"); } printf("\n1.继续查询2.返回上一级:"); scanf("%d",&choose); } } void EnSearch() { char en[MAX_CHAR]; //英文形式 int choose=1; int flag; int k=-1;//查询到当前单词的英文下标 while(choose==1) { //查找下一个单词时重置标志变量 k 和 flag k=-1; flag = 1; printf("输入要查询的单词的英文形式:"); scanf("%s",en); for(int i=0;i<num;i++)//遍历单词英文形式 { if(strcmp(s[i].en,en)==0 && k==-1) { printf("\n 英文 中文 "); printf("\n %-12s%12s",s[i].en,s[i].ch); k=i;//记录当前下标 } else if(strcmp(s[i].en,en)==0)//遍历到一词多译 { printf("\n %-12s%12s",s[i].en,s[i].ch); k=i;//更新下标 } } if(k==-1)//没有找到单词的情况 { flag = 0; printf("啊哦,没有找到当前单词哦~"); } printf("\n1.继续查询2.返回上一级:"); scanf("%d",&choose); } } //查询单词 void Search() { int i=0; char en[MAX_CHAR]; //英文形式 int choice; while(1) { printf("1.查中文2.查英文3.返回上一级:"); scanf("%d",&choice); switch(choice) { case 1: ChSearch(); break; case 2: EnSearch(); case 3: return; } } } //背诵错误的单词 void ReciteWrongVocabulary() { ReadWrongVocabularyFile(); int right = 0;//标志变量 答对right记为1 答错right置为0 char b1[20]; int choose=1; int i; int j=0; i = rand()%num; srand((unsigned int)time(NULL)); int n;//由n产生一个1或2的随机数 1对应英译中测试 2对应中译英测试 while(choose) { right=0; n = rand()%2+1; i = rand()%num; if(right==0) { if(n==1)//英译中 { printf("\n【%s】请输入中文意思:",s[i].en); scanf("%s",b1); if(strcmp(b1,w[i].ch)!=0)//输入的单词和词库的单词不相同 { printf("\n你记错啦,要加油哦!"); right = 0; } else right =1; } else if(n==2)//中译英 { printf("\n【%s】请输入英文单词:",s[i].ch); scanf("%s",b1); if(strcmp(b1,w[i].en)!=0) { printf("\n你记错啦,要加油哦!\n"); right =0; } else right=1; } j++; } if(right==1) { printf("\n恭喜你,回答正确,加10分!\n\n"); right = 1; } printf("\n继续测试请选1,返回上一级请选0:"); scanf("%d",&choose); } if(choose==0) return; WriteWrongVocabularyFile(); } //成绩列表 void List() { printf("--------------------英雄榜--------------------\n"); printf("\n 共计输入错误:%d次 **每次扣10分**\n",c); printf(" 共计输入正确:%d次 **每次加10分**\n",d); printf(" 你的总得分为:%d分\n\n",10*d-10*c); } //帮助 void Help() { printf("\n本系统主要实现英语单词学习的功能。用户可对词典文件中的单词进行预览,增删改查。"); printf("\n同时还可进行中英、英中测试。本系统还提供了测试成绩的显示功能。"); } //英雄榜 void Rank() { printf("该系统尚未开放!"); } //词典维护 void Maintain() { int choose;//维护功能选择 printf(" ------------------\n"); printf(" 1.增加单词\n"); printf(" 2.修改单词\n"); printf(" 3.删除单词\n"); printf(" 4.查询单词\n"); printf(" 5.退出本菜单\n"); printf(" ------------------\n"); while(1) { printf(" \n请输入维护功能编号:"); scanf("%d",&choose); switch(choose) { case 1: Add(); WriteVocabularyFile(); break; case 2: Modify(); WriteVocabularyFile(); break; case 3: del(); WriteVocabularyFile(); break; case 4: Search(); break; case 5: return; default: printf("\n请在1-5之间选择"); } } } //用户界面 void Menu() { int item; printf("\n"); printf("*********************************************************\n"); printf("# #\n"); printf("# 英语单词小助手 #\n"); printf("# #\n"); printf("# 版本 : v1.1 #\n"); printf("# #\n"); printf(" ********************************************************\n"); printf("# #\n"); printf("# 0.词库维护 1.单词预览 #\n"); printf("# #\n"); printf("# 2.单词背诵 3.强化记忆 #\n"); printf("# #\n"); printf("# 4.查询成绩 5.帮助 #\n"); printf("# #\n"); printf("# 6.错词预览 7.英雄榜 #\n"); printf("# #\n"); printf("# 8.退出系统 #\n"); printf("# #\n"); printf(" ********************************************************\n"); printf("\n"); printf("请选择您需要的操作序号(0-7)按回车确认:"); scanf("%d",&item); printf("\n"); ReadVocabularyFile(); switch(item) { case 0: Maintain(); break; case 1: ShowAllVocabulary(); break; case 2: ReciteAllVocabulary(); break; case 3: ReciteWrongVocabulary(); break; case 4: List(); break; case 5: Help(); break; case 6: ShowWrongVocabulary(); case 7: Rank(); break; case 8: select =0; default: printf("请在0-6之间选择\n"); } } int main() { while(select) { Menu(); } system("pause"); return 0; }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持亿速云。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。