es
Feedback
PHP | LeetCode

PHP | LeetCode

Ir al canal en Telegram
1 385
Suscriptores
Sin datos24 horas
+17 días
-730 días
Archivo de publicaciones
Задача: 26. Remove Duplicates from Sorted Array #easy Условие:
Учитывая целочисленный массив чисел, отсортированный в неубывающем порядке, удалите дубликаты на месте так, чтобы каждый уникальный элемент появлялся только один раз. Относительный порядок элементов должен оставаться неизменным. Затем верните количество уникальных элементов в числах. Считайте, что количество уникальных элементов чисел равно k. Чтобы вас приняли, вам нужно сделать следующее: Измените массив nums так, чтобы первые k элементов nums содержали уникальные элементы в том порядке, в котором они присутствовали в nums изначально. Остальные элементы nums не важны, как и размер nums. Вернуть К.
Решение:
class Solution {
  function removeDuplicates(&$nums) { $index = 1;
    for ($i = 1; $i < count($nums); $i++) {
     if($nums[$i] !== $nums[$i-1]){ 
         $nums[$index++] = $nums[$i];
     }  
    }  
      return $index;
  }
}
Пояснение: Данный код представляет функцию removeDuplicates, которая принимает массив чисел по ссылке и удаляет дубликаты из этого массива. Функция возвращает новую длину массива после удаления дубликатов и модифицирует сам массив, оставляя только уникальные элементы. Внутри функции задается индекс $index, который начинается с 1. Затем запускается цикл for, проходящий по всем элементам массива, начиная со второго элемента ($i = 1). Внутри цикла проверяется условие: если текущий элемент $nums[$i] не равен предыдущему элементу $nums[$i-1], то текущий элемент добавляется в массив на позицию $index с помощью оператора постфиксного инкремента $index++. Это позволяет удалять дубликаты, так как только уникальные элементы будут добавлены на новые позиции. После завершения цикла функция возвращает значение $index, которое представляет новую длину массива без дубликатов. Таким образом, в массиве $nums остаются только уникальные элементы, а его длина изменена на количество уникальных элементов.

Задача: 25. Reverse Nodes in k-Group #hard Условие:
Учитывая заголовок связанного списка, поменяйте местами узлы списка k за раз и верните измененный список. k — целое положительное число, меньшее или равное длине связанного списка. Если количество узлов не кратно k, то пропущенные узлы в конечном итоге должны остаться такими, какие они есть. Вы не можете изменять значения в узлах списка, можно изменять только сами узлы.
Решение:
/** 
 * Definition for a singly-linked list. 
 * class ListNode { 
 *     public $val = 0; 
 *     public $next = null; 
 *     function __construct($val = 0, $next = null) { 
 *         $this->val = $val; 
 *         $this->next = $next; 
 *     } 
 * } 
 */ 
class Solution { 
    public function reverseKGroup($head, $k) { 
        if ($head === null || $k === 1) { 
            return $head; 
        } 
 
        $dummy = new ListNode(0); 
        $dummy->next = $head; 
        $groupPrev = $dummy; 
 
        while (true) { 
            $kth = $this->getKthNode($groupPrev, $k); 
            if ($kth === null) { 
                break; 
            } 
 
            $groupNext = $kth->next; 
            // Reverse group 
            $prev = $groupNext; 
            $curr = $groupPrev->next; 
 
            while ($curr !== $groupNext) { 
                $tmp = $curr->next; 
                $curr->next = $prev; 
                $prev = $curr; 
                $curr = $tmp; 
            } 
 
            $tmp = $groupPrev->next; 
            $groupPrev->next = $kth; 
            $groupPrev = $tmp; 
        } 
 
        return $dummy->next; 
    } 
 
    private function getKthNode($curr, $k) { 
        while ($curr !== null && $k > 0) { 
            $curr = $curr->next; 
            $k--; 
        } 
        return $curr; 
    } 
} 
 
// Helper function to convert array to linked list 
function arrayToList($arr) { 
    $dummy = new ListNode(); 
    $current = $dummy; 
    foreach ($arr as $val) { 
        $current->next = new ListNode($val); 
        $current = $current->next; 
    } 
    return $dummy->next; 
} 
 
// Helper function to convert linked list to array 
function listToArray($head) { 
    $arr = []; 
    while ($head !== null) { 
        $arr[] = $head->val; 
        $head = $head->next; 
    } 
    return $arr; 
}
Пояснение: 1)Создается класс listnode для представления узла связанного списка, который содержит значение узла и ссылку на следующий узел. 2)Создается класс solution, который содержит метод reversekgroup для переворачивания групп узлов размером k. 3)В методе reversekgroup проверяется, если головной узел равен null или k равно 1, то возвращается головной узел. 4)Создается фиктивный узел dummy, который является заголовочным узлом и его next указывает на головной узел. 5)Затем начинается цикл, в котором извлекается k-ый узел группы с помощью метода getkthnode. 6)Если k-ый узел не существует, происходит выход из цикла. 7)Выделяется следующий после группы узлов узел и происходит переворот этой группы узлов. 8)После переворота группы узлов обновляются ссылки узлов и продолжается цикл для следующей группы. 9)В конце возвращается обновленный головной узел из dummy->next. 10)Есть также вспомогательные функции arraytolist для преобразования массива в связанный список и listtoarray для преобразования связанного списка в массив.

Задача: 24. Swap Nodes in Pairs #medium Условие:
Учитывая связанный список, поменяйте местами каждые два соседних узла и верните его голову. Вы должны решить проблему, не изменяя значения в узлах списка (т. е. изменять можно только сами узлы).
Решение:
/** 
 * Definition for a singly-linked list. 
 * class ListNode { 
 *     public $val = 0; 
 *     public $next = null; 
 *     function __construct($val = 0, $next = null) { 
 *         $this->val = $val; 
 *         $this->next = $next; 
 *     } 
 * } 
 */ 
class Solution { 
    /** 
     * @param ListNode $head 
     * @return ListNode 
     */ 
    function swapPairs($head) { 
        if($head == null) { 
            return []; 
        } 
 
        $iteration = 1; 
        $current = $head; 
        while($current->next != null) { 
            if($iteration % 2 != 0) { 
                $currentNodeValue = $current->val; 
                $nextNodeValue = $current->next->val; 
                $current->next->val = $currentNodeValue; 
                $current->val = $nextNodeValue; 
            } 
            $current = $current->next; 
            $iteration++; 
        } 
        return $head; 
    } 
}
Пояснение: 1. Устанавливает переменную iteration равной 1. Эта переменная используется для отслеживания текущей итерации. 2. Перебирает узлы в списке, используя переменную current. 3. На каждой итерации, если iteration нечетно (т. е. 1, 3, 5...), то выполняется обмен значений между текущим узлом и следующим узлом: - currentNodeValue получает значение текущего узла. - nextNodeValue получает значение следующего узла. - Значение следующего узла обновляется до currentNodeValue. - Значение текущего узла обновляется до nextNodeValue. 4. Перемещает указатель current на следующий узел. 5. Увеличивает iteration на 1. 6. После перебора всех узлов в списке функция возвращает измененный список head.

Задача: 22. Generate Parentheses #medium Условие:
Учитывая n пар круглых скобок, напишите функцию, которая генерирует все комбинации правильных круглых скобок.
Решение:
class Solution { 
 
    private array $answer = []; 
 
    private string $current = ''; 
 
    /** 
     * @param Integer $n 
     * @return String[] 
     */ 
    function generateParenthesis($n) { 
        $this->generateParentheses($n, 0); 
        return $this->answer; 
    } 
 
    function generateParentheses($toOpen, $toClose) { 
        if ($toOpen == 0 && $toClose == 0) { 
            $this->answer[] = $this->current; 
            return; 
        } 
 
        if ($toOpen > 0) { 
            $this->current .= '('; 
            $this->generateParentheses($toOpen - 1, $toClose + 1); 
            $this->current = substr($this->current, 0, -1); 
        } 
 
        if ($toClose > 0) { 
            $this->current .= ')'; 
            $this->generateParentheses($toOpen, $toClose - 1); 
            $this->current = substr($this->current, 0, -1); 
        }         
    } 
}
Пояснение: 1. Базовый случай: Если количество открывающихся и закрывающихся скобок равно нулю, текущая скобочная последовательность действительна, и она добавляется в массив $answer.2. Если количество открывающихся скобок больше нуля: * Добавляется открывающаяся скобка к $current. * Рекурсивно вызывается метод generateParenthesis с уменьшенным значением $toOpen и увеличенным значением $toClose. * Открывающаяся скобка удаляется из $current.3. Если количество закрывающихся скобок больше нуля: * Добавляется закрывающаяся скобка к $current. * Рекурсивно вызывается метод generateParenthesis с тем же значением $toOpen и уменьшенным значением $toClose. * Закрывающаяся скобка удаляется из $current.

Задача: №21. Merge Two Sorted Lists #easy Условие:
Вам даны заголовки двух отсортированных связанных списков list1 и list2. Объедините два списка в один отсортированный список. Список должен быть составлен путем сращивания узлов первых двух списков. Возвращает заголовок объединенного связанного списка.
Решение:
/ 
 * Definition for a singly-linked list. 
 * class ListNode { 
 *     public $val = 0; 
 *     public $next = null; 
 *     function __construct($val = 0, $next = null) { 
 *         $this->val = $val; 
 *         $this->next = $next; 
 *     } 
 * } 
 */ 
class Solution { 
 
    / 
     * @param ListNode $list1 
     * @param ListNode $list2 
     * @return ListNode 
     */ 
    function mergeTwoLists($list1, $list2) { 
        $newList = new ListNode(); 
        $current = $newList; 
        while($list1 || $list2) { 
            if ($list1 && !$list2) { 
                $current->next = new ListNode($list1->val); 
                $current = $current->next; 
                $list1 = $list1->next; 
            } elseif (!$list1 && $list2) { 
                $current->next = new ListNode($list2->val); 
                $current = $current->next; 
                $list2 = $list2->next; 
            } else { 
                if ($list1->val === $list2->val) { 
                    $current->next = new ListNode($list1->val); 
                    $current = $current->next; 
                    $current->next = new ListNode($list2->val); 
                    $current = $current->next; 
                    $list1 = $list1->next; 
                    $list2 = $list2->next; 
                } elseif ($list1->val < $list2->val) { 
                    $current->next = new ListNode($list1->val); 
                    $current = $current->next; 
                    $list1 = $list1->next; 
                } else { 
                    $current->next = new ListNode($list2->val); 
                    $current = $current->next; 
                    $list2 = $list2->next; 
                } 
            } 
        } 
 
        return $newList->next; 
    } 
}
Пояснение: Шаг 1: Инициализация Мы начинаем с создания нового узла списка $newList, который будет служить якорем для нашего нового отсортированного списка. $current указывает на текущий узел нового списка. Шаг 2: Цикл слияния Этот цикл продолжается до тех пор, пока хотя бы один из исходных списков не станет пустым. Шаг 3: Добавление элементов в новый список - Случай 1: Только $list1 не пустой Если только $list1 не пустой, добавляем его текущий элемент в новый список и переходим к следующему элементу $list1. - Случай 2: Только $list2 не пустой Если только $list2 не пустой, добавляем его текущий элемент в новый список и переходим к следующему элементу $list2. - Случай 3: Оба списка не пустые - Если значения текущих узлов обоих списков равны, мы добавляем оба элемента в новый список. - Если значение текущего узла $list1 меньше значения текущего узла $list2, добавляем элемент из $list1. - В других случаях, добавляем элемент из $list2. Шаг 4: Возвращаем результат Возвращаем начальную часть нашего нового списка $newList (пропуская первый псевдоузел).

Задача: №20. Valid Parentheses #easy Условие:
Учитывая строку s, содержащую только символы '(', ')', '{', '}', '[' и ']', определите, является ли входная строка допустимой. Входная строка действительна, если: Открытые скобки должны закрываться скобками того же типа. Открытые скобки должны закрываться в правильном порядке.Каждой закрывающей скобке соответствует открытая скобка того же типа.
Решение:
class Solution 
{ 
    /** 
     * @param String $s 
     * @return Boolean 
     */ 
    function isValid(string $s): bool { 
        $sLength = strlen($s); 
        if ($sLength % 2 !== 0) return false; 
        $bracketSet = ['(' => ')', '[' => ']', '{' => '}']; 
 
        $bracketStack = []; 
 
        for ($i = 0; $i < $sLength; $i++) { 
            if (array_key_exists($s[$i], $bracketSet)) { 
                $bracketStack[] = $bracketSet[$s[$i]]; 
            } elseif (array_pop($bracketStack) !== $s[$i]) { 
                return false; 
            } 
        } 
        return count($bracketStack) === 0; 
    } 
}
Пояснение: 1. Длина строки: - Сначала мы проверяем длину входной строки $s. Если длина нечетная, то строка не может быть валидной, так как каждой открывающей скобке должна соответствовать закрывающая скобка. Поэтому возвращаем false. 2. Создание словаря соответствия скобок: - Создаем ассоциативный массив $bracketSet, в котором хранятся пары открывающих и закрывающих скобок. 3. Обход строки: - Создаем стек $bracketStack, в котором будем хранить ожидаемые закрывающие скобки. - Проходим по каждому символу в строке $s: - Если символ является открывающей скобкой, то добавляем соответствующую ей закрывающую скобку в стек. - Если символ является закрывающей скобкой, то сравниваем его с последним элементом в стеке. Если они не соответствуют, то возвращаем false, так как мы встретили неожиданную закрывающую скобку. - Если они соответствуют, то извлекаем последний элемент из стека. 4. Проверка пустоты стека: - После обхода всей строки, если стек пуст, то это означает, что все открывающие скобки нашли свои закрывающие пары, и возвращаем true. - Если в стеке остались элементы, то это означает, что не все открывающие скобки нашли свои закрывающие пары, и возвращаем false. Временная сложность этого решения - O(n), где n - длина входной строки. Это связано с тем, что мы проходим по всей строке один раз, выполняя простые операции с элементами стека. Пространственная сложность - O(n), так как в худшем случае, когда все скобки в строке являются открывающими, мы добавим все их в стек.

Задача: №19. Remove Nth Node From End of List #medium Условие:
Учитывая заголовок связанного списка, удалите n-й узел из конца списка и верните его заголовок.
Решение:
class Solution { 
  
    /** 
     * @param ListNode $head 
     * @param Integer $n 
     * @return ListNode 
     */ 
    function removeNthFromEnd($head, $n) { 
        // Paso #1 
        $tamaño = 0; 
        $aux = $head; 
  
        while ($aux !== null) { 
            $aux = $aux->next; 
            $tamaño++; 
        } 
  
        // Paso #2 
        if ($tamaño == 1) { 
            return null; 
        } 
        if ($tamaño == $n) { 
            return $head->next; 
        } 
  
        // Paso #3 
        $aux = $head; 
        for ($i = 0; $i < $tamaño - $n - 1; $i++) { 
            $aux = $aux->next; 
        } 
  
        // Paso #4 
        $aux->next = $aux->next->next; 
  
        // Paso #5 
        return $head; 
  
    } 
}
Пояснение: 1. Определение размера связного списка: - Создаем вспомогательную переменную $aux и присваиваем ей значение $head (начало списка). - Пока $aux не равен null, увеличиваем счетчик $tamaño и перемещаем $aux на следующий узел. - Таким образом, мы определяем количество узлов в списке и сохраняем его в $tamaño. 2. Обработка граничных случаев: - Если размер списка равен 1, то мы возвращаем null, так как удаление единственного узла оставит список пустым. - Если $n равно размеру списка, то мы возвращаем $head->next, так как мы удаляем первый узел. 3. Перемещение по списку до нужного узла: - Создаем вспомогательную переменную $aux и снова присваиваем ей значение $head. - Используем цикл, чтобы переместить $aux на $tamaño - $n - 1 узлов. Это позволит нам найти узел, предшествующий узлу, который нужно удалить. 4. Удаление узла: - После того, как мы нашли нужный узел, мы просто обновляем ссылку $aux->next, чтобы пропустить узел, который нужно удалить. 5. Возвращение обновленного списка: - Наконец, мы возвращаем $head - корень обновленного связного списка. Временная сложность этого решения - O(n), где n - количество узлов в списке. Это связано с тем, что мы проходим по списку дважды: один раз, чтобы определить его размер, и второй раз, чтобы найти узел, предшествующий узлу, который нужно удалить. Пространственная сложность - O(1), поскольку мы используем только несколько вспомогательных переменных и не создаем дополнительные структуры данных.

Задача: №18. 4Sum #medium Условие:
Учитывая массив nums из n целых чисел, верните массив всех уникальных четверок [nums[a], nums[b], nums[c], nums[d]] таких, что: 0 <= a, b, c, d < na, b, c и d различны. nums[a] + nums[b] + nums[c] + nums[d] == цельВы можете вернуть ответ в любом порядке.
Решение:
class Solution { 
 
    /** 
     * @param Integer[] $nums 
     * @param Integer $target 
     * @return Integer[][] 
     */ 
    function fourSum($nums, $target) { 
 
        sort($nums); 
        return $this->kSum($nums, $target, 4); 
 
    } 
 
    function kSum($nums, $target, $k) 
    { 
        $res = []; 
 
        if (!count($nums)) { 
            return $res; 
        } 
 
        $averageValue = $target / $k; 
 
        if ($averageValue < $nums[0] || $nums[count($nums)-1] < $averageValue) { 
            return $res; 
        } 
 
        if ($k == 2) { 
            return $this->twoSum($nums, $target); 
        } 
 
        for ($i = 0; $i < count($nums); $i++) { 
            if ($i == 0 || $nums[$i - 1] != $nums[$i]) { 
                $kSum = $this->kSum(array_slice($nums, $i+1), $target - $nums[$i], $k - 1); 
                foreach ($kSum as $item) { 
                    $res[] = array_merge([$nums[$i]], $item); 
                } 
            } 
        } 
 
        return $res; 
 
    } 
 
    function twoSum($nums, $target) 
    { 
        $res = []; 
        $lo = 0; 
        $hi = count($nums) - 1; 
 
        while ($lo < $hi) { 
            $currSum = $nums[$lo] + $nums[$hi]; 
            if ($currSum < $target || ( $lo > 0 && $nums[$lo] == $nums[$lo - 1])) { 
                $lo++; 
            } elseif ($currSum > $target || ($hi < count($nums) - 1 && $nums[$hi] == $nums[$hi + 1])) { 
                $hi--; 
            } else { 
                $res[] = [$nums[$lo], $nums[$hi]]; 
                $lo++; 
                $hi--; 
            } 
        } 
        return $res; 
    } 
}
Пояснение: 1. Сортировка входного массива: Сначала мы сортируем входной массив $nums в порядке возрастания. Это необходимо для эффективного поиска решений. 2. Реализация функции kSum: Эта функция рекурсивно находит все уникальные наборы из $k чисел, сумма которых равна $target. Она использует следующую логику: - Если $k == 2, вызываем функцию twoSum для поиска всех пар чисел, сумма которых равна $target. - Если $k > 2, то перебираем все числа в $nums, пропуская дубликаты, и рекурсивно вызываем kSum для оставшейся части массива с $k-1 и $target - $nums[$i]. 3. Реализация функции twoSum: Эта функция используется внутри kSum для поиска всех уникальных пар чисел, сумма которых равна $target. Она использует два указателя: $lo и $hi, указывающие на начало и конец отсортированного массива. Эти указатели двигаются навстречу друг другу, пока не найдут все решения. 4. Возвращение результата: Основная функция fourSum сначала сортирует входной массив, затем вызывает kSum с $k=4 и $target, и возвращает результат. Временная сложность этого решения - O(n^(k-1)), где n - длина входного массива $nums, и k - количество чисел, которые нужно найти. Это связано с тем, что для каждого числа в массиве мы рекурсивно вызываем kSum с $k-1. Пространственная сложность - O(n^(k-1)), поскольку мы сохраняем все уникальные наборы чисел в результирующем массиве.

Задача: №17. Letter Combinations of a Phone Number #medium Условие:
Учитывая строку, содержащую цифры от 2 до 9 включительно, верните все возможные комбинации букв, которые может представлять число. Верните ответ в любом порядке. Соответствие цифр буквам (как на кнопках телефона) приведено ниже. Обратите внимание, что 1 не соответствует никаким буквам.
Решение:
class Solution { 
 
    public $result = []; 
 
    /** 
     * @param String $digits 
     * @return String[] 
     */ 
    function letterCombinations($digits) { 
        if(strlen($digits) == 0) { 
            return []; 
        } 
 
        $layouts = [ 
            2 => range('a', 'c'), 
            3 => range('d', 'f'), 
            4 => range('g', 'i'), 
            5 => range('j', 'l'),  
            6 => range('m', 'o'),  
            7 => range('p', 's'), 
            8 => range('t', 'v'), 
            9 => range('w', 'z'), 
        ]; 
 
        return $this->recursive(0, $digits, $layouts); 
    } 
 
    function recursive ($index, $chars, $layouts, $combine = '') { 
        foreach($layouts[$chars[$index]] as $currentLayout) { 
            if($layouts[$chars[$index + 1]]) { 
                $this->recursive($index + 1, $chars, $layouts, $combine . $currentLayout); 
            } else { 
                $this->result[] = $combine . $currentLayout; 
            } 
        } 
 
        return $this->result; 
    } 
}
Пояснение: 1. Проверяем, является ли входная строка пустой. Если да, возвращаем пустой массив. 2. Создаем ассоциативный массив $layouts, в котором каждой цифре соответствует массив букв, которые можно набрать на телефонной клавиатуре. 3. Вызываем рекурсивную функцию recursive(), передавая ей следующие аргументы: - $index - индекс текущей цифры в входной строке; - $chars - входная строка, состоящая из цифр; - $layouts - ассоциативный массив с соответствием цифр и букв; - $combine - строка, представляющая текущую комбинацию букв. 4. В рекурсивной функции recursive(): - Перебираем все буквы, соответствующие текущей цифре в $layouts. - Если есть следующая цифра в $chars, то рекурсивно вызываем recursive(), передавая новый индекс, текущую комбинацию букв и обновленные $chars и $layouts. - Если следующей цифры нет, то добавляем текущую комбинацию букв в результирующий массив $result. - Возвращаем $result. 5. Возвращаем конечный результат из основной функции letterCombinations(). Временная сложность этого решения - O(3^N * 4^M), где N - количество цифр, которые имеют 3 соответствующие буквы, а M - количество цифр, которые имеют 4 соответствующие буквы. Это связано с тем, что для каждой цифры мы перебираем все возможные комбинации букв. Пространственная сложность - O(3^N * 4^M), поскольку мы храним все возможные комбинации букв в результирующем массиве.

Задача: №16. 3Sum Closest #medium Условие:
Учитывая целочисленный массив nums длины n и целочисленную цель, найдите три целых числа в nums, сумма которых наиболее близка к цели. Возвращает сумму трех целых чисел. Вы можете предположить, что каждый вход будет иметь ровно одно решение.
Решение:
class Solution {  
  
    function threeSumClosest($nums, $target) {  
        $fIsCloserS = function ($base, $f, $s) {  
            return abs($base - $f) < abs($base - $s);  
        };  
  
        sort($nums);  
        $res = $middle = INF;  
        $key_left = 0;  
        $key_right = array_key_last($nums);  
  
        while ($key_left + 1 < $key_right) {  
            $needle = $target - $nums[$key_right] - $nums[$key_left];  
            $bin_key_left = $key_left + 1;  
            $bin_key_right = $key_right - 1;  
  
            while ($bin_key_left <= $bin_key_right) {  
                $bin_key_middle = (int)round(($bin_key_left + $bin_key_right) / 2);  
                if ($fIsCloserS($needle, $nums[$bin_key_middle], $middle)) {  
                    $middle = $nums[$bin_key_middle];  
                }  
                if (($needle - $nums[$bin_key_middle]) < 0) {  
                    $bin_key_right = $bin_key_middle - 1;  
                }  
                else $bin_key_left = $bin_key_middle + 1;  
            }  
  
            $sum = $nums[$key_right] + $nums[$key_left] + $middle;  
            if ($fIsCloserS($target, $sum, $res)) {  
                if ($sum === $target) {  
                    return $sum;  
                }  
                $res = $sum;  
            }  
  
            $new_left = $nums[$key_left + 1];  
            $new_right = $nums[$key_right - 1];  
            if ($fIsCloserS($needle, $new_left, $new_right)  
                || ($new_left === $new_right && $new_left === $nums[$key_left])  
            ) $key_right--;  
            else $key_left++;  
        }  
        return $res;  
    }  
} 
Пояснение: 1. Сначала мы сортируем массив $nums в порядке возрастания.2. Затем мы используем две указателя - $key_left и $key_right, которые указывают на начало и конец массива соответственно. 3. Для каждой пары чисел $nums[$key_left] и $nums[$key_right] мы ищем третье число, которое делает сумму этих трех чисел максимально близкой к $target. Для этого мы используем двоичный поиск в оставшейся части массива.4. Во время двоичного поиска мы отслеживаем наиболее близкое к $target значение суммы трех чисел и обновляем $res соответственно. 5. После того, как мы нашли наиболее близкую сумму, мы двигаем указатели $key_left и $key_right к центру, чтобы найти следующую пару чисел. Ключевые моменты: 1. Использование функции $fIsCloserS для определения, какая сумма ближе к $target.2. Применение двоичного поиска для поиска третьего числа, которое делает сумму наиболее близкой к $target. 3. Обновление $res (наиболее близкой суммы) после каждой итерации.4. Перемещение указателей $key_left и $key_right к центру, чтобы найти следующую пару чисел. Общая временная сложность этого решения - O(n^2 * log n), где n - длина массива $nums. Это связано с сортировкой массива (O(n * log n)) и двумя вложенными циклами, каждый из которых выполняет двоичный поиск (O(log n)).