【位运算 状态压缩 枚举子集】1178. 猜字谜

简介: 【位运算 状态压缩 枚举子集】1178. 猜字谜

本文涉及知识点

枚举子集 位运算

LeetCode 1178. 猜字谜

外国友人仿照中国字谜设计了一个英文版猜字谜小游戏,请你来猜猜看吧。

字谜的迷面 puzzle 按字符串形式给出,如果一个单词 word 符合下面两个条件,那么它就可以算作谜底:

单词 word 中包含谜面 puzzle 的第一个字母。

单词 word 中的每一个字母都可以在谜面 puzzle 中找到。

例如,如果字谜的谜面是 “abcdefg”,那么可以作为谜底的单词有 “faced”, “cabbage”, 和 “baggage”;而 “beefed”(不含字母 “a”)以及 “based”(其中的 “s” 没有出现在谜面中)都不能作为谜底。

返回一个答案数组 answer,数组中的每个元素 answer[i] 是在给出的单词列表 words 中可以作为字谜迷面 puzzles[i] 所对应的谜底的单词数目。

示例:

输入:

words = [“aaaa”,“asas”,“able”,“ability”,“actt”,“actor”,“access”],

puzzles = [“aboveyz”,“abrodyz”,“abslute”,“absoryz”,“actresz”,“gaswxyz”]

输出:[1,1,3,2,4,0]

解释:

1 个单词可以作为 “aboveyz” 的谜底 : “aaaa”

1 个单词可以作为 “abrodyz” 的谜底 : “aaaa”

3 个单词可以作为 “abslute” 的谜底 : “aaaa”, “asas”, “able”

2 个单词可以作为 “absoryz” 的谜底 : “aaaa”, “asas”

4 个单词可以作为 “actresz” 的谜底 : “aaaa”, “asas”, “actt”, “access”

没有单词可以作为 “gaswxyz” 的谜底,因为列表中的单词都不含字母 ‘g’。

提示:

1 <= words.length <= 105

4 <= words[i].length <= 50

1 <= puzzles.length <= 104

puzzles[i].length == 7

words[i][j], puzzles[i][j] 都是小写英文字母

每个 puzzles[i] 所包含的字符都不重复。

状态压缩

状态压缩:如果存在字母ch ,则第(ch-‘a’)位为1。

mMaskCnt[i] 记录包括字母’a’+i的所有words[k]的mask的数量

通过p枚举puzzles

假定p的mask是mask1,则枚举mask所有的自己sub,累加:mMaskCnt[p[0]-‘a’][sub]

注意

for (int sub = iMask; sub; sub = (sub - 1) & iMask)

这样的时间复杂度是:O(27)

不能for(sub = 0 ;sub < iMask;sub++) 这样的时间复杂度是O(226)

代码

核心代码

class Solution {
public:
  vector<int> findNumOfValidWords(vector<string>& words, vector<string>& puzzles) {
    unordered_map<int, int> mMaskCnt[26];
    for (const auto& s : words) {
      const int iMask = Mask(s);
      for (int i = 0; i < 26; i++) {
        if ((1 << i) & iMask) {
          mMaskCnt[i][iMask]++;
        }
      }
    }
    vector<int> vRet;
    for (const auto& s : puzzles) {
      const int iMask = Mask(s);
      int iRet = 0;
      for (int sub = iMask; sub; sub = (sub - 1) & iMask) {
        if (mMaskCnt[s[0] - 'a'].count(sub)) {
          iRet += mMaskCnt[s[0] - 'a'][sub];
        }       
      }
      vRet.emplace_back(iRet);
    }
    return vRet;
  }
  int Mask(const string& s) {
    int iMask = 0;
    for (const auto& ch : s) {
      iMask |= (1 << (ch - 'a'));
    }
    return iMask;
  }
};

测试用例

template<class T>
void Assert(const T& t1, const T& 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> words, puzzles;
  {
    Solution sln;
    words = { "apple","pleas","please" },
      puzzles = { "aelpsxy","saelpxy","xaelpsy" };
    auto res = sln.findNumOfValidWords(words, puzzles);
    Assert(vector<int>{ 3, 2, 0}, res);
  }
  {
    Solution sln;
    words = { "apple","pleas","please" },
      puzzles = { "aelwxyz","aelpxyz","aelpsxy","saelpxy","xaelpsy" };
    auto res = sln.findNumOfValidWords(words, puzzles);
    Assert(vector<int>{0, 1, 3, 2, 0}, res);
  }
  {
    Solution sln;
    words = { "aaaa", "asas", "able", "ability", "actt", "actor", "access" },
      puzzles = { "aboveyz", "abrodyz", "abslute", "absoryz", "actresz", "gaswxyz" };
    auto res = sln.findNumOfValidWords(words, puzzles);
    Assert(vector<int>{1, 1, 3, 2, 4, 0}, res);
  }
  
}


相关文章
|
7月前
|
移动开发 C++
【洛谷 P1157】组合的输出 题解(深度优先搜索+枚举子集)
该问题要求编程输出从1到n中选择r个元素的所有组合,组合按字典序排列。输入包含两自然数n和r(1&lt;n&lt;21, 0≤r≤n)。输出每个组合时,每个数字占据3个字符宽度。提供的AC代码使用C++,通过递归搜索方法枚举子集。样例输入为5 3,输出显示所有3个元素的组合。
61 0
|
8月前
|
机器学习/深度学习 算法 测试技术
【位运算 子集状态压缩】982按位与为零的三元组
【位运算 子集状态压缩】982按位与为零的三元组
|
8月前
|
算法 测试技术 C++
【动态规划 区间dp 位运算】3117. 划分数组得到最小的值之和
【动态规划 区间dp 位运算】3117. 划分数组得到最小的值之和
【动态规划 区间dp 位运算】3117. 划分数组得到最小的值之和
|
7月前
|
算法
数据结构和算法——散列函数的构造方法(直接定址法、除留余数法、数字分析法、折叠法、平方取中法、ASCII码加和法、前三字符移位法)
数据结构和算法——散列函数的构造方法(直接定址法、除留余数法、数字分析法、折叠法、平方取中法、ASCII码加和法、前三字符移位法)
91 0
|
8月前
|
存储 算法 测试技术
位运算、状态压缩、枚举子集汇总
位运算、状态压缩、枚举子集汇总
|
存储 算法
【LeetCode】每日一题&最后一个单词的长度&投票法求解多数元素&异或操作符巧解只出现一次的数字&整数反转
【LeetCode】每日一题&最后一个单词的长度&投票法求解多数元素&异或操作符巧解只出现一次的数字&整数反转
|
8月前
|
存储 算法 Java
【算法训练-数组 二】【元素组合】两数之和、三数之和
【算法训练-数组 二】【元素组合】两数之和、三数之和
66 0
|
人工智能
前缀和 差分数组编程题集合
前缀和 差分数组编程题集合
|
算法 Go C++
leetcode-2321. 拼接数组的最大分数(差分+枚举)
但其实是在找两个数组之间[Left:Right]区间最大差值,差值我们自然而然的就可以想到差分,但是我们这里求的差分数组并不是两个数组前后元素的差值,而是两个数组同一下标的元素差值,这样我们只要找这段差分区间和的最大值就行了
88 0
leetcode-2321. 拼接数组的最大分数(差分+枚举)
|
算法
算法练习——(1)找数组中唯一成对的那个数(异或)
——如何找数组中唯一成对的那个数(数组特殊) 1-1000这一千个数放在含有1001个元素的数组中,只有唯一的一个元素值重复,其他均只出现一次。 每个数组元素只能访问一次,设计一个算法,将他找出来;不用辅助储存空间,设计一个算法实现.
124 0

热门文章

最新文章

下一篇
开通oss服务