You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
64 lines
2.9 KiB
Java
64 lines
2.9 KiB
Java
package com.supervision.util;
|
|
|
|
import com.supervision.domain.QaSimilarityQuestionAnswer;
|
|
import lombok.extern.slf4j.Slf4j;
|
|
import org.springframework.ai.document.Document;
|
|
import org.springframework.ai.vectorstore.RedisVectorStore;
|
|
import org.springframework.ai.vectorstore.SearchRequest;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Comparator;
|
|
import java.util.List;
|
|
import java.util.Optional;
|
|
|
|
@Slf4j
|
|
public class SimilarityUtil {
|
|
|
|
private static final RedisVectorStore redisVectorStore = SpringBeanUtil.getBean(RedisVectorStore.class);
|
|
|
|
/**
|
|
* 相似度比较,只找最高的
|
|
*
|
|
* @param question 问题
|
|
* @return 最高的TOP
|
|
*/
|
|
public static Optional<QaSimilarityQuestionAnswer> talkRedisVectorWithScoreByFirst(String question) {
|
|
List<QaSimilarityQuestionAnswer> qaSimilarityQuestionAnswers = talkRedisVectorWithScore(question);
|
|
return qaSimilarityQuestionAnswers.stream().findFirst();
|
|
}
|
|
|
|
/**
|
|
* 相似度比较,找出所有的
|
|
*
|
|
* @param question 问题
|
|
* @return 相似度最高的top10
|
|
*/
|
|
public static List<QaSimilarityQuestionAnswer> talkRedisVectorWithScore(String question) {
|
|
log.info("开始调用talkQaSimilarity,问题:{}", question);
|
|
try {
|
|
// 走Redis向量库进行比较,找出最高的top10
|
|
List<Document> documents = redisVectorStore.similaritySearch(SearchRequest
|
|
.query(question)
|
|
.withTopK(10));
|
|
return documents.stream().map(document -> {
|
|
QaSimilarityQuestionAnswer qaSimilarityQuestionAnswer = new QaSimilarityQuestionAnswer();
|
|
qaSimilarityQuestionAnswer.setMatchQuestion(document.getContent());
|
|
qaSimilarityQuestionAnswer.setQuestionId(String.valueOf(document.getMetadata().get("questionId")));
|
|
qaSimilarityQuestionAnswer.setDictId(String.valueOf(document.getMetadata().get("dictId")));
|
|
qaSimilarityQuestionAnswer.setLibraryQuestionId(String.valueOf(document.getMetadata().get("libraryQuestionId")));
|
|
// 1- 可以使数据进行排序,相似度越高,数值越大(redis相似度给的数据是越小相似度越高)
|
|
// -0.25目的是使数据趋近于中间,相似度不要太大(太大也不好调整),以使我们数据和张总之前提供的方法相似度差异稍小一点,但是不能小于0,如果小于0,取一个较大的值
|
|
double score = Math.max(0, 1 - Double.parseDouble(String.valueOf(document.getMetadata().get("vector_score"))) - 0.25);
|
|
qaSimilarityQuestionAnswer.setMatchScore(score);
|
|
return qaSimilarityQuestionAnswer;
|
|
// 排序,降序,取最高的
|
|
}).sorted(Comparator.comparing(QaSimilarityQuestionAnswer::getMatchScore).reversed()).toList();
|
|
} catch (Exception e) {
|
|
log.error("调用talkQaSimilarity error ", e);
|
|
return new ArrayList<>();
|
|
}
|
|
}
|
|
|
|
|
|
}
|