C語言實現(xiàn)單詞小助手改進版
本文為大家分享了C語言單詞小助手的具體實現(xiàn)代碼,供大家參考,具體內(nèi)容如下
一.題目分析
單詞小助手,要求完成以下任務:
1.改寫程序為良好程序風格(文檔注釋,函數(shù)注釋,語句注釋)。
2.將單詞測試中的功能完善,可針對做錯的單詞重復記憶。
3.查詢單詞的功能添加英文詞查詢,中文查詢的功能完善(考慮如何顯示同樣中文意義,不同的英文單詞)
提高要求:
可將程序中的不足(如數(shù)據(jù)驗證,排名功能)等根據(jù)自己的能力與理解完成。
二.算法構造(新增函數(shù))
流程圖
(1)總流程圖

(2)search()函數(shù)流程圖

(3)repeat()函數(shù)流程圖

三.程序實現(xiàn)
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#define MAX_CHAR 20 // 最大字符
#define MAX_NUM 200 // 單詞的最大個數(shù)
struct word
//單詞的結構體
{
char en[MAX_CHAR]; // 英文形式
char ch[MAX_CHAR]; //中文形式
}
s[MAX_NUM]; //單詞數(shù)組
int num; //單詞個數(shù)
int select=1;//select 為是否退出系統(tǒng)的標記
int d=0,c=0;//c為答錯的次數(shù),d為答對的次數(shù)
//幫助
void help()
{
printf("\n本系統(tǒng)主要實現(xiàn)英語單詞學習的功能。用戶可對詞典文件中的單詞進行預覽,增刪改查。");
printf("\n同時還可進行中英、英中測試。本系統(tǒng)還提供了測試成績的顯示功能。");
}
//從文件中讀取單詞的信息
void readfile()
{
FILE *fp;
int i=0;
fp=fopen("E:\\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(i==0)
printf("\n文件為空,請選擇詞典維護增加詞條!");
else
printf("\n");
fclose(fp);//關閉文件
}
//向文件中寫入單詞
void writefile()
{
FILE *fp;
int i=0;
fp=fopen("E:\\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);
}
}
}
//定義記憶文件函數(shù),用來記憶背誦錯誤的單詞,將背誦過程中錯誤的單詞寫入Repeat文件
void readRepeatfile()
{
FILE *fp;
int i=0;
fp=fopen("E:\\Repeat.txt","r");//增強記憶文件
if(!fp)
{
printf("\n打開文件失敗");
}
while(fscanf(fp,"%s %s",s[i].en,s[i].ch)==2)
{
i++;
}
num=i;
if(i==0)//沒有錯題
printf("\n沒有錯題");
else
printf("\n");
fclose(fp);
}
//添加單詞信息
void add()
{
int i=num,j,flag=1;
int a=0,k=0;
printf("你想添加幾個單詞:");
scanf("%d",&a);
while(flag)
{
for(k=0;k<a;k++)
{
flag=0;
printf("\n請輸入第%d個單詞的英文形式:",k+1);
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; /*如有重復立即退出該層循環(huán),提高判斷速度*/
}
printf("請輸入第%d個單詞的中文形式:",k+1);
scanf("%s",s[i].ch);
num++;
printf("您輸入的信息為: 英文: %s 中文: %s ",s[i].en,s[i].ch);
printf("\n");
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--; //數(shù)量減少 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繼續(xù)修改請選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;//flag標記
int index = -1; //檢索到該單詞的中文或者英文的下標
int option = 0;
char ch[MAX_CHAR]; //中文形式
char en[MAX_CHAR]; //英文形式
while(1)
{
printf("請輸入1.以中文查詢單詞2.以英文查詢單詞0.退出");
scanf("%d", &option);//可選中英文查詢
switch(option)
{
case 1:
while(1)
{
printf("\n請輸入你要查詢的單詞中文形式:");
scanf("%s", ch);
for(i=0; i<num; i++)//遍歷找到所有該中文對應的單詞
if(strcmp(s[i].ch, ch) == 0 && index == -1)//輸出該單詞的中英文形式
{
printf("\n 中文形式 英文形式 ");
printf("\n %-12s%12s", s[i].en, s[i].ch);
index = i;
}
else if(strcmp(s[i].ch, ch) == 0)
{
printf("\n %-12s%12s", s[i].en, s[i].ch);
index = i;
}
if(index == -1) //如果沒找到該單詞,下標仍為初始值-1
{
flag = 0;//讓flag = 0; 表示沒有該單詞
}
if(!flag)
printf("\n沒有這個單詞!");
index = -1; //將index 和flag的值重新初始化為原來的值
flag = 1;
printf("\n繼續(xù)查詢請選1,返回上一級請選0:"); //選擇是否繼續(xù)查詢
scanf("%d", &choose);
if(choose == 0)
break;
}
break;
case 2://查詢該英文形式對應的單詞
while(1)
{
printf("\n請輸入你要查詢的單詞英文形式:");
scanf("%s", en);
for(i=0; i<num; i++)//先找到該英文形式對應的序號
if(strcmp(s[i].en, en) == 0)
{
printf("\n 中文形式 英文形式 ");
printf("\n %-12s%12s", s[i].en, s[i].ch);
index = i;
}
if(index == -1)
{
flag = 0;
}
if(!flag)
printf("\n沒有這個單詞!");
index = -1;
flag = 1;
printf("\n繼續(xù)查詢請選1,返回上一級請選0:");
scanf("%d", &choose);
if(choose == 0)
break;
}
default:
return;
}
}
}
//中譯英測試
void zytest()
{
char b1[20];
int z;
int choose=1;
int i;
//打開文件
FILE *fp;
while(choose)
{
fp=fopen("E:\\Repeat.txt","a+");
if(!fp)
{
printf("\n打開文件失敗");
}
i = rand()%num;
printf("\n【%s】請輸入英文單詞:",s[i].ch);
scanf("%s",b1);
for(z=0;strcmp(b1,s[i].en)!=0;z=z)
{
//回答錯誤時寫入增強記憶文件
fprintf(fp,"\n%s %s",s[i].en,s[i].ch);
printf("\n輸入錯誤??!請重新輸入:");
scanf("%s",b1);
c=c+1;
num++;//單詞數(shù)目增加
}
fclose(fp);
printf("\n恭喜你,回答正確,加10分!\n\n");d=d+1;
printf("\n繼續(xù)測試請選1,返回上一級請選0:");
scanf("%d",&choose);
if(choose==0)
return;
}
}
//英譯中測試
void yztest()
{
char b1[20];
int z,x=41;
int choose=1;
int i;
//打開文件
FILE *fp;
while(choose)
{
i = rand()%num;
fp=fopen("E:\\Repeat.txt","a+");//打開記憶文件
if(!fp)
{
printf("\n打開文件失敗");
}
printf("【%s】請輸入中文意思:",s[i].en);
scanf("%s",b1);
for(z=0;strcmp(b1,s[i].ch)!=0;z=z)
{
printf("輸入錯誤!!請重新輸入:");
fprintf(fp,"\n%s %s",s[i].ch,s[i].en);
scanf("%s",b1);
c=c+1;
num++;//文件中單詞數(shù)目增加
}
fclose(fp);//關閉文件
printf("\n恭喜你,回答正確,加10分!\n\n");d=d+1;
printf("\n繼續(xù)測試請選1,返回上一級請選0:");
scanf("%d",&choose);
if(choose==0)
return;
}
}
//增強錯誤單詞的背誦功能
void Repeat()
{
char b1[20];
int z;
int choose=1;
int i;
int n=0;
int a;
while(1)
{
printf("\n請輸入1.英中錯題復習 2.中英錯題復習 0.退出");//選擇
scanf("%d",&a);
switch(a)
{
case 1:
while(1)
{
i=rand()%num;//從repeat.txt文件中隨機拿出單詞
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;//錯題中我采用的是答錯扣分,答對不加分的原則
num++;//文件中單詞數(shù)目增加
}
printf("\n恭喜你,回答正確!\n\n");
printf("\n繼續(xù)測試請選1,返回上一級請選0:");
scanf("%d",&choose);
if(choose==0)
return;
}break;
case 2:
while(1)
{
i=rand()%num;
printf("[%s]請輸入英文單詞:",s[i].ch);//輸出漢語意思,考查能否寫出英文單詞
scanf("%s",b1);
for(z=0;strcmp(b1,s[i].en)!=0;z=z)
{
printf("輸入錯誤??!請重新輸入:");
scanf("%s",b1);
c=c+1;
num++;//文件中單詞數(shù)目增加
}
printf("\n恭喜你,回答正確!\n\n");
printf("\n繼續(xù)測試請選1,返回上一級請選0:");
scanf("%d",&choose);
if(choose==0)
return;
}break;
default:
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 x=0;
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(" - 版本 : v2.0 --\n");
printf(" - --\n");
printf(" ---------------------------------------------------------\n");
printf(" - --\n");
printf(" - 1.詞庫維護 2.單詞預覽 --\n");
printf(" - --\n");
printf(" - 3.單詞背誦(中英) 4.單詞背誦(英中) --\n");
printf(" - --\n");
printf(" - 5.查詢成績 6.幫助 --\n");
printf(" - --\n");
printf(" - 7.錯題復習 8.退出系統(tǒng) --\n");
printf(" ---------------------------------------------------------\n");
printf("\n");
printf(" - 請選擇您需要的操作序號(1-8)按回車確認:");
scanf("%d",&item);
printf("\n");
readfile();
switch(item)
{
case 1:
maintain();
break;
case 2:
show();
break;
case 3:
zytest();
break;
case 4:
yztest();
break;
case 5:
list();
break;
case 6:
help();
break;
case 7:
readRepeatfile();
Repeat();
break;
case 8:
select =0;
break;
default:
printf("請在1-8之間選擇\n");
}
}
int main()
{
while(select)
{
menu();
}
system("pause");
return 0;
}
四.經(jīng)驗歸納
這次的上機實驗很有趣,但是也很高級,老師一開始就給了一個基本框架了,我只需要往里面增添,修改一些些就好了。
(1)首先,我把主界面的選擇序號改了一點點,因為以我個人而言,我用不慣從0開始。也在里面根據(jù)我自己的理解,加了些注釋。
(2)在search()函數(shù)中,因為一個相同的中文意思可能會有多個英語單詞與之對應,原來的代碼進行遍歷之后就break了循環(huán),自然只能查詢到一個英文單詞,我做的是遍歷整個數(shù)組,同時用一個下標index = -1 進行標記,同時打印該單詞的中英文,下標index的值改為當前數(shù)組的下標i ,繼續(xù)遍歷打印,同時標記,遍歷完整個數(shù)組就可以打印多個擁有相同中文解釋的不同的英文單詞. 而index的作用就是,如果遍歷完整個數(shù)組index仍等于-1,那么就說明沒有查到該單詞,可以進行查詢?yōu)榭盏呐袛?
(3)在add()函數(shù)中,我添加了一個for循環(huán),是錄入單詞變得更簡潔了。
(4)我也加入了錯題功能,在測試中,回答錯誤的單詞會被加入在repeat.txt文件中,然后用隨機數(shù),抽取出來,再次回答,在這里,我改了一下下計分,在錯題庫的中出來重新回答的題目,答對不加分,答錯還是會扣分,所以,還是最好有了沒記住的單詞 就趕快記下它哦。
(5)調試中發(fā)現(xiàn)我在search()中把s[i].en和s[i].ch給弄反了。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
用pybind11封裝C++實現(xiàn)的函數(shù)庫的方法示例
這篇文章主要介紹了用pybind11封裝C++實現(xiàn)的函數(shù)庫,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-02-02
C++中的std::funture和std::promise實例詳解
在線程池中獲取線程執(zhí)行函數(shù)的返回值時,通常使用 std::future 而不是 std::promise 來傳遞返回值,這篇文章主要介紹了C++中的std::funture和std::promise實例詳解,需要的朋友可以參考下2024-05-05

