AI-тьютор для VR-обучения
VR-тренажёры без адаптивного компонента — это дорогое видео с контролем. Обучаемый проходит один и тот же сценарий вне зависимости от ошибок: ошибся в шаге 3, но сценарий идёт дальше. AI-тьютор меняет логику: система отслеживает каждое действие, выявляет паттерн ошибок, адаптирует следующий сценарий и даёт персонализированную обратную связь в реальном времени.
Архитектура адаптивного тьютора
from langchain_openai import ChatOpenAI
from dataclasses import dataclass, field
from typing import Optional
import json
from datetime import datetime
@dataclass
class LearnerProfile:
learner_id: str
session_history: list[dict] = field(default_factory=list)
skill_scores: dict = field(default_factory=dict) # skill → 0.0–1.0
common_errors: list[str] = field(default_factory=list)
learning_pace: float = 1.0 # коэффициент скорости обучения
@dataclass
class VRAction:
action_type: str # step_completed, error, timeout, skip
step_id: str
timestamp: float
details: dict # конкретные параметры действия
correct: bool
class AdaptiveTutor:
FEEDBACK_PROMPT = """Ты — AI-тьютор в VR-тренажёре. Дай короткую (1–2 предложения) голосовую обратную связь.
Обучаемый только что: {action_description}
Правильно ли выполнено: {is_correct}
{"Ошибка: " + "{error_details}" if not "{is_correct}" else ""}
История ошибок обучаемого: {error_history}
Тон: поддерживающий, конкретный. Не хвали за очевидное.
Если ошибка повторяется — объясни принцип, не просто скажи "неправильно"."""
def __init__(self, llm: ChatOpenAI):
self.llm = llm
def analyze_action(self, action: VRAction, profile: LearnerProfile) -> dict:
"""Анализирует действие и определяет следующий шаг"""
# Обновляем скилл-скор
skill = self._get_skill_for_step(action.step_id)
if skill:
current = profile.skill_scores.get(skill, 0.5)
if action.correct:
profile.skill_scores[skill] = min(1.0, current + 0.05)
else:
profile.skill_scores[skill] = max(0.0, current - 0.1)
if action.action_type not in profile.common_errors:
profile.common_errors.append(action.step_id)
# Адаптация сценария
next_scenario = self._select_next_scenario(profile)
return {
"feedback": self._generate_feedback(action, profile),
"next_scenario": next_scenario,
"skill_update": profile.skill_scores.get(skill, 0.5)
}
def _select_next_scenario(self, profile: LearnerProfile) -> dict:
"""Выбирает следующий сценарий на основе профиля обучаемого"""
weak_skills = [
skill for skill, score in profile.skill_scores.items()
if score < 0.6
]
if weak_skills:
# Фокусируем на слабых зонах
return {
"type": "remedial",
"target_skills": weak_skills[:2],
"difficulty": "easier",
"reason": f"Требует отработки: {', '.join(weak_skills[:2])}"
}
elif all(s > 0.8 for s in profile.skill_scores.values()):
return {
"type": "advancement",
"difficulty": "harder",
"reason": "Хорошее освоение базовых навыков"
}
else:
return {"type": "standard", "difficulty": "normal"}
async def _generate_feedback(
self,
action: VRAction,
profile: LearnerProfile
) -> str:
repeated_error = action.step_id in profile.common_errors
prompt = f"""Дай обратную связь обучаемому в VR.
Действие: {action.action_type} на шаге {action.step_id}
Правильно: {action.correct}
Повторяющаяся ошибка: {repeated_error}
Детали ошибки: {json.dumps(action.details, ensure_ascii=False)}
{"Это ошибка повторяется. Объясни принцип, а не только что неправильно." if repeated_error else ""}
1 предложение. Говори на "ты". Конкретно."""
result = await self.llm.ainvoke(prompt)
return result.content
Оценка компетенций и прогресс-трекинг
class CompetencyTracker:
"""Отслеживает освоение компетенций через IRT (Item Response Theory)"""
def update_ability(
self,
learner_theta: float, # текущий уровень (-3 до +3)
item_difficulty: float, # сложность задания (-3 до +3)
correct: bool
) -> float:
"""Обновляет оценку способности по 3PL модели IRT"""
# 3-parameter logistic model
a = 1.7 # discrimination
c = 0.25 # guessing parameter
# Вероятность правильного ответа
p = c + (1 - c) / (1 + math.exp(-a * (learner_theta - item_difficulty)))
# Обновление по градиентному шагу
info = a**2 * (p - c)**2 * (1 - p) / ((1 - c)**2 * p)
delta = 0.3 * (1 if correct else -1) / (info + 0.01)
return max(-3, min(3, learner_theta + delta))
Unity: интеграция тьютора в сцену
public class VRTutorController : MonoBehaviour
{
private AdaptiveTutorClient tutorClient;
private AudioSource feedbackAudio;
public async void OnActionCompleted(string stepId, bool isCorrect, Dictionary<string, object> details)
{
var action = new VRAction(
actionType: isCorrect ? "step_completed" : "error",
stepId: stepId,
timestamp: Time.time,
details: details,
correct: isCorrect
);
var result = await tutorClient.AnalyzeAction(action, currentLearnerId);
// Воспроизводим голосовую обратную связь
string feedbackText = result.feedback;
byte[] audio = await TTSService.Synthesize(feedbackText);
feedbackAudio.PlayOneShot(AudioService.BytesToClip(audio));
// Адаптируем следующий сценарий
ScenarioManager.LoadScenario(result.nextScenario);
}
}
Кейс: обучение операторов нефтехимического завода работе с аварийным оборудованием. 200 операторов, 12 обязательных компетенций. До внедрения: фиксированный сценарий, все проходят одинаково, средний балл итогового теста — 71%. После 3 месяцев с AI-тьютором (персонализированные повторы слабых зон, адаптивная сложность): средний балл вырос до 84%, время до освоения сертификационного уровня сократилось с 8 до 5 занятий.
Сроки: базовый тьютор с feedback и прогрессом: 6–8 недель; адаптивный движок с IRT и аналитикой: 3–4 месяца.







