L1-064 估值一亿的AI核心代码 (20 分) (字符串处理max)

以上图片来自新浪微博。

本题要求你实现一个稍微更值钱一点的 AI 英文问答程序,规则是:

  • 无论用户说什么,首先把对方说的话在一行中原样打印出来;
  • 消除原文中多余空格:把相邻单词间的多个空格换成 1 个空格,把行首尾的空格全部删掉,把标点符号前面的空格删掉;
  • 把原文中所有大写英文字母变成小写,除了 I
  • 把原文中所有独立的 can youcould you 对应地换成 I canI could—— 这里“独立”是指被空格或标点符号分隔开的单词;
  • 把原文中所有独立的 I 和 me 换成 you
  • 把原文中所有的问号 ? 换成惊叹号 !
  • 在一行中输出替换后的句子作为 AI 的回答。

输入格式:

输入首先在第一行给出不超过 10 的正整数 N,随后 N 行,每行给出一句不超过 1000 个字符的、以回车结尾的用户的对话,对话为非空字符串,仅包括字母、数字、空格、可见的半角标点符号。

输出格式:

按题面要求输出,每个 AI 的回答前要加上 AI: 和一个空格。

输入样例:

6
Hello ?
 Good to chat   with you
can   you speak Chinese?
Really?
Could you show me 5
What Is this prime? I,don 't know

输出样例:

Hello ?
AI: hello!
 Good to chat   with you
AI: good to chat with you
can   you speak Chinese?
AI: I can speak chinese!
Really?
AI: really!
Could you show me 5
AI: I could show you 5
What Is this prime? I,don 't know
AI: what Is this prime! you,don't know

有两次 if(s[len-1]==' ')s.replace(len-1,1,""); 的原因是使用replace函数会产生莫名的字符串内容的错误 如图

 第一次去除前后总共两个空格后末尾依旧会有一个空格,所以再次去除末尾空格!

此代码在编译器运行会报警告 warning: comparison between signed and unsigned integer expressions [-Wsign-compare]| ,不需要将该行的int改为unsigned int ,否则pta平台会报段错误

代码如下:


#include<bits/stdc++.h>
using namespace std;
string s;
string s1[]={"can you","could you","I","me"};
string s2[]={"@","#","you","you"};     /*使用符号作为中间替代,避免将"can you"替换为"I can"之后"I"被替换为"you"*/
string s3[]={"@","#","?"};
string s4[]={"I can","I could","!"};
void trim(){
    int pos=s.find("  ");        
    while(pos!=-1){                //将字符串所有的大于等于两个空格的替换为一个空格
        s.replace(pos,2," ");
        pos=s.find("  ");
    }
    //find()函数在字符串中寻找到括号中子字符串的下标并返回,失败返回-1
    int len=s.length();
    if(s[0]==' ')                  //去除字符串首部空格
        s.replace(0,1,"");
    if(s[len-1]==' ')              //去除字符串末尾空格
        s.replace(len-1,1,"");
    len=s.length();
    for(int i=len-1;i>=0;i--){     //将标点符号前面的空格去除
        if(s[i-1]==' ' && ispunct(s[i]))
            s.replace(i-1,1,"");
    }
    //ispunct()函数判断括号内字符是否为标点符号,标点符号返回true,否则返回false
    len=s.length();
    if(s[len-1]==' ')              //去除字符串末尾空格
        s.replace(len-1,1,"");
}


void change(){
    for(int i=0;i<4;i++)     //将"can you","could you","I","me"替换为"@","#","you","you"
    {
        int leni=s1[i].length();
        for(int j=0; j<s.length(); j++){
            if(s.substr(j,leni)==s1[i] && !isalnum(s[j-1]) && !isalnum(s[j+leni]))
                s.replace(j,leni,s2[i]);
        }
    //substr()函数截取字符串中的某一段
    }
    //isalnum()函数,判断一个字符是否是字母或者(十进制)数字,若为字母或者数字,则返回True(非0值),否者返回False(0)
    for(int i=0;i<3;i++){    //将"@","#","?"替换为"I can","I could","!"
        int pos=s.find(s3[i]);
        while(pos!=-1){
            s.replace(pos,1,s4[i]);
            pos=s.find(s3[i]);
        }
    }
}

int main()
{
    int n ;
    cin>>n;
    getchar();
    while(n--){
        getline(cin,s);
        cout<<s<<endl;
        trim();
        int len=s.length();
        for(int i=0;i<len;i++){    //将除"I"之外所有字母变成小写字母
            if(s[i]!='I')
                s[i]=tolower(s[i]);
        }
        change();
        cout<<"AI: "<<s<<endl;
    }

    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值