【字典树】【字符串】【 前缀】100268. 最长公共后缀查询

简介: 【字典树】【字符串】【 前缀】100268. 最长公共后缀查询

本文涉及知识点

字典树 字符串 前缀

LeetCode 100268. 最长公共后缀查询

给你两个字符串数组 wordsContainer 和 wordsQuery 。

对于每个 wordsQuery[i] ,你需要从 wordsContainer 中找到一个与 wordsQuery[i] 有 最长公共后缀 的字符串。如果 wordsContainer 中有两个或者更多字符串有最长公共后缀,那么答案为长度 最短 的。如果有超过两个字符串有 相同 最短长度,那么答案为它们在 wordsContainer 中出现 更早 的一个。

请你返回一个整数数组 ans ,其中 ans[i]是 wordsContainer中与 wordsQuery[i] 有 最长公共后缀 字符串的下标。

示例 1:

输入:wordsContainer = [“abcd”,“bcd”,“xbcd”], wordsQuery = [“cd”,“bcd”,“xyz”]

输出:[1,1,1]

解释:

我们分别来看每一个 wordsQuery[i] :

对于 wordsQuery[0] = “cd” ,wordsContainer 中有最长公共后缀 “cd” 的字符串下标分别为 0 ,1 和 2 。这些字符串中,答案是下标为 1 的字符串,因为它的长度为 3 ,是最短的字符串。

对于 wordsQuery[1] = “bcd” ,wordsContainer 中有最长公共后缀 “bcd” 的字符串下标分别为 0 ,1 和 2 。这些字符串中,答案是下标为 1 的字符串,因为它的长度为 3 ,是最短的字符串。

对于 wordsQuery[2] = “xyz” ,wordsContainer 中没有字符串跟它有公共后缀,所以最长公共后缀为 “” ,下标为 0 ,1 和 2 的字符串都得到这一公共后缀。这些字符串中, 答案是下标为 1 的字符串,因为它的长度为 3 ,是最短的字符串。

示例 2:

输入:wordsContainer = [“abcdefgh”,“poiuygh”,“ghghgh”], wordsQuery = [“gh”,“acbfgh”,“acbfegh”]

输出:[2,0,2]

解释:

我们分别来看每一个 wordsQuery[i] :

对于 wordsQuery[0] = “gh” ,wordsContainer 中有最长公共后缀 “gh” 的字符串下标分别为 0 ,1 和 2 。这些字符串中,答案是下标为 2 的字符串,因为它的长度为 6 ,是最短的字符串。

对于 wordsQuery[1] = “acbfgh” ,只有下标为 0 的字符串有最长公共后缀 “fgh” 。所以尽管下标为 2 的字符串是最短的字符串,但答案是 0 。

对于 wordsQuery[2] = “acbfegh” ,wordsContainer 中有最长公共后缀 “gh” 的字符串下标分别为 0 ,1 和 2 。这些字符串中,答案是下标为 2 的字符串,因为它的长度为 6 ,是最短的字符串。

提示:

1 <= wordsContainer.length, wordsQuery.length <= 104

1 <= wordsContainer[i].length <= 5 * 103

1 <= wordsQuery[i].length <= 5 * 103

wordsContainer[i] 只包含小写英文字母。

wordsQuery[i] 只包含小写英文字母。

wordsContainer[i].length 的和至多为 5 * 105

wordsQuery[i].length 的和至多为 5 * 105

字典树

将字符串全部反序,则后缀变成前缀。用字典树实现。

每个节点,包括根节点都要记录经过此节点的字符串的长度和下标。

内存超了的原因:

字典树的字符数最多是 5*105 每个字符 26种。合计:1.3e7, 接近内存超出的边缘。

将向量改成 哈希映射后,还是不行。

开始用set<pair<int,int>> 记录最小值,换成pair就可以了。

代码

template<class TData = char, int iTypeNum = 26, TData cBegin = 'a'>
class CTrieNodeEx
{
public:
  CTrieNodeEx* AddChar(TData ele)
  {
    const int index = ele - cBegin;
    if (!m_vPChilds.count(index))   
    {
      m_vPChilds[index] = new CTrieNodeEx();
    }
    return m_vPChilds[index];
  }
  CTrieNodeEx* GetChild(TData ele)
  {
    const int index = ele - cBegin;
    if (m_vPChilds.count(index))
    {
      return m_vPChilds[index];
    }
    return nullptr;
  }
public:
  pair<int, int> m_lenIndex = { 1000'1000,0 };
protected:
  unordered_map<int, CTrieNodeEx*> m_vPChilds;
};
template<class TData = char, int iTypeNum = 26, TData cBegin = 'a'>
class CTrieEx
{
public: 
  void Add(const string& str,int index)
  {
    auto lenIndex = std::make_pair((int)str.length(),index);
    auto pNode = &m_root;
    if (lenIndex < pNode->m_lenIndex)
    {
      pNode->m_lenIndex = lenIndex;
    }
    for (const auto& ch : str )
    {
      pNode = pNode->AddChar(ch);
      if (lenIndex < pNode->m_lenIndex)
      {
        pNode->m_lenIndex = lenIndex;
      }
    }
  }
  int Find(const string& str)
  {
    auto pNode = &m_root;
    for (const auto& ch : str)
    {
      auto tmp = pNode->GetChild(ch);
      if (nullptr == tmp)
      {
        break;
      }
      pNode = tmp;
    }
    return pNode->m_lenIndex.second;
  }
  CTrieNodeEx<TData, iTypeNum, cBegin> m_root;  
};
class Solution {
public:
  vector<int> stringIndices(vector<string>& wordsContainer, vector<string>& wordsQuery) {
    CTrieEx trie;
    for (int i = 0; i < wordsContainer.size(); i++)
    {
      string strRev(wordsContainer[i].rbegin(), wordsContainer[i].rend());
      trie.Add(strRev, i);
    }
    vector<int> vRet;
    for (int i = 0; i < wordsQuery.size(); i++)
    {
      string strRev(wordsQuery[i].rbegin(), wordsQuery[i].rend());
      vRet.emplace_back(trie.Find(strRev));
    }
    return vRet;
  }
};

测试用例

template<class T, class T2>
void Assert(const T& t1, const T2& t2)
{
  assert(t1 == t2);
}
template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{
  if (v1.size() != v2.size())
  {
    assert(false);
    return;
  }
  for (int i = 0; i < v1.size(); i++)
  {
    Assert(v1[i], v2[i]);
  }
}
int main()
{
  vector<string> wordsContainer, wordsQuery;
  {
    Solution sln;
    wordsContainer = { "abcd", "bcd", "xbcd" }, wordsQuery = { "cd", "bcd", "xyz" };
    auto res = sln.stringIndices(wordsContainer, wordsQuery);
    Assert({ 1,1,1 }, res);
  }
  {
    Solution sln;
    wordsContainer = { "abcdefgh", "poiuygh", "ghghgh" }, wordsQuery = { "gh", "acbfgh", "acbfegh" };
    auto res = sln.stringIndices(wordsContainer, wordsQuery);
    Assert({ 2,0,2 }, res);
  }
}


扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。

https://edu.csdn.net/course/detail/38771

如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程

https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版

https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17

或者 操作系统:win10 开发环境: VS2022 C++17

如无特殊说明,本算法用**C++**实现。

相关文章
|
2月前
删除字符串中的除字母外的字符
【10月更文挑战第31天】删除字符串中的除字母外的字符。
41 4
|
8月前
|
算法 测试技术 C#
【前缀和】3085. 成为 K 特殊字符串需要删除的最少字符数
【前缀和】3085. 成为 K 特殊字符串需要删除的最少字符数
|
8月前
|
人工智能 自然语言处理 算法
【动态规划】【字符串】【前缀和】1639通过给定词典构造目标字符串的方案数
【动态规划】【字符串】【前缀和】1639通过给定词典构造目标字符串的方案数
|
数据安全/隐私保护 索引
labview字符串数据长度连接子字符串大小写替换删除插入日期匹配
labview字符串数据长度连接子字符串大小写替换删除插入日期匹配
269 0
有一个长度是10的数组,数组内有10个人名,要求去掉重复的人名,并输出
有一个长度是10的数组,数组内有10个人名,要求去掉重复的人名,并输出
330 0
AcWing 779. 最长公共字符串后缀
AcWing 779. 最长公共字符串后缀
110 0
AcWing 779. 最长公共字符串后缀
|
机器学习/深度学习 算法
每日算法刷题Day10-字符串最大跨距、最长公共字符串后缀
⭐每日算法题解系列文章旨在精选重点与易错的算法题,总结常见的算法思路与可能出现的错误,与笔者另一系列文章有所区别,并不是以知识点的形式提升算法能力,而是以实战习题的形式理解算法,使用算法。
262 0
遍历所有子串,回文串,公共前缀,旋转字符串
遍历所有子串,回文串,公共前缀,旋转字符串
103 0
|
关系型数据库 MySQL 索引
B+树索引使用(7)匹配列前缀,匹配值范围(十九)
B+树索引使用(7)匹配列前缀,匹配值范围(十九)