List.contains踩坑

背景

表结构

存在字段 产品经理id ,存储所有的产品经理id,例如 31834,41437,44710

复制代码
`pm_user_ids` varchar(50)  DEFAULT NULL COMMENT '产品经理id',

代码

java 复制代码
private boolean isPM(Long userId) {
    List<String> pmUserIds= projectResponsiblePersonMapper.pmAllUserIds();
    if (pmUserIds.contains(String.valueOf(userId))) {
        log.debug("用户ID为:{},是产品经理", userId);
        return true;
    } else {
        log.debug("用户ID为:{},不是产品经理", userId);
        return false;
    }
}
sql 复制代码
<select id="pmAllUserIds" resultType="java.lang.String">
    select
        pm_user_ids
    from inja_project_responsible_person
</select>

问题

userId=44710,判断时,被判断为不是产品经理

原因

查询返回的是pm_user_ids字段的值,而不是用户ID列表。

从数据库表结构可以看出,pm_user_ids是一个包含多个用户ID的字符串(用逗号分隔),而不是单个用户ID。

java 复制代码
List<String> pmUserIds= projectResponsiblePersonMapper.pmAllUserIds();

这个方法获取到的是一个List,其中每个元素都是一个pm_user_ids字符串(如"45,46,47"),而不是单个用户ID的列表(如["45", "46", "47"])。

所以当检查pmUserIds.contains("47")时,实际上是在检查List中是否有元素恰好等于"47",而不是检查是否有元素包含"47"。

解决

  1. 修改pmAllUserIds查询,将所有pm_user_ids字段中的用户ID提取出来并去重。
xml 复制代码
<select id="pmAllUserIds" resultType="java.lang.String">
    SELECT DISTINCT SUBSTRING_INDEX(SUBSTRING_INDEX(pm_user_ids, ',', numbers.n), ',', -1) AS user_id
    FROM inja_project_responsible_person
    CROSS JOIN (
        SELECT 1 n UNION SELECT 2 UNION SELECT 3 UNION SELECT 4 UNION SELECT 5
        UNION SELECT 6 UNION SELECT 7 UNION SELECT 8 UNION SELECT 9 UNION SELECT 10
    ) numbers
    WHERE CHAR_LENGTH(pm_user_ids) - CHAR_LENGTH(REPLACE(pm_user_ids, ',', '')) >= numbers.n - 1
    AND pm_user_ids IS NOT NULL
    AND pm_user_ids != ''
</select>
  1. 或者修改java代码
java 复制代码
// ... existing code ...
    private boolean isPM(Long userId) {
        List<String> pmUserIds = projectResponsiblePersonMapper.pmAllUserIds();
        String userIdStr = String.valueOf(userId);
        
        // 遍历所有pm_user_ids字符串,检查是否包含当前用户ID
        for (String pmIds : pmUserIds) {
            if (pmIds != null && !pmIds.isEmpty()) {
                // 按逗号分割并检查是否包含userId
                String[] ids = pmIds.split(",");
                for (String id : ids) {
                    if (id.trim().equals(userIdStr)) {
                        log.debug("用户ID为:{},是产品经理", userId);
                        return true;
                    }
                }
            }
        }
        
        log.debug("用户ID为:{},不是产品经理", userId);
        return false;
    }
// ... existing code ...
相关推荐
jinanwuhuaguo19 分钟前
(第三十三篇)五月的文明奠基:OpenClaw 2026.5.2版本的文明级解读
android·java·开发语言·人工智能·github·拓扑学·openclaw
xmjd msup1 小时前
spring security 超详细使用教程(接入springboot、前后端分离)
java·spring boot·spring
952361 小时前
SpringBoot统一功能处理
java·spring boot·后端
Lyyaoo.1 小时前
优惠券秒杀业务分析
java·开发语言
消失的旧时光-19431 小时前
统一并发模型:线程、Reactor、协程本质是一件事(从线程到协程 · 第6篇·终章)
java·python·算法
勿忘初心12211 小时前
Java 国密 SM4 加密工具类实战(Hutool + BouncyCastle)|企业级数据加密 + 兼容 JDK8
java·数据安全·数据加密·后端开发·企业级开发·国密 sm4
庞轩px1 小时前
第8篇:原子类与CAS底层原理——无锁并发的实现
java·cas·乐观锁·aba·无锁编程·自旋
rleS IONS2 小时前
SpringBoot中自定义Starter
java·spring boot·后端
苍煜2 小时前
慢SQL优化实战教学
java·数据库·sql
AI进化营-智能译站2 小时前
ROS2 C++开发系列16-智能指针管理传感器句柄|告别ROS2节点内存泄漏与野指针
java·c++·算法·ai