修改用户个人资料识别逻辑

This commit is contained in:
zk
2026-03-26 19:05:31 +08:00
parent 3f8af947f0
commit fcc13d3496
3 changed files with 348 additions and 331 deletions
@@ -0,0 +1,341 @@
package org.jiayunet.service;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.toolkit.IdWorker;
import com.fasterxml.jackson.databind.JsonNode;
import lombok.extern.slf4j.Slf4j;
import org.jiayunet.ai.AiChatAbility;
import org.jiayunet.mapper.*;
import org.jiayunet.pojo.po.*;
import org.jiayunet.pojo.vo.UserHonorsVo;
import org.jiayunet.tool.HttpTool;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
import java.time.Instant;
import java.util.*;
import java.util.stream.Collectors;
/**
* 用户简历分析服务
* <p>主要功能:用户保存简历后异步分析,提取学校等级、经历质量、专业、技能等维度数据</p>
* <p>依赖:AiChatAbilityAI调用)、DictCacheService(专业分类缓存)</p>
* <p>使用表:bg_user_profile(主表,读取/更新)、bg_user_profile_*5张子表,读取)、
* bg_skill_tag(技能入库)、bg_user_profile_skill_tag_relation(关联表)</p>
*
* @author zk
*/
@Slf4j
@Service
public class UserProfileAnalyzeService {
@Autowired
private AiChatAbility aiChatAbility;
@Autowired
private DictCacheService dictCacheService;
@Autowired
private UserProfileMapper userProfileMapper;
@Autowired
private UserProfileEducationMapper educationMapper;
@Autowired
private UserProfileWorkMapper workMapper;
@Autowired
private UserProfileInternshipMapper internshipMapper;
@Autowired
private UserProfileProjectMapper projectMapper;
@Autowired
private UserProfileCompetitionMapper competitionMapper;
@Autowired
private SkillTagMapper skillTagMapper;
@Autowired
private UserProfileSkillTagRelationMapper relationMapper;
/**
* 异步分析用户简历
* <p>1. 查询用户完整简历 2. 第一次AI综合分析 3. 第二次AI专业归一化 4. 第三次AI技能提取</p>
*/
@Async("userProfileAsyncExecutor")
public void analyzeUserProfile(Long userId) {
try {
log.info("开始分析用户简历, userId={}", userId);
// 1. 查询用户完整简历
UserProfile profile = userProfileMapper.selectOne(new LambdaQueryWrapper<UserProfile>().eq(UserProfile::getUserId, userId));
if (profile == null) {
log.info("用户主表为空,清空技能标签, userId={}", userId);
clearSkillRelations(userId);
return;
}
List<UserProfileEducation> educationList = educationMapper.selectList(new LambdaQueryWrapper<UserProfileEducation>().eq(UserProfileEducation::getUserId, userId));
List<UserProfileWork> workList = workMapper.selectList(new LambdaQueryWrapper<UserProfileWork>().eq(UserProfileWork::getUserId, userId));
List<UserProfileInternship> internshipList = internshipMapper.selectList(new LambdaQueryWrapper<UserProfileInternship>().eq(UserProfileInternship::getUserId, userId));
List<UserProfileProject> projectList = projectMapper.selectList(new LambdaQueryWrapper<UserProfileProject>().eq(UserProfileProject::getUserId, userId));
List<UserProfileCompetition> competitionList = competitionMapper.selectList(new LambdaQueryWrapper<UserProfileCompetition>().eq(UserProfileCompetition::getUserId, userId));
// 2. 数据有效性检查
if (educationList.isEmpty() && workList.isEmpty() && internshipList.isEmpty() && projectList.isEmpty() && competitionList.isEmpty()) {
log.info("用户所有子表为空,清空技能标签, userId={}", userId);
clearSkillRelations(userId);
return;
}
String profileJson = buildProfileJson(profile, educationList, workList, internshipList, projectList, competitionList);
// 3. 第一次AI:简历综合分析(失败不影响后续)
try {
analyzeProfileDimensions(profile.getId(), profileJson);
} catch (Exception e) {
log.warn("简历综合分析失败, userId={}", userId, e);
}
// 4. 第二次AI:专业归一化(失败不影响后续)
try {
analyzeMajor(profile.getId(), profileJson);
} catch (Exception e) {
log.warn("专业归一化失败, userId={}", userId, e);
}
// 5. 第三次AI:技能提取
try {
extractSkillTags(userId, profileJson);
} catch (Exception e) {
log.warn("技能提取失败, userId={}", userId, e);
}
log.info("用户简历分析完成, userId={}", userId);
} catch (Exception e) {
log.error("用户简历分析异常, userId={}", userId, e);
}
}
/**
* 第一次AI:简历综合分析
* <p>一次性识别学校等级、公司背书、经历时长、职责深度、量化产出、荣誉</p>
*/
private void analyzeProfileDimensions(Long profileId, String profileJson) {
String systemPrompt = """
你是一个简历分析助手。根据用户简历,分析以下维度并返回JSON。
返回JSON格式:
{
"schoolRank": 1-4的数字(1=C9/985/QS前50 2=211/双一流/QS前200 3=普通一本/QS前500 4=其他,无教育经历则null),
"companyPrestige": 1-4的数字(1=名企(世界500强/互联网大厂/知名国企/独角兽) 2=普通实习 3=仅校内活动 4=无经历,取最高档),
"experienceDuration": 1-3的数字(1=实习≥3月 2=1-3月 3=≤1月,取最长的一段,无实习则null),
"roleDepth": 1-3的数字(1=主导/创新(主导、核心负责、从0到1、架构设计) 2=执行/应用(负责、开发、实现、维护) 3=辅助/学习(协助、参与、了解),取最高档),
"outputQuality": 1-3的数字(1=有量化结果(含数字比率如提升20%) 2=有具体产出(明确产出物但无数字) 3=纯描述,取最高档),
"honors": {"national":["具体奖项名"],"provincial":["具体奖项名"],"school":["具体奖项/证书名"],"paper":["具体论文名"]}
}
规则:
1. 每个维度独立判断,某维度信息不足则设为null
2. 多段经历取最高档(最优表现)
3. honors中只填简历中明确提到的,没有则空数组
4. 只返回JSON,不要其他内容
""";
String userMessage = "【用户简历】\n" + profileJson;
String aiResponse = aiChatAbility.chat(systemPrompt, userMessage);
String json = cleanAiResponse(aiResponse);
try {
JsonNode root = HttpTool.objectMapper.readTree(json);
UserProfile update = new UserProfile();
update.setId(profileId);
update.setSchoolRank(root.path("schoolRank").isNull() ? null : root.path("schoolRank").asInt());
update.setCompanyPrestige(root.path("companyPrestige").isNull() ? null : root.path("companyPrestige").asInt());
update.setExperienceDuration(root.path("experienceDuration").isNull() ? null : root.path("experienceDuration").asInt());
update.setRoleDepth(root.path("roleDepth").isNull() ? null : root.path("roleDepth").asInt());
update.setOutputQuality(root.path("outputQuality").isNull() ? null : root.path("outputQuality").asInt());
// 解析 honors
JsonNode honorsNode = root.path("honors");
if (!honorsNode.isMissingNode() && !honorsNode.isNull()) {
UserHonorsVo honors = HttpTool.objectMapper.treeToValue(honorsNode, UserHonorsVo.class);
update.setHonors(honors);
}
update.setUpdateTime(Instant.now());
userProfileMapper.updateById(update);
} catch (Exception e) {
log.warn("简历综合分析AI返回解析失败: {}", json, e);
}
}
/**
* 第二次AI:专业归一化
* <p>传入教育经历和三级专业分类列表,AI返回用户专业ID数组 → 更新 bg_user_profile.major_ids</p>
*/
private void analyzeMajor(Long profileId, String profileJson) {
String systemPrompt = """
你是一个专业归一化助手。根据用户的教育经历,从专业分类列表中匹配用户的专业。
返回JSON格式:
{
"majorIds": [专业ID数组,从专业列表中选择最匹配的,可多个(如本科+硕士不同专业),无法判断则空数组]
}
规则:
1. 只能从给定专业列表中选择ID
2. 每段教育经历匹配一个最相关的专业
3. 无教育经历或专业信息不足时返回空数组
4. 只返回JSON,不要其他内容
""";
String userMessage = "【用户简历】\n" + profileJson + "\n\n【专业分类列表】\n" + dictCacheService.getMajorCategoryText();
String aiResponse = aiChatAbility.chat(systemPrompt, userMessage);
String json = cleanAiResponse(aiResponse);
try {
JsonNode root = HttpTool.objectMapper.readTree(json);
List<Long> majorIds = new ArrayList<>();
JsonNode idsNode = root.path("majorIds");
if (idsNode.isArray()) {
for (JsonNode node : idsNode) {
long id = node.asLong(0);
if (id > 0 && !majorIds.contains(id)) {
majorIds.add(id);
}
}
}
UserProfile update = new UserProfile();
update.setId(profileId);
update.setMajorIds(majorIds.isEmpty() ? null : majorIds);
update.setUpdateTime(Instant.now());
userProfileMapper.updateById(update);
} catch (Exception e) {
log.warn("专业归一化AI返回解析失败: {}", json, e);
}
}
/**
* 第三次AI:技能提取
* <p>自由提取用户技能,prompt与岗位侧保持一致,INSERT IGNORE入bg_skill_tag → 写关联表</p>
*/
private void extractSkillTags(Long userId, String profileJson) {
String systemPrompt = """
你是一个技能提取助手。根据用户简历,提取该用户具备的核心专业能力和工具技能。
返回JSON数组格式,如:["java", "spring boot", "mysql", "redis"]
规则:
1. 统一使用小写字母
2. 尽量简短,使用业界通用缩写(如 js 而非 javascriptk8s 而非 kubernetes
3. 提取范围包括:技术栈、专业领域知识、行业工具、专业资质能力等
4. 不提取纯软技能(如沟通能力、团队协作、学习能力、积极主动)
5. 如果简历中完全没有专业能力体现,返回空数组 []
6. 最多15个,按熟练度排序
7. 只返回JSON数组,不要其他内容
示例1(技术岗):
输入:3年Java开发经验,熟悉Spring Boot、MySQL、Redis
输出:["java", "spring boot", "mysql", "redis"]
示例2(财务岗):
输入:负责费用管理与审核,月度经营利润分析,会计学专业
输出:["财务管理", "会计核算", "经营分析", "费用审核"]
示例3(科研岗):
输入:开展生物合成相关研究,发酵工程方向博士
输出:["生物合成", "发酵工程"]
示例4(无专业能力):
输入:具备较强的沟通能力和创新意识,积极主动
输出:[]
""";
String userMessage = "【用户简历】\n" + profileJson;
String aiResponse = aiChatAbility.chat(systemPrompt, userMessage);
String json = cleanAiResponse(aiResponse);
try {
JsonNode arrayNode = HttpTool.objectMapper.readTree(json);
if (!arrayNode.isArray() || arrayNode.isEmpty()) {
clearSkillRelations(userId);
return;
}
List<Long> skillTagIds = new ArrayList<>();
for (JsonNode node : arrayNode) {
String skillName = node.asText("").trim().toLowerCase();
if (skillName.isBlank() || skillName.length() > 50) {
continue;
}
Long tagId = findOrCreateSkillTag(skillName);
if (tagId != null && !skillTagIds.contains(tagId)) {
skillTagIds.add(tagId);
}
}
replaceSkillRelations(userId, skillTagIds);
} catch (Exception e) {
log.warn("技能提取AI返回解析失败: {}", json, e);
}
}
/**
* 查找或创建技能标签(依靠数据库唯一索引保证并发安全)
*/
private Long findOrCreateSkillTag(String name) {
skillTagMapper.insertIgnore(IdWorker.getId(), name, Instant.now());
SkillTag tag = skillTagMapper.selectOne(new LambdaQueryWrapper<SkillTag>().eq(SkillTag::getName, name).last("LIMIT 1"));
return tag != null ? tag.getId() : null;
}
/** 构建用户简历JSON */
private String buildProfileJson(UserProfile profile, List<UserProfileEducation> educationList,
List<UserProfileWork> workList, List<UserProfileInternship> internshipList,
List<UserProfileProject> projectList, List<UserProfileCompetition> competitionList) {
Map<String, Object> data = new HashMap<>();
data.put("name", profile.getName());
data.put("skills", profile.getSkills());
data.put("certificates", profile.getCertificates());
data.put("education", educationList);
data.put("work", workList);
data.put("internship", internshipList);
data.put("project", projectList);
data.put("competition", competitionList);
try {
return HttpTool.objectMapper.writeValueAsString(data);
} catch (Exception e) {
log.error("用户资料序列化失败", e);
return "{}";
}
}
/** 清空用户技能标签关联 */
private void clearSkillRelations(Long userId) {
relationMapper.delete(new LambdaQueryWrapper<UserProfileSkillTagRelation>().eq(UserProfileSkillTagRelation::getUserId, userId));
}
/** 全量替换用户技能标签关联 */
private void replaceSkillRelations(Long userId, List<Long> skillTagIds) {
relationMapper.delete(new LambdaQueryWrapper<UserProfileSkillTagRelation>().eq(UserProfileSkillTagRelation::getUserId, userId));
if (skillTagIds.isEmpty()) {
return;
}
Instant now = Instant.now();
List<UserProfileSkillTagRelation> relations = skillTagIds.stream().map(tagId -> {
UserProfileSkillTagRelation relation = new UserProfileSkillTagRelation();
relation.setUserId(userId);
relation.setSkillTagId(tagId);
relation.setCreateTime(now);
return relation;
}).collect(Collectors.toList());
relationMapper.batchInsert(relations);
}
/** 清理AI返回的markdown代码块和控制字符 */
private String cleanAiResponse(String response) {
String json = response.trim();
if (json.startsWith("```")) {
json = json.replaceAll("^```\\w*\\n?", "").replaceAll("\\n?```$", "").trim();
}
json = json.replaceAll("[\\x00-\\x08\\x0B\\x0C\\x0E-\\x1F]", "");
return json;
}
}
@@ -1,324 +0,0 @@
package org.jiayunet.service;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.fasterxml.jackson.databind.JsonNode;
import lombok.extern.slf4j.Slf4j;
import org.jiayunet.ai.AiChatAbility;
import org.jiayunet.mapper.*;
import org.jiayunet.pojo.po.*;
import org.jiayunet.tool.HttpTool;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
import java.time.Instant;
import java.util.*;
import java.util.stream.Collectors;
/**
* 用户技能标签匹配服务
* <p>主要功能:根据用户个人资料,AI识别匹配的技能标签</p>
* <p>依赖:AiChatAbilityAI调用)</p>
* <p>使用表:bg_user_profile(主表)、bg_user_profile_*5张子表)、
* bg_job_category(岗位分类)、bg_skill_tag(技能标签)、
* bg_user_profile_skill_tag_relation(关联表)</p>
*
* @author zk
*/
@Slf4j
@Service
public class UserSkillTagMatchService {
@Autowired
private AiChatAbility aiChatAbility;
@Autowired
private UserProfileMapper userProfileMapper;
@Autowired
private UserProfileEducationMapper educationMapper;
@Autowired
private UserProfileWorkMapper workMapper;
@Autowired
private UserProfileInternshipMapper internshipMapper;
@Autowired
private UserProfileProjectMapper projectMapper;
@Autowired
private UserProfileCompetitionMapper competitionMapper;
@Autowired
private JobCategoryMapper jobCategoryMapper;
@Autowired
private SkillTagMapper skillTagMapper;
@Autowired
private UserProfileSkillTagRelationMapper relationMapper;
/**
* 异步匹配用户技能标签
* <p>1. 查询用户完整个人资料 2. 第一次AI调用识别二级分类 3. 查询三级分类
* 4. 查询候选技能标签 5. 第二次AI调用匹配技能标签 6. 全量替换关联表</p>
*/
@Async("userProfileAsyncExecutor")
public void matchUserSkillTags(Long userId) {
try {
log.info("开始匹配用户技能标签, userId={}", userId);
// 1. 查询用户完整个人资料
UserProfile profile = userProfileMapper.selectOne(new LambdaQueryWrapper<UserProfile>().eq(UserProfile::getUserId, userId));
if (profile == null) {
log.info("用户主表为空,清空技能标签, userId={}", userId);
clearRelations(userId);
return;
}
List<UserProfileEducation> educationList = educationMapper.selectList(new LambdaQueryWrapper<UserProfileEducation>().eq(UserProfileEducation::getUserId, userId));
List<UserProfileWork> workList = workMapper.selectList(new LambdaQueryWrapper<UserProfileWork>().eq(UserProfileWork::getUserId, userId));
List<UserProfileInternship> internshipList = internshipMapper.selectList(new LambdaQueryWrapper<UserProfileInternship>().eq(UserProfileInternship::getUserId, userId));
List<UserProfileProject> projectList = projectMapper.selectList(new LambdaQueryWrapper<UserProfileProject>().eq(UserProfileProject::getUserId, userId));
List<UserProfileCompetition> competitionList = competitionMapper.selectList(new LambdaQueryWrapper<UserProfileCompetition>().eq(UserProfileCompetition::getUserId, userId));
// 2. 数据有效性检查
if (educationList.isEmpty() && workList.isEmpty() && internshipList.isEmpty() && projectList.isEmpty() && competitionList.isEmpty()) {
log.info("用户所有子表为空,清空技能标签, userId={}", userId);
clearRelations(userId);
return;
}
// 3. 第一次 AI 调用(识别二级分类)
List<Long> categoryIds = identifyCategories(profile, educationList, workList, internshipList, projectList, competitionList);
if (categoryIds.isEmpty()) {
log.info("AI未识别出岗位分类,清空技能标签, userId={}", userId);
clearRelations(userId);
return;
}
// 4. 查询三级分类(技能标签挂在三级分类下)
List<JobCategory> level3Categories = jobCategoryMapper.selectList(new LambdaQueryWrapper<JobCategory>().eq(JobCategory::getLevel, 3).in(JobCategory::getParentId, categoryIds));
if (level3Categories.isEmpty()) {
log.info("二级分类下无三级分类,清空技能标签, userId={}", userId);
clearRelations(userId);
return;
}
// 5. 查询候选技能标签(使用三级分类ID)
List<Long> level3CategoryIds = level3Categories.stream().map(JobCategory::getId).collect(Collectors.toList());
List<SkillTag> candidateTags = skillTagMapper.selectList(new LambdaQueryWrapper<SkillTag>().in(SkillTag::getCategoryId, level3CategoryIds));
if (candidateTags.isEmpty()) {
log.info("候选技能标签为空,清空技能标签, userId={}", userId);
clearRelations(userId);
return;
}
// 6. 第二次 AI 调用(匹配技能标签)
List<Long> matchedTagIds = matchSkillTags(profile, educationList, workList, internshipList, projectList, competitionList, candidateTags);
// 7. 全量替换关联表
replaceRelations(userId, matchedTagIds);
log.info("用户技能标签匹配完成, userId={}, 匹配{}个标签", userId, matchedTagIds.size());
} catch (Exception e) {
log.error("用户技能标签匹配异常, userId={}", userId, e);
}
}
/**
* 第一次 AI 调用:识别用户所属的二级岗位分类
*/
private List<Long> identifyCategories(UserProfile profile, List<UserProfileEducation> educationList,
List<UserProfileWork> workList, List<UserProfileInternship> internshipList,
List<UserProfileProject> projectList, List<UserProfileCompetition> competitionList) {
// 构建用户资料 JSON
Map<String, Object> profileData = new HashMap<>();
profileData.put("name", profile.getName());
profileData.put("workYears", profile.getWorkYears());
profileData.put("education", educationList);
profileData.put("work", workList);
profileData.put("internship", internshipList);
profileData.put("project", projectList);
profileData.put("competition", competitionList);
String profileJson;
try {
profileJson = HttpTool.objectMapper.writeValueAsString(profileData);
} catch (Exception e) {
log.error("用户资料序列化失败", e);
return List.of();
}
// 查询所有二级分类
List<JobCategory> allCategories = jobCategoryMapper.selectList(new LambdaQueryWrapper<JobCategory>().eq(JobCategory::getLevel, 2));
// 构建分类列表文本(带一级分类名称)
Map<Long, String> parentNameMap = new HashMap<>();
List<JobCategory> parentCategories = jobCategoryMapper.selectList(new LambdaQueryWrapper<JobCategory>().eq(JobCategory::getLevel, 1));
for (JobCategory parent : parentCategories) {
parentNameMap.put(parent.getId(), parent.getName());
}
StringBuilder categoryText = new StringBuilder();
for (JobCategory category : allCategories) {
String parentName = parentNameMap.getOrDefault(category.getParentId(), "未知");
categoryText.append(String.format("ID: %d, 名称: %s > %s\n", category.getId(), parentName, category.getName()));
}
// 构建 Prompt
String systemPrompt = """
任务:根据用户个人资料,选择最匹配的二级岗位分类。
规则:
1. 选择 1-10 个最匹配的二级分类(优先核心岗位,按匹配度排序)
2. 信息不足无法判断时返回空数组 []
3. 只返回纯数字数组 JSON,不要任何其他文字
返回格式示例:
- 有分类:[23, 45, 67]
- 无法判断:[]
""";
String userMessage = "【用户个人资料】\n" + profileJson + "\n\n【二级岗位分类列表(格式:一级分类 > 二级分类)】\n" + categoryText;
try {
String aiResponse = aiChatAbility.chat(systemPrompt, userMessage);
return parseIdArray(aiResponse, allCategories.stream().map(JobCategory::getId).collect(Collectors.toSet()));
} catch (Exception e) {
log.error("AI识别岗位分类失败", e);
return List.of();
}
}
/**
* 第二次 AI 调用:从候选标签中匹配用户技能标签
*/
private List<Long> matchSkillTags(UserProfile profile, List<UserProfileEducation> educationList,
List<UserProfileWork> workList, List<UserProfileInternship> internshipList,
List<UserProfileProject> projectList, List<UserProfileCompetition> competitionList,
List<SkillTag> candidateTags) {
// 构建用户资料 JSON
Map<String, Object> profileData = new HashMap<>();
profileData.put("name", profile.getName());
profileData.put("workYears", profile.getWorkYears());
profileData.put("education", educationList);
profileData.put("work", workList);
profileData.put("internship", internshipList);
profileData.put("project", projectList);
profileData.put("competition", competitionList);
String profileJson;
try {
profileJson = HttpTool.objectMapper.writeValueAsString(profileData);
} catch (Exception e) {
log.error("用户资料序列化失败", e);
return List.of();
}
// 构建候选标签文本(按分类分组)
Map<Long, String> categoryNameMap = new HashMap<>();
Set<Long> categoryIds = candidateTags.stream().map(SkillTag::getCategoryId).collect(Collectors.toSet());
List<JobCategory> categories = jobCategoryMapper.selectList(new LambdaQueryWrapper<JobCategory>().in(JobCategory::getId, categoryIds));
for (JobCategory category : categories) {
categoryNameMap.put(category.getId(), category.getName());
}
StringBuilder tagText = new StringBuilder();
Map<Long, List<SkillTag>> groupedTags = candidateTags.stream().collect(Collectors.groupingBy(SkillTag::getCategoryId));
for (Map.Entry<Long, List<SkillTag>> entry : groupedTags.entrySet()) {
String categoryName = categoryNameMap.getOrDefault(entry.getKey(), "未知");
tagText.append("").append(categoryName).append("\n");
for (SkillTag tag : entry.getValue()) {
tagText.append(String.format("ID: %d, 名称: %s\n", tag.getId(), tag.getName()));
}
tagText.append("\n");
}
// 构建 Prompt
String systemPrompt = """
任务:根据用户个人资料,从候选技能标签中选出匹配的标签。
规则:
1. 只选择用户明确具备或可推断的技能
2. 无法匹配返回空数组 []
3. 只返回纯数字数组 JSON,不要任何其他文字
返回格式示例:
- 有匹配:[12, 34, 56]
- 无匹配:[]
""";
String userMessage = "【用户个人资料】\n" + profileJson + "\n\n【候选技能标签(按分类分组)】\n" + tagText;
try {
String aiResponse = aiChatAbility.chat(systemPrompt, userMessage);
return parseIdArray(aiResponse, candidateTags.stream().map(SkillTag::getId).collect(Collectors.toSet()));
} catch (Exception e) {
log.error("AI匹配技能标签失败", e);
return List.of();
}
}
/**
* 解析 AI 返回的 ID 数组
*/
private List<Long> parseIdArray(String aiResponse, Set<Long> validIds) {
String json = aiResponse.trim();
if (json.startsWith("```")) {
json = json.replaceAll("^```\\w*\\n?", "").replaceAll("\\n?```$", "").trim();
}
try {
JsonNode arrayNode = HttpTool.objectMapper.readTree(json);
List<Long> result = new ArrayList<>();
if (arrayNode.isArray()) {
for (JsonNode node : arrayNode) {
long id = node.asLong(0);
if (id > 0 && validIds.contains(id) && !result.contains(id)) {
result.add(id);
}
}
}
return result;
} catch (Exception e) {
log.warn("AI返回解析失败: {}", json, e);
return List.of();
}
}
/**
* 清空用户技能标签关联
*/
public void clearRelations(Long userId) {
relationMapper.delete(new LambdaQueryWrapper<UserProfileSkillTagRelation>().eq(UserProfileSkillTagRelation::getUserId, userId));
}
/**
* 全量替换用户技能标签关联
*/
public void replaceRelations(Long userId, List<Long> skillTagIds) {
// 先删除
relationMapper.delete(new LambdaQueryWrapper<UserProfileSkillTagRelation>().eq(UserProfileSkillTagRelation::getUserId, userId));
// 结果为空则只删除
if (skillTagIds.isEmpty()) {
return;
}
// 批量插入
Instant now = Instant.now();
List<UserProfileSkillTagRelation> relations = skillTagIds.stream().map(tagId -> {
UserProfileSkillTagRelation relation = new UserProfileSkillTagRelation();
relation.setUserId(userId);
relation.setSkillTagId(tagId);
relation.setCreateTime(now);
return relation;
}).collect(Collectors.toList());
relationMapper.batchInsert(relations);
}
}