【码道初阶】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 版本:它不仅更快,而且把"宝石种类是集合"这个事实写进了代码结构里,读起来也更像在讲人话。

相关推荐
JAVA学习通1 小时前
北京明光云振铎数据科技Java面经
java·开发语言·科技
贫民窟的勇敢爷们7 小时前
SpringBoot整合AOP切面编程实战,实现日志统一记录+接口权限校验
java·spring boot·spring
浅念-8 小时前
递归解题指南:LeetCode经典题全解析
数据结构·算法·leetcode·职场和发展·排序算法·深度优先·递归
Kiling_07048 小时前
Java集合进阶:Set与Collections详解
算法·哈希算法
AC赳赳老秦8 小时前
供应链专员提效:OpenClaw自动跟踪物流信息、更新库存数据,异常自动提醒
java·大数据·服务器·数据库·人工智能·自动化·openclaw
迈巴赫车主8 小时前
Java基础:list、set、map一遍过
java·开发语言
灵犀学长9 小时前
基于 Spring ThreadPoolTaskScheduler + CronTrigger 实现的动态定时任务调度系统
java·数据库·spring
洛水水9 小时前
【力扣100题】33.验证二叉搜索树
算法·leetcode·职场和发展
好家伙VCC10 小时前
【无标题】
java
小碗羊肉11 小时前
【JavaWeb | 第十一篇】文件上传(本地&阿里云OSS)
java·阿里云·servlet