AI-система инспекции дорожного покрытия
Ямы, трещины, выбоины — ежегодные расходы на их устранение исчисляются миллиардами. Проблема в том, что к моменту визуального обнаружения дорожным рабочим, дефект уже перешёл в критическую стадию. AI-инспекция с камер транспортных средств или специальных машин позволяет обнаруживать ранние признаки деградации покрытия и приоритизировать ремонт.
Классификация дефектов дорожного покрытия
Стандарт ASTM D6433 выделяет 20 типов дистресса. На практике работаем с 7–8 ключевыми:
| Тип дефекта | Метод обнаружения | Сложность |
|---|---|---|
| Ямы (potholes) | Object detection (bbox) | Средняя |
| Продольные трещины | Segmentation | Высокая |
| Поперечные трещины | Segmentation | Средняя |
| Сетчатые трещины (alligator) | Texture classification | Высокая |
| Колейность (rutting) | 3D профиль / stereo | Очень высокая |
| Выбоины (raveling) | Texture + anomaly | Средняя |
| Просадка (depression) | 3D профиль | Высокая |
Модель детекции и сегментации
import torch
import numpy as np
import segmentation_models_pytorch as smp
from ultralytics import YOLO
import cv2
class PavementInspector:
def __init__(self, seg_model_path: str, det_model_path: str):
# Сегментация трещин: UNet++ с ResNet50 энкодером
# Дообучен на RDD2022 (Road Damage Dataset, 47k изображений)
self.seg_model = smp.UnetPlusPlus(
encoder_name='resnet50',
encoder_weights=None,
in_channels=3,
classes=4, # background, longitudinal, transverse, alligator
)
seg_ckpt = torch.load(seg_model_path)
self.seg_model.load_state_dict(seg_ckpt)
self.seg_model.eval()
# YOLOv8m для ям и выбоин (bbox достаточно)
self.det_model = YOLO(det_model_path)
# Маппинг классов сегментации
self.seg_classes = {
0: 'background',
1: 'longitudinal_crack',
2: 'transverse_crack',
3: 'alligator_crack'
}
# Маппинг для оценки тяжести (PCI-based)
self.severity_thresholds = {
'pothole': {'low': 0.01, 'medium': 0.05}, # % площади кадра
'crack': {'low': 0.02, 'medium': 0.08}
}
@torch.no_grad()
def inspect(self, frame: np.ndarray) -> dict:
h, w = frame.shape[:2]
# 1. Сегментация трещин
input_tensor = self._preprocess(frame)
seg_output = self.seg_model(input_tensor)
seg_mask = seg_output.argmax(dim=1)[0].numpy()
crack_analysis = self._analyze_cracks(seg_mask, w * h)
# 2. Детекция ям
det_results = self.det_model(frame, conf=0.45)
potholes = self._analyze_potholes(det_results, w * h)
# 3. Индекс состояния покрытия (упрощённый PCI)
pci = self._compute_pci(crack_analysis, potholes)
return {
'crack_analysis': crack_analysis,
'potholes': potholes,
'pci_score': pci,
'condition': self._pci_to_condition(pci),
'seg_mask': seg_mask
}
def _analyze_cracks(self, mask: np.ndarray,
total_pixels: int) -> dict:
analysis = {}
for cls_id, cls_name in self.seg_classes.items():
if cls_id == 0:
continue
crack_pixels = int((mask == cls_id).sum())
ratio = crack_pixels / total_pixels
analysis[cls_name] = {
'pixel_count': crack_pixels,
'area_ratio': ratio,
'severity': 'high' if ratio > 0.08 else
'medium' if ratio > 0.02 else 'low'
}
return analysis
def _compute_pci(self, cracks: dict, potholes: list) -> float:
"""
PCI 0–100: 100 = идеальное покрытие, 0 = полная деградация.
Упрощённая формула на основе ASTM D6433.
"""
deduct = 0.0
for crack_type, data in cracks.items():
ratio = data['area_ratio']
if ratio > 0.08:
deduct += 25
elif ratio > 0.02:
deduct += 12
elif ratio > 0.005:
deduct += 5
for pothole in potholes:
area = pothole['area_ratio']
if area > 0.03:
deduct += 30
elif area > 0.01:
deduct += 15
return max(0, 100 - deduct)
def _pci_to_condition(self, pci: float) -> str:
if pci >= 85: return 'excellent'
elif pci >= 70: return 'good'
elif pci >= 55: return 'fair'
elif pci >= 40: return 'poor'
elif pci >= 25: return 'very_poor'
else: return 'failed'
Мобильная инспекция: камера на транспортном средстве
Для дорог общего пользования — камера под передним бампером или в решётке радиатора, запись со скоростью 25fps, привязка к GPS. Дополнительно — акселерометр для автоматической фиксации ям по вибрации.
class MobileRoadSurvey:
def __init__(self, gps_logger, inspector: PavementInspector):
self.gps = gps_logger
self.inspector = inspector
self.survey_log = []
def process_frame_with_geotagging(self, frame: np.ndarray,
timestamp: float) -> dict:
gps_coords = self.gps.get_coords(timestamp)
results = self.inspector.inspect(frame)
record = {
'timestamp': timestamp,
'lat': gps_coords['lat'],
'lon': gps_coords['lon'],
'pci': results['pci_score'],
'condition': results['condition'],
'defects': results
}
self.survey_log.append(record)
return record
Кейс: инспекция 120 км дорог города
Задача: приоритизация дорожного ремонта. Инструмент: Ford Transit с 4 камерами (перед + 2 борта + зад), GPS RTK. За 3 дня съёмки покрыто 120 км.
- Обработано: 1.2 млн кадров
- Обнаружено: 3400 ям (P > 0.5), 47 км трещин (сегментация)
- Из них критических (PCI < 25): 8.2 км — первоочерёдный ремонт
- Экономия по сравнению с ручным обследованием: 12 рабочих дней → 6 часов обработки + 3 часа верификации
| Тип проекта | Срок |
|---|---|
| Детектор ям (базовый) | 3–5 недель |
| Полная инспекционная система (+ трещины, PCI) | 7–12 недель |
| Мобильная система с GIS-интеграцией | 10–16 недель |







