修改-试卷总分设置项

This commit is contained in:
2025-12-26 18:39:17 +08:00
parent 7b52abfea3
commit 42fcb71bae
5 changed files with 538 additions and 371 deletions

View File

@@ -44,149 +44,14 @@ export class QuizController {
});
}
let questions: Question[] = [];
const remainingScore = subject.totalScore;
// 构建包含所有类别的数组,根据比重重复对应次数
const allCategories: string[] = [];
for (const [category, catRatio] of Object.entries(subject.categoryRatios)) {
if (catRatio > 0) {
// 根据比重计算该类别应占的总题目数比例
const count = Math.round((catRatio / 100) * 100); // 放大100倍避免小数问题
for (let i = 0; i < count; i++) {
allCategories.push(category);
}
}
}
// 确保总题目数至少为1
if (allCategories.length === 0) {
allCategories.push('通用');
}
// 按题型分配题目
for (const [type, typeRatio] of Object.entries(subject.typeRatios)) {
if (typeRatio <= 0) continue;
// 计算该题型应占的总分
const targetTypeScore = Math.round((typeRatio / 100) * subject.totalScore);
let currentTypeScore = 0;
let typeQuestions: Question[] = [];
// 尝试获取足够分数的题目
while (currentTypeScore < targetTypeScore) {
// 随机选择一个类别
const randomCategory = allCategories[Math.floor(Math.random() * allCategories.length)];
// 获取该类型和类别的随机题目
const availableQuestions = await QuestionModel.getRandomQuestions(
type as any,
10, // 一次获取多个,提高效率
[randomCategory]
);
if (availableQuestions.length === 0) {
break; // 该类型/类别没有题目,跳过
}
// 过滤掉已选题目
const availableUnselected = availableQuestions.filter(q =>
!questions.some(selected => selected.id === q.id)
);
if (availableUnselected.length === 0) {
break; // 没有可用的新题目了
}
// 选择分数最接近剩余需求的题目
const remainingForType = targetTypeScore - currentTypeScore;
const selectedQuestion = availableUnselected.reduce((prev, curr) => {
const prevDiff = Math.abs(remainingForType - prev.score);
const currDiff = Math.abs(remainingForType - curr.score);
return currDiff < prevDiff ? curr : prev;
});
// 添加到题型题目列表
typeQuestions.push(selectedQuestion);
currentTypeScore += selectedQuestion.score;
// 防止无限循环
if (typeQuestions.length > 100) {
break;
}
}
questions.push(...typeQuestions);
}
// 如果总分不足,尝试补充题目
let totalScore = questions.reduce((sum, q) => sum + q.score, 0);
while (totalScore < subject.totalScore) {
// 获取所有类型的随机题目
const allTypes = Object.keys(subject.typeRatios).filter(type => subject.typeRatios[type as keyof typeof subject.typeRatios] > 0);
if (allTypes.length === 0) break;
const randomType = allTypes[Math.floor(Math.random() * allTypes.length)];
const availableQuestions = await QuestionModel.getRandomQuestions(
randomType as any,
10,
allCategories
);
if (availableQuestions.length === 0) break;
// 过滤掉已选题目
const availableUnselected = availableQuestions.filter(q =>
!questions.some(selected => selected.id === q.id)
);
if (availableUnselected.length === 0) break;
// 选择分数最接近剩余需求的题目
const remainingScore = subject.totalScore - totalScore;
const selectedQuestion = availableUnselected.reduce((prev, curr) => {
const prevDiff = Math.abs(remainingScore - prev.score);
const currDiff = Math.abs(remainingScore - curr.score);
return currDiff < prevDiff ? curr : prev;
});
questions.push(selectedQuestion);
totalScore += selectedQuestion.score;
// 防止无限循环
if (questions.length > 200) {
break;
}
}
// 如果总分超过,尝试移除一些题目
while (totalScore > subject.totalScore) {
// 找到最接近剩余差值的题目
const excessScore = totalScore - subject.totalScore;
let closestIndex = -1;
let closestDiff = Infinity;
for (let i = 0; i < questions.length; i++) {
const diff = Math.abs(questions[i].score - excessScore);
if (diff < closestDiff) {
closestDiff = diff;
closestIndex = i;
}
}
if (closestIndex === -1) break;
// 移除该题目
totalScore -= questions[closestIndex].score;
questions.splice(closestIndex, 1);
}
const result = await ExamSubjectModel.generateQuizQuestions(subject);
res.json({
success: true,
data: {
questions,
totalScore,
timeLimit: subject.timeLimitMinutes
questions: result.questions,
totalScore: result.totalScore,
timeLimit: result.timeLimitMinutes
}
});
} catch (error: any) {