刷题-关键字标识

题目描述

给定一个关键词集合 words 和一个字符串 inputStr,要求将 inputStr 中出现的所有关键词用标签标记:

  • 关键词标签的起始为<b>,结束为</b>。
  • 把可以合并的标签进行合并,即使用最少的标签。合并规则如下:
    • 关键词相邻则进行合并,如 ab、cd为关键词,且在字符串中相邻,则合并为 <b>abcd</b>
    • 关键词相交则进行合并,如 zhi、hid为关键词,且在字符串中相交,则合并为 <b>zhid</b>

解答要求时间限制:1000ms, 内存限制:256MB

输入

第1行一个整数 count,表示 words 中的关键词的个数,取值范围:[1, 64]
第2行 count 个字符串,表示关键字词列表 words,每个关键词仅含英文小写字母,长度范围:[1,16]
第3行一个字符串 inputStr,仅含英文小写字母(无空格),长度范围:[1,512]

输出

一个用最少关键词标签标记的字符串

样例

输入样例 1 

4
cd df op qr
opqracdfg

输出样例 1

<b>opqr</b>a<b>cdf</b>g

提示样例 1

  • 关键词 op qr 出现在字符串中,且在字符串中相邻,因此可合并为 opqr;
  • cd df 出现在字符串中,且在字符串中相交,因此可合并为 cdf ;
  • 其它字符保持不变。

处理后为 <b>opqr</b>a<b>cdf</b>g,使用了 2 个标签 。
<b>op</b><b>qr</b>a<b>cd</b><b>df</b>g也是一种标记方式,但关键词未作合并,需要用到 4 个标签,标签个数不是最少的。


 

输入样例 2 

3
abbb def bbg
aabbbgz

输出样例 2

a<b>abbbg</b>z

提示样例 2

关键字abbb和bbg合并为abbbg

解题代码

#include <iostream>
#include <sstream>
#include <string>
#include <vector>
#include <utility>
#include <algorithm>
using namespace std;

class Solution {
public:
    // 待实现函数,在此函数中填入答题代码;
    static bool cmp(const pair<int, int> &a, const pair<int, int> &b)
    {
        if (a.first == b.first)
            return a.second < b.second;
        return a.first < b.first;
    }
    
    string GetTaggedString(const vector<string> &words, const string &inputStr)
    {
        string str = inputStr;
        vector<pair<int, int>> keyPos;

        //找到所有关键字在str的位置, 记录在keyPos中
        for (string s : words) {
            int pos = 0;
            int index = 0;
            while ((index = str.find(s, pos)) != string::npos) {
                keyPos.push_back({index, index+s.size()});
                pos = index + s.size(); 
            }
        }
        
        if (keyPos.size() == 0)
            return str;

        //按位置升序排列keyPos,用于后续合并
        sort(keyPos.begin(), keyPos.end(), cmp);

        int left = keyPos[0].first;
        int right = keyPos[0].second;
        vector<pair<int, int>> myPos;
        
        //keyPos区间合并, 跳过重叠字符位置
        for (int i = 1; i < keyPos.size(); i++) {
            pair<int, int> p = keyPos[i];
            if (p.first == right) {
                right = p.second;
            } else if (p.first < right) {
                right = max(right, p.second);
            } else {
                myPos.push_back({left, right});
                left = p.first;
                right = p.second;
            }
        }
        myPos.push_back({left, right});


        //根据myPos中记录的关键词位置,插入新字符
        string start = "<b>";
        string end = "</b>";

        for (int i = 0; i < myPos.size(); i++) {
            left = myPos[i].first + i*(start.size() + end.size());
            right = myPos[i].second + i*(start.size() + end.size()) + start.size();
            str.insert(left, start);
            str.insert(right, end);
        }

        return str;
    }
};

inline int ReadInt()
{
    int number;
    std::cin >> number;
    return number;
}

void SplitString(const string &input, char sperChar, vector<string> &outArray)
{
    stringstream sstr(input);
    string token;
    while (getline(sstr, token, sperChar)) {
        outArray.push_back(token);
    }
}

inline string ReadLine()
{
    string line;
    getline(cin, line);
    return line;
}

int main()
{   
    int wordsNum = ReadInt();
    cin.ignore();
    vector<string> words;
    string wordsList = ReadLine();
    SplitString(wordsList, ' ', words);
    words.resize(wordsNum);
    string inputStr = ReadLine();
    
    Solution solu;
    string result = solu.GetTaggedString(words, inputStr);
    cout << result;
    return 0;
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值