问题描述
对于任意给定的单词小助手程序(现有功能可以实现查单词的增删改查、中英、英中测试和分数显示功能),完善其单词查询、错误单词重复记忆和排行功能。
问题分析
首先附上给出的初始代码(已经可以实现基本功能)
#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;}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持武林网。
新闻热点
疑难解答
图片精选