DS第四章学习小结

本章最令人印象深刻的题就是AI核心代码这题了。如下

说实话,刚看到这题真的懵了,虽然只是一道题,但总给人6题的感觉。尽管困难重重,但还是在陈晓梅老师的指导下,大致完成了此题。

逻辑分析

先定义数据结构,自然是字符数组/字符串
最初考虑主函数逻辑,主要是读取输入语句,并调用接口处理输入语句并输出。基本流程是,读入一句,对其进行扫描、判断、操作,再存到新的字符串,最后输出新的字符串。

一开始输入部分就卡住了,因为不知道怎么处理回车后还能输入数据,进度就陷入了停滞,不过课上根据老师所说getchar()可以把回车吸收掉之后,又有所进展

int main() {
    int n;        //行数n,s存放输入的语句
    string s;
    cin >> n;
    getchar();
    for (int i = 1; i <= n; i++) {
        getline(cin, s);
        cout << s << endl;        //先输出原话,再输出处理后的AI回答
        cout << "AI: ";
        answer(s);    //处理并输出回答
    }
    return 0;
}

然后对接口answer()进行定义。
首先考虑把多余的空格删到只剩一个

void answer(string s) {        //根据s处理并输出回答
    int i, j;        //i定位到s的第一个非空
    for (i = 0; s[i] == ‘ ‘; i++) {

    }
 }

然后对边界问题进行考虑。当字符串全为空格的时候呢?会死循环或越界溢出吗?
其实不会,因为字符串有个结尾符‘\0’,所以及时字符串全空,到最后的结尾符也会停止循环。
定位到第一个非空字符后就可以开始输入。

void answer(string s) {        //根据s处理并输出回答
    string t;    //t为处理后的字符
    int i, j = 0;        //i定位到s的第一个非空,j表示t串的字符
    for (i = 0; s[i] == ‘ ‘; i++) {
        //仅仅用于定位.因为字符串有个结尾符‘\0’,所以及时字符串全空,到最后的结尾符也会停止循环
    }
    while (s[i] != ‘\0‘) {    //进行输入
        if (s[i] == ‘ ‘&&s[i - 1] == ‘ ‘) {        //跳过多余的空格
            i++;
            continue;
        }
        t[j] = s[i];    //将s串的非空或者单个空格给到t串,之后分别+1进行下一轮输入
        j++;
        i++;
    }
 }

但是这个时候有一个个问题,就是t串结尾没有结尾符,这个好办,最后给他加一个。此外,我们在一个个判断的时候,可以顺便实现问号变感叹号,大写变小写的功能。

id answer(string s) {
    string t;    //t为处理后的字符串
    int i, j = 0;        //i定位到s的第一个非空,j表示t串的字符
    for (i = 0; s[i] == ‘ ‘; i++) {
        //仅仅用于定位.因为字符串有个结尾符‘\0’,所以及时字符串全空,到最后的结尾符也会停止循环
    }
    while (s[i] != ‘\0‘) {    //进行输入
        if (s[i] == ‘ ‘&&s[i - 1] == ‘ ‘) {        //跳过多余的空格
            i++;
            continue;
        }
        if (s[i] == ‘?‘) {        //将输入的问号变为感叹号
            t[j] = ‘!‘;
            i++;
            j++;
            continue;
        }

        if (s[i] != ‘I‘) {        //将除了I的大写变小写
            t[j] = tolower(s[i]);
            i++;
            j++;
        }
        else {
            t[j] = s[i];    //将s串的非空或者单个空格给到t串,之后分别+1进行下一轮输入
            j++;
            i++;
        }
    }
    t[j] = ‘\0‘;    //为t串末尾增加结尾符

这里用到了tolower()函数将大写变成小写。
将s串的有效都给了t串之后,我们可以遍历t串来进行进一步操作,比如将
独立的I和me变成you

j = 0;
    while (t[j] != ‘\0‘) {
        //独立的I,意味着左右均是分隔符
        if (t[j] == ‘I‘&&(j==0||isIndependent(t[j - 1])) && isIndependent(t[j + 1])) {
            cout << "you";
            j++;
            continue;
        }
        //独立的me
        if (t[j] == ‘m‘&&t[j + 1] == ‘e‘&& (j == 0 || isIndependent(t[j - 1])) && isIndependent(t[j + 2])) {
            cout << "you";
            j += 2;
            continue;
        }

因为需要判断是否独立,所以构造了isIndependent()函数。但是遇到了j+1,j-1等,需要考虑边界情况。
当独立的I,me是在字符串开头,j=0,所以j-1会导致非法访问,我们额外加一个条件(j==0||isIndependent(t[j - 1]),这样j=0的时候前面的条件成立,程序不再判断后面的条件,于是避免了非法访问。
而因为while循环中有s[i] != ‘\0’的条件,所以及时在串的末尾,之后还有一个\0,也就是说最坏情况下t[j+1]等于\0,不会导致越界。
这里给出是否为分隔符的判断

bool isIndependent(char ch) {
    bool result = true;
    ch = tolower(ch);
    if ((ch >= ‘0‘ && ch <= ‘9‘) || (ch >= ‘a‘&&ch <= ‘z‘)) {
        result = false;
    }
    return result;
}

到此,大致的程序已经出来了,我们还差末尾符号前的空格和把 can you 换成 I can。

首先是末尾符号前的空格
我们经过对s串的遍历,已经确保连续的空格变成了单个空格,也就是说,如果现在t串末尾是符号,那么我们要去掉符号前的空格。
可以用上之前的isIndependent(),如果这个空格需要去掉,那么他之后一定不是数字或字母(要么符号,要么\0,这两种情况都要去空格)

//如果是标点前的空格就不输出
        if (t[j] == ‘ ‘&&isIndependent(t[j+1])) {
            j++;
            continue;
        }
        cout << t[j];
        j++;
    }

然后是把 can you 换成 I can,具体方法和me变成you类似

bool isCanyou(char ch[],int n) {
    bool result = false;
    if (ch[n] == ‘c‘&&ch[n + 1] == ‘a‘&&ch[n + 2] == ‘n‘&&ch[n + 3]==‘ ‘ && ch[n + 4] == ‘y‘&&ch[n + 5] == ‘o‘&&ch[n + 6] == ‘u‘) {
        if ((n == 0 || isIndependent(ch[n - 1])) && isIndependent(ch[n + 7])) {
            result = true;
        }
    }
    return result;
}

然后在上面的answer函数中添加判断,如果是,就输出改变“I can”。记得要让j加上can you的长度(7)来跳过其他字符。

······
//独立的can you
        if (isCanyou(t, j)) {
            cout << "I can";
            j += 7;
            continue;
        }

        //独立的could you
        if (isCouldyou(t, j)) {
            cout << "I could";
            j += 9;
            continue;
        }
······

最后整合一下

#include <iostream>
#include <string>
#include <cstring>
#include <cstdio>

using namespace std;

void answer(string s);
bool isIndependent(char ch);
bool isCanyou(char ch[], int n);

int main() {
    int n;        //行数n,s存放输入的语句
    string s;
    cin >> n;
    getchar();
    for (int i = 1; i <= n; i++) {
        getline(cin, s);
        cout << s << endl;        //先输出原话,再输出处理后的AI回答
        cout << "AI: ";
        answer(s);    //处理并输出回答
    }
    return 0;
}

//根据s处理并输出回答
void answer(string s) {
    char t[3002];    //t为处理后的字符串
    int i, j = 0;        //i定位到s的第一个非空,j表示t串的字符
    for (i = 0; s[i] == ‘ ‘; i++) {
        //仅仅用于定位.因为字符串有个结尾符‘\0’,所以及时字符串全空,到最后的结尾符也会停止循环
    }
    while (s[i] != ‘\0‘) {    //进行输入
        if (s[i] == ‘ ‘&&s[i - 1] == ‘ ‘) {        //跳过多余的空格
            i++;
            continue;
        }
        if (s[i] == ‘?‘) {        //将输入的问号变为感叹号
            t[j] = ‘!‘;
            i++;
            j++;
            continue;
        }

        if (s[i] != ‘I‘) {        //将除了I的大写变小写
            t[j] = tolower(s[i]);
            i++;
            j++;
        }
        else {
            t[j] = s[i];    //将s串的非空或者单个空格给到t串,之后分别+1进行下一轮输入
            j++;
            i++;
        }
    }
    t[j] = ‘\0‘;    //为t串末尾增加结尾符

    //I,me变成you; can you 换成 I can
    j = 0;
    while (t[j] != ‘\0‘) {
        //独立的I,意味着左右均是分隔符
        if (t[j] == ‘I‘ && (j == 0 || isIndependent(t[j - 1])) && isIndependent(t[j + 1])) {
            cout << "you";
            j++;
            continue;
        }
        //独立的me
        if (t[j] == ‘m‘&&t[j + 1] == ‘e‘ && (j == 0 || isIndependent(t[j - 1])) && isIndependent(t[j + 2])) {
            cout << "you";
            j += 2;
            continue;
        }

        //独立的can you
        if (isCanyou(t, j)) {
            cout << "I can";
            j += 7;
            continue;
        }

        //如果是标点前的空格就不输出
        if (t[j] == ‘ ‘&&isIndependent(t[j+1])) {
            j++;
            continue;
        }
        cout << t[j];
        j++;
    }
    cout << endl;
}

//判断字符是否为分隔符
bool isIndependent(char ch) {
    bool result = true;
    ch = tolower(ch);
    if ((ch >= ‘0‘ && ch <= ‘9‘) || (ch >= ‘a‘&&ch <= ‘z‘)) {
        result = false;
    }
    return result;
}

//判断是否为独立的can you
bool isCanyou(char ch[],int n) {
    bool result = false;
    if (ch[n] == ‘c‘&&ch[n + 1] == ‘a‘&&ch[n + 2] == ‘n‘&&ch[n + 3]==‘ ‘ && ch[n + 4] == ‘y‘&&ch[n + 5] == ‘o‘&&ch[n + 6] == ‘u‘) {
        if ((n == 0 || isIndependent(ch[n - 1])) && isIndependent(ch[n + 7])) {
            result = true;
        }
    }
    return result;
}

终于啃下这块硬骨头,虽然花费很久时间,但也学到了很多东西,比如getchar()吸收回车,tolower把大写字母变成小写,还有最重要的一点就是把一个复杂的问题细化成多个问题,再逐一解决,最后合并到一起......

上次的目标基本完成,但是我的学习主动性还不是太高,希望这周能提高学习的热情,把kmp算法和十字链表解决吧

原文地址:https://www.cnblogs.com/hxyawsl/p/10708160.html

时间: 2024-11-08 08:06:09

DS第四章学习小结的相关文章

数据结构第四章学习小结

第四章主要是串和数组的学习,之前对串和数组的应用仅限于对其单独处理,本章学习着重于对具体题目的实际操作,实践了串的模式匹配算法,对其有了更深入的了解,数组的应用拓展到了稀疏矩阵的存储的实现. 一.串 串的模式匹配 BF算法 首先未匹配到串尾时,将两个字符串一一匹配,可用C++自带的length()函数实现 while(i<=S.length()&&j<=T.length()) 接下来就是匹配的过程 if(S[i]==T[j]){ i++;j++; }//若匹配则继续比较下一位

【数据结构】第四章学习小结

串.数组 在第四章中,我学到的主要是关于串与数组的内容,至于广义表,既然老师让我们课后有时间去看,那我这里就先不讲广义表了(其实只是粗略的看了一下,还没看懂) 在上学期的c++中我就已经学过有关串的一些知识,对于串还是可以很好的理解的,在数据结构中,串将被看作是一种特殊的线性表,跟线性表一样,串也有两种基本存储结构,个人觉得顺序存储结构对于串的运用比较方便,简单易懂,所以在作业题中会首先考虑使用顺序串. 串有许多很重要的应用,例如搜索引擎,数据压缩等,这些应用都离不开串的模式匹配算法(子串的定位

DS第4章学习小结

你对本章内容的小结 完成作业或实践时解决困难的经验分享 这段时间,你参考了哪些值得向大家分享的资料?每一项推荐都请说明推荐理由及列出相关链接(或书目名称,具体页码) 目前学习过程中存在的困难,待解决或待改进的问题 上次博客确定的目标达到了吗?如果没达到,请分析原因 接下来的目标 一.你对本章内容的小结 第4章主要学习了串.数组 串主要学习了有关 模式匹配算法 的两种算法:T(n)=O(m*n)的BF算法 和 T(n)=O(m+n)的KMP算法, KMP算法难在求出 模式的next数组.(即求ne

DS第七章学习小结

第七章小结 先列出一些基本的概念: ①关键字:数据元素(记录)中某个数据项的值,用它可以表示一个数据元素. ②动态查找表/静态查找表:若在查找的过程中进行修改操作(插入或删除),则相应的表为动态查找表,否则为静态查找表. ③平均查找长度:为确定记录在查找表中的位置,需和给定值进行比较的关键字个数的期望值称为查找算法在查找成功时的平均查找长度.公式如下:ASL=∑PiCi (i=1,2,3,…,n),可以简单以数学上的期望来这么理解.其中:Pi 为查找表中第i个数据元素的概率,Ci为找到第i个数据

第四章学习小结 串的模式匹配 解题心得体会

串的模式匹配 解题心得体会 关于串,模式匹配是其一个很重要的问题.针对这个问题,书上讲了两种模式匹配的算法,即BF算法和KMP算法,下面针对这两种算法的实现谈谈我的心得. 一.BF算法的探索 [错误代码1] #include<iostream> #include<string.h> using namespace std; typedef struct{ char ch[1000002]; int length; }SString; void Index_BF(SString S,

第四章学习小结

这两周上课学的较多且比较有印象的是串的模式匹配算法那道题 同时可以用BF或者KMP算法进行解决. 一开始学BF算法时就在想如何改进可以使匹配更加简便 后来KMP的学习让我对串的学习有了更深入的了解 同时BF和KMP算法之间的联系也是一个算法改进的很好学习范例 7-1 串的模式匹配 给定一个主串S(长度<=10^6)和一个模式T(长度<=10^5),要求在主串S中找出与模式T相匹配的子串,返回相匹配的子串中的第一个字符在主串S中出现的位置. 输入格式: 输入有两行: 第一行是主串S: 第二行是模

第四章 学习小结

我想谈谈我写模式匹配题时的心得体会: 如果是用BF算法我觉得的这道题真的非常简单,我是用的BF算法, 一开始我把string和书上的sstring搞混淆了 就像下图的代码所示: S.[i]==T.[j] S.length 但是string是头文件<cstring>包含的所以应该打成这样 S[i]==T[j] S.length() 老师上课时讲过这个,但我当时没有认真记下来,课后翻书,就直接按书上的打了,所以就出现错误.后来查阅了相关资料才修改过来了 因为用的是string所以我的BF算法函数如

第三章学习小结—-转

[学习目标] 01掌握一维数组的声明和使用方法(OK) 02掌握二维数组的声明和使用方法(OK) 03掌握字符串的声明.赋值.比较和连接方法(连接很少用) 04熟悉字符的ASCII码和ctype.h中的字符函数 05正确认识++.+=等能修改变量的运算符(OK) 06学会用编译选项-Wall获得更多的警告信息(OK) 07了解不同操作系统中换行符的表示方法(嗯) 08掌握fgetc和getchar的使用方法(fgetc基本没用过) 09掌握预处理和迭代开发的技巧(嗯) 程序3-1 逆序输出 输入

数据结构 第一章学习小结

数据结构   第一章学习小结 1.数据结构第1章的心得体会: 这周学习了数据结构的绪论及第一章.初步了解了数据结构与算法的相关概念,一开始看书看视频时觉得还挺抽象的,不能够完全理解.但是反复多看了几遍之后,结合例题,自己去操作去跑代码,慢慢觉得容易理解接受起来了.由于现在以网课形式进行教学,老师上课的同时基本还是靠自己去理解学习.当然老师也发挥很大的作用,比如让我们更深入的了解递归的空间复杂度为什么与问题规模有关,又怎样去找到该函数的临界值等等.既锻炼了我们深入思考的能力,也让我们更加清楚了解不