Вот рабочий рекурсивный метод, который вернет первый квалифицирующий набор данных.
Он работает путем удаления элементов из массива «требуемых субъектов» и массива «доступных слотов» при добавлении элементов в «заполненный» slots "массив.
Использование уточняющих значений от array_replace()
до pu sh в третьем параметре, доставляемом в рекурсивном вызове, предотвращает нарушение состояния переменной $filledSlots
, которая будет продолжать использоваться циклы foreach.
Функция: ( Demo )
function fillSlots($requiredSubjects, $availableSlots, $filledSlots = []) {
if (!$requiredSubjects) {
ksort($filledSlots);
return $filledSlots;
}
foreach ($requiredSubjects as $rIndex => $subject) {
foreach ($availableSlots[$subject] as $sIndex => $slot) {
if (!isset($filledSlots[$slot])) {
unset($requiredSubjects[$rIndex], $availableSlots[$subject][$sIndex]);
$result = fillSlots(
$requiredSubjects,
$availableSlots,
array_replace($filledSlots, [$slot => $subject])
);
if ($result) {
return $result;
}
}
}
}
}
Входные переменные:
$subjectSlots = [
"tamil" => [1, 2, 3, 4, 5],
"english" => [4, 5, 6],
"maths" => [1, 5, 8],
"social" => [1, 2, 5],
"pt" => [1],
"hindi" => [3, 4, 7]
];
$requiredSubjects = ['tamil', 'tamil', 'pt', 'maths', 'maths', 'social', 'hindi', 'english'];
Чтобы разрешить функция для выполнения в тысячи раз быстрее (я тестировал, потому что мне было любопытно), подготовьте входные данные так, чтобы сначала выполнялись итерации субъектов с меньшим количеством слотов.
Сортировка перед выполнением: ( Посмотреть эффект )
usort($requiredSubjects, function($a, $b) use ($subjectSlots) {
return $subjectSlots[$a] <=> $subjectSlots[$b]; // sort by length, then by values
});
Первоначальный вызов:
var_export(fillSlots($requiredSubjects, $subjectSlots));
Вывод:
array (
1 => 'pt',
2 => 'social',
3 => 'tamil',
4 => 'tamil',
5 => 'maths',
6 => 'english',
7 => 'hindi',
8 => 'maths',
)