问题 I: 单词检查(Ⅰ)- 顺序表实现

问题 I: 单词检查(Ⅰ)- 顺序表实现

题目描述
许多应用程序,如字处理软件,邮件客户端等,都包含了单词检查特性。单词检查是根据字典,找出输入文本中拼错的单词,我们认为凡是不出现在字典中的单词都是错误单词。不仅如此,一些检查程序还能给出类似拼错单词的修改建议单词。 例如字典由下面几个单词组成:
bake cake main rain vase
如果输入文件中有词vake ,检查程序就能发现其是一个错误的单词,并且给出 bake, cake或vase做为修改建议单词。
修改建议单词可以采用如下生成技术:
(1)在每一个可能位置插入‘a-‘z’中的一者
(2)删除单词中的一个字符
(3)用‘a’-'z’中的一者取代单词中的任一字符
很明显拼写检查程序的核心操作是在字典中查找某个单词,如果字典很大,性能无疑是非常关键的。
你写的程序要求读入字典文件,然后对一个输入文件的单词进行检查,列出其中的错误单词并给出修改建议。

课程设计必须采用如下技术完成并进行复杂度分析及性能比较。
(1)朴素的算法,用线性表维护字典
(2)使用二叉排序树维护字典
(3)采用hash技术维护字典

本题要求使用顺序表实现。

输入
输入分为两部分。
第一部分是字典,每个单词占据一行,最后以仅包含’#'的一行表示结束。所有的单词都是不同的,字典中最多10000个单词。

输入的第二部分包含了所有待检测的单词,单词数目不超过50。每个单词占据一行,最后以仅包含’#'的一行表示结束。

字典中的单词和待检测的单词均由小写字母组成,并且单词最大长度为15。
输出
按照检查次序每个单词输出一行,该行首先输出单词自身。如果单词在字典中出现,接着输出" is correct"。如果单词是错误的,那么接着输出’:‘,如果字典中有建议修改单词,则按照字典中出现的先后次序输出所有的建议修改单词(每个前面都添加一个空格),如果无建议修改单词,在’:'后直接换行。

在这里插入图片描述

样例输出

aware: award
m: i my me
contest is correct
hav: has have
oo: too
or:
i is correct
fi: i
mre: more me

实现过程

顺序表
1)什么是顺序表
顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改。

顺序表:可动态增长的数组,要求数据是连续存储的

2)顺序表的定义
1、静态顺序表:使用定长数组存储元素

缺陷:给小了不够用,给大了可能浪费,非常不实用

#define N 10
typedef int SLDataType;

typedef struct SeqList
{
	SLDataType array[N];  //定长数组
	size_t size;          //有效数据个数
}SeqList;

2、动态顺序表:使用动态开辟的数组存储元素

动态顺序表可根据我们的需要分配空间大小
size 表示当前顺序表中已存放的数据个数
capacity 表示顺序表总共能够存放的数据个数
typedef int SLDataType; //类型重命名,后续要存储其它类型时方便更改

typedef int SLDataType; //类型重命名,后续要存储其它类型时方便更改

typedef struct SeqList
{
	SLDataType* a;    //指向动态开辟的数组
	size_t size;      //有效数据个数
	size_t capacity;  //容量大小
}SeqList;

以上顺序表定义原文链接:https://blog.csdn.net/weixin_48025315/article/details/119778068

此题实现过程:
1.定义顺序表用于存储单词及长度
2.创建单词表并计算其长度
3.判断长度相同的两个单词是否相等
4.删除字母进行比对
5.添加字母进行比对
6.改变字母进行比对
7.得出结果right or wrong

下面是对代码的详细解析:

  1. 头文件和命名空间

    • 包含 <iostream><cstdio><cstring> 头文件。
    • 使用 using namespace std; 来避免在标准库类型和函数前加 std::
  2. 单词结构体定义

    • 定义了一个结构体 word,包含一个字符数组 data 用于存储单词,和一个整型 length 用于存储单词的长度。
  3. 全局变量

    • Dic 是一个大小为 10001 的 word 类型数组,用作单词字典。
    • num 是一个整型变量,用作单词在字典中的下标。
  4. 创建单词表函数 creat

    • 使用 cin 读取单词直到遇到 # 符号,将每个单词存储到 Dic 数组中,并更新 num
  5. 判断单词是否相等函数 judge

    • 接收两个 word 类型的参数,判断它们是否完全相同。
  6. 删除字母进行对比函数 found1

    • 尝试删除一个字母来比较输入的单词 x 和字典中的单词 y 是否相等。
  7. 添加字母进行对比函数 found2

    • 尝试添加一个字母来比较输入的单词 x 和字典中的单词 y 是否相等。
  8. 改变字母进行对比函数 found3

    • 尝试改变一个字母来比较输入的单词 x 和字典中的单词 y 是否相等。
  9. 检查单词函数 found

    • 接收一个 word 类型的参数 phrase
    • 遍历字典,使用 judge 函数检查 phrase 是否与字典中的单词完全相同。
    • 如果 phrase 的长度比字典中的单词长度多1、少1或相等,分别调用 found1found2found3 函数进行检查。
    • 输出检查结果。
  10. 主函数 main

    • 首先调用 creat 函数创建单词表。
    • 然后读取用户输入的单词,直到遇到 # 符号。
    • 对每个输入的单词调用 found 函数进行检查,并输出结果。
  11. 程序结束

    • 返回 0,表示程序正常结束。

代码逻辑分析

  • 这段代码实现了一个基于控制台的单词检查工具,它可以检查用户输入的单词是否正确,或者是否可以通过简单的字母变换(添加、删除、替换)来形成正确的单词。
  • 它首先读取一个单词列表到字典中,然后读取用户输入的单词,通过一系列的函数来判断单词的正确性。

改进建议

  • 对用户输入进行检查,确保不会超过 word 结构体中 data 数组的大小。
  • 为了提高代码的健壮性,可以考虑使用异常处理来捕获和处理潜在的错误。

部分实现

定义单词及长度 ,创建单词字典 ,初始化单词下标

typedef struct word{
    char data[15];
    int length=0;
}word;//定义单词及长度 
word Dic[10001];//创建单词字典 
int num=0;//初始化单词下标

2.创建单词表并计算其长度

void creat(){//创建单词表 
    while(cin>>Dic[num].data){
    if(Dic[num].data[0]=='#')break;
    Dic[num].length=strlen(Dic[num].data);//计算单词长度 
       num++;
    } 
}

3.判断长度相同的两个单词是否相等

int judge(word x,word y){
  int flag=1;
  for(int i=0;i<x.length;i++)if(x.data[i]!=y.data[i]){
      flag=0;break;
  }
  return flag;
}

4.删除字母进行比对

void found1(word x,word y){
  word phrase1=x;
   int flag=1;int count=0;
      for(int i=0,j=0;i<y.length;i++,j++){
          if (count>1)break;
          if(phrase1.data[j]!=y.data[i]){
              j--;count++;
          }
      }
  if(count==1){
  cout<<" "<<x.data;flag=0;
  }
}

5.添加字母进行比对

void found2(word x,word y){
      word phrase2=y;int flag=1;int count=0;
      for(int i=0,j=0;i<x.length;i++,j++){
          if (count>1)break;
          if(phrase2.data[j]!=x.data[i]){
              j--;count++;
          }
      }
  if(count==1){
  cout<<" "<<x.data;flag=0;
  }
}

6.改变字母进行比对

void found3(word x,word y){
      word phrase3=y;int flag=1;int count=0;
      for(int i=0,j=0;i<x.length;i++,j++){
          if (count>1)break;
          if(phrase3.data[j]!=x.data[i]){
          count++;
          }
      }
  if(count==1){
  cout<<" "<<x.data;flag=0;
  }
}

7.将上述所有函数组合到一起实现单词的检查并得出结果

void found(word phrase){
  int flag=1;
  for(int i=0;i<num&&flag==1;i++){
      if(flag){
          if(phrase.length==Dic[i].length)if(judge(phrase,Dic[i])) {
              cout<<phrase.data<<" is correct";flag=0;
          }
      }
  }
   if(flag)cout<<phrase.data<<":";
  if(flag)for(int i=0;i<num;i++){
  if(phrase.length-1==Dic[i].length)found1(Dic[i],phrase);
          if(phrase.length+1==Dic[i].length)found2(Dic[i],phrase);
              if(phrase.length==Dic[i].length)    found3(Dic[i],phrase);
  }   cout<<endl;
   
}

AC代码

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
 
 
typedef struct word{
    char data[15];
    int length=0;
}word;//定义单词及长度 
 
 
word Dic[10001];//创建单词字典 
 
 
int num=0;//初始化单词下标
 
 
 void creat(){//创建单词表 
     while(cin>>Dic[num].data){
     if(Dic[num].data[0]=='#')break;
     Dic[num].length=strlen(Dic[num].data);//计算单词长度 
        num++;
     } 
 }
  
  
 int judge(word x,word y){//判断长度相同的两个单词是否相等 
    int flag=1;
    for(int i=0;i<x.length;i++)if(x.data[i]!=y.data[i]){
        flag=0;break;
    }
    return flag;
 }
  
  
 void found1(word x,word y){//删除字母进行对比 
    word phrase1=x;
     int flag=1;int count=0;
        for(int i=0,j=0;i<y.length;i++,j++){
            if (count>1)break;
            if(phrase1.data[j]!=y.data[i]){
                j--;count++;
            }
        }
    if(count==1){
    cout<<" "<<x.data;flag=0;
    }
 }
  
  
 void found2(word x,word y){//添加字母进行对比 
        word phrase2=y;int flag=1;int count=0;
        for(int i=0,j=0;i<x.length;i++,j++){
            if (count>1)break;
            if(phrase2.data[j]!=x.data[i]){
                j--;count++;
            }
        }
    if(count==1){
    cout<<" "<<x.data;flag=0;
    }
 }
  
  
 void found3(word x,word y){//改变字母进行对比 
        word phrase3=y;int flag=1;int count=0;
        for(int i=0,j=0;i<x.length;i++,j++){
            if (count>1)break;
            if(phrase3.data[j]!=x.data[i]){
            count++;
            }
        }
    if(count==1){
    cout<<" "<<x.data;flag=0;
    }
 }
  
  
 void found(word phrase){
     //将上述所有函数组合到一起实现单词的检查
    int flag=1;
    for(int i=0;i<num&&flag==1;i++){
        if(flag){
            if(phrase.length==Dic[i].length)if(judge(phrase,Dic[i])) {
                cout<<phrase.data<<" is correct";flag=0;
            }
        }
    }
     if(flag)cout<<phrase.data<<":";
    if(flag)for(int i=0;i<num;i++){
    if(phrase.length-1==Dic[i].length)found1(Dic[i],phrase);
            if(phrase.length+1==Dic[i].length)found2(Dic[i],phrase);
                if(phrase.length==Dic[i].length)    found3(Dic[i],phrase);
    }   cout<<endl;
     
 }
  
  
int main(){
    creat();
    word phrase;
    while(cin>>phrase.data){
        if(phrase.data[0]=='#')break;
        phrase.length=strlen(phrase.data);
        found(phrase);
    } 
    return 0;
}
  • 51
    点赞
  • 38
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值