【码道初阶】Leetcode771 宝石与石头:Set 判成员 vs List 判成员(同题两种写法的差距)

771 宝石与石头:Set 判成员 vs List 判成员(同题两种写法的差距)

题目给两个字符串:

  • jewels:宝石的种类(每个字符都是一种宝石类型,而且保证不重复)
  • stones:我拥有的石头(每个字符是一块石头的类型)

目标:统计 stones 中有多少字符出现在 jewels 里。

题目还强调一点:区分大小写 ,所以 'a''A' 不是一种类型。


1. 把题目翻译成"算法语言"

这是一个标准的"集合成员判断 + 计数"问题:

  • 先把 jewels 的所有字符放到一个"可快速判断是否存在"的结构里
  • 再遍历 stones,每看到一个字符就问一句:"它是不是宝石?"
  • 如果是,计数 +1

关键就在"可快速判断是否存在"的数据结构选型。


2. 解法一:HashSet(最贴合题意,平均 O(1) 成员判断)

java 复制代码
class Solution {
    public int numJewelsInStones(String jewels, String stones) {
        HashSet<Character> set = new HashSet<>();
        for(int i = 0;i<jewels.length();i++){
           set.add(jewels.charAt(i));
        }
        int hasJews = 0;
        for(int i = 0;i<stones.length();i++){
            if(set.contains(stones.charAt(i))) hasJews ++;
        }
        return hasJews;
    }
}

核心思路

HashSet 的天然用途就是:去重 + 快速判断是否存在

  • set.add(x):把元素放进集合
  • set.contains(x):判断元素是否在集合里(平均 O(1))

题目保证 jewels 的字符唯一,这跟 set 的特性完美契合。

代码逻辑拆解

java 复制代码
HashSet<Character> set = new HashSet<>();
for (int i = 0; i < jewels.length(); i++) {
    set.add(jewels.charAt(i));
}

这一步是"建宝石集合":把每种宝石类型存到 set 中。

接着遍历 stones 并计数:

java 复制代码
int hasJews = 0;
for (int i = 0; i < stones.length(); i++) {
    if (set.contains(stones.charAt(i))) hasJews++;
}
return hasJews;

每块石头只做一次 contains 判断,是宝石就加一。

复杂度

  • 建 set:O(|jewels|)
  • 遍历 stones:O(|stones|),每次 contains 平均 O(1)
  • 总体:O(|jewels| + |stones|)
  • 空间:O(|jewels|)

这种写法属于"题目想让我写的答案"。


3. 解法二:List + contains(能做对,但成员判断慢很多)

java 复制代码
class Solution {
    public int numJewelsInStones(String jewels, String stones) {
        List<Character> jew = new ArrayList<>();
        int hasJews = 0;
        for(int i = 0;i<jewels.length();i++){
            jew.add(jewels.charAt(i));
        }
        for(int i = 0;i<stones.length();i++){
            if(jew.contains(stones.charAt(i))) hasJews++;
        }
        return hasJews;
    }
}

核心思路

先把 jewels 存到 List<Character> 里,然后对 stones 的每个字符调用 list.contains() 来判断是否是宝石。

代码逻辑拆解

java 复制代码
List<Character> jew = new ArrayList<>();
for(int i = 0; i < jewels.length(); i++){
    jew.add(jewels.charAt(i));
}

这一步把宝石类型顺序放进列表。

计数阶段:

java 复制代码
for(int i = 0; i < stones.length(); i++){
    if(jew.contains(stones.charAt(i))) hasJews++;
}

这里的关键是:ArrayList.contains(x) 内部是线性扫描

它会从头到尾挨个比较,直到找到 x 或扫完整个列表才停。

复杂度

  • 建 list:O(|jewels|)
  • 遍历 stones:每次 contains 是 O(|jewels|)
  • 总体:O(|stones| * |jewels|)
  • 空间:O(|jewels|)

在本题数据范围(最多 50)时,体感差别不大;但如果把这个思路迁移到更大数据规模,这种写法会明显慢。


4. 两种写法对比:为什么 HashSet 更推荐?

4.1 成员判断的时间复杂度差距

  • HashSet.contains:平均 O(1)
  • ArrayList.containsO(m)(m = jewels 长度)

这题的核心操作就是"判断某个字符是不是宝石",判断次数是 |stones| 次。

把每次判断从 O(m) 优化到 O(1),整体就从乘法变加法。

4.2 语义表达更贴题

  • jewels 本质是"一个集合",用 Set 表达非常自然
  • List 更像"有序容器",但这里根本不需要顺序

4.3 可扩展性更强

今天 jewelsstones 长度最多 50;

如果换成长度 10 万的字符串,List 版本会立刻吃不消,而 Set 版本仍然稳。


5. 小细节:为什么 Character 也能放进 HashSet?

因为 Java 会把 char 自动装箱成 Character 对象(autoboxing)。
HashSet<Character> 存的是对象,但写 set.add(jewels.charAt(i)) 依然很自然。


总结

这题最直接的建模就是:

  • jewels 是"宝石类型集合"
  • stones 是"待检测序列"
  • 遍历 stones,遇到属于集合的就计数

因此:

  • HashSet :成员判断快,复杂度 O(n + m),表达也贴题
  • List :成员判断要线性扫,复杂度 O(n*m),能做对但不够优雅

在面试或工程里,我会优先选择 HashSet 版本:它不仅更快,而且把"宝石种类是集合"这个事实写进了代码结构里,读起来也更像在讲人话。

相关推荐
独自归家的兔3 小时前
Java性能优化实战:从基础调优到系统效率倍增 - 1
java·开发语言·性能优化
Coder_Boy_3 小时前
基于SpringAI的在线考试系统-DDD(领域驱动设计)核心概念及落地架构全总结 (2)
java·人工智能·spring boot·架构·serverless·ddd·服务网格
Coder_Boy_3 小时前
基于SpringAI的在线考试系统-考试系统DDD(领域驱动设计)实现步骤详解(2)
java·前端·数据库·人工智能·spring boot
long3163 小时前
弗洛伊德·沃肖算法 Floyd Warshall Algorithm
java·后端·算法·spring·springboot·图论
有一个好名字3 小时前
力扣-咒语和药水的成功对数
java·算法·leetcode
H Corey3 小时前
Java--面向对象之继承与多态
java·开发语言·windows·学习·算法·intellij-idea
難釋懷3 小时前
Redis命令-Hash命令
数据库·redis·哈希算法
ejinxian3 小时前
2026 年 Java 开发计划-Oracle公布
java·开发语言·java 开发计划
難釋懷3 小时前
Redis命令-List命令
数据库·redis·list
Sylvia-girl3 小时前
Java之日志框架
java·开发语言