Библиотека собеса по PHP | вопросы с собеседований
3.38K subscribers
140 photos
2 videos
68 links
Вопросы с собеседований по PHP и ответы на них.

По рекламе: @proglib_adv

Учиться у нас: https://proglib.io/w/9f3affba

Работать у нас: https://job.proglib.io/

Наши каналы: https://yangx.top/proglibrary/9197
加入频道
Что такое Attributes в PHP 8 и когда их использовать?

PHP 8 представил атрибуты — мощный инструмент для добавления структурированных метаданных к вашему коду. Теперь вы можете аннотировать классы, методы и свойства без использования phpDoc-комментариев. Это делает код более чистым и читаемым.​

В нашем примере атрибут Route определяет маршрут для метода listUsers, делая код более декларативным и понятным.​

Когда использовать атрибуты?

Фреймворки и библиотеки: Многие современные фреймворки, такие как Symfony или Laravel, используют атрибуты для конфигурации маршрутов, привязки зависимостей и других аспектов, что упрощает настройку и делает код более читаемым.​

ORM (Object-Relational Mapping): Атрибуты могут использоваться для определения маппинга между классами и таблицами базы данных, заменяя устаревшие комментарии-аннотации. Это улучшает производительность и упрощает поддержку кода. ​

Валидация и сериализация: Атрибуты позволяют задавать правила валидации или определять, как свойства класса должны сериализоваться, что делает код более декларативным и понятным.

Преимущества атрибутов:

🔸 Структурированность: Атрибуты обеспечивают строгий синтаксис для метаданных, что упрощает их обработку и снижает вероятность ошибок.​

🔸Производительность: В отличие от phpDoc, атрибуты обрабатываются на уровне языка, что может положительно сказываться на производительности приложения.​

🔸 Интеграция: Атрибуты легко интегрируются с рефлексией, позволяя динамически получать информацию о метаданных и адаптировать поведение приложения.
Как использовать Eager loading в Laravel?

Eager loading в Laravel — это способ предварительной загрузки связанных моделей, чтобы уменьшить количество запросов к базе данных и повысить производительность приложения. При использовании ленивой загрузки (lazy loading) связанные данные загружаются только тогда, когда к ним обращаются, что может привести к множественным запросам к базе данных (проблема N+1 запросов). В то время как eager loading загружает все необходимые данные заранее, используя объединение (join) или отдельные запросы с использованием ключевого слова with.

Eager loading помогает значительно снизить нагрузку на базу данных и ускорить работу приложения за счет уменьшения количества выполняемых запросов.
Что такое traits?

Traits в PHP — это механизм, который позволяет повторно использовать код в классах без необходимости использования множественного наследования. Traits предоставляют решение для проблемы, когда несколько классов должны иметь одинаковый функционал, но наследование не является подходящим вариантом.

Также traits позволяют разделить функционал, что упрощает структуру кода и делает его более логичным и понятным.

Альтернативным решением для использования функционала traits может быть использование интерфейсов. Однако интерфейсы требуют реализации всех методов, которые определены в интерфейсе, в классе, в то время как трейты могут предоставлять реализацию функционала по умолчанию.

Оба варианта имеют свои преимущества, но трейты в PHP предоставляют большую гибкость при повторном использовании кода в разных классах.

Так как каждый класс может реализовать множество трейтов, могут возникать конфликты когда один и тот же метод реализован в разных трейтах и/или самом классе. Подробней о трейтах в документации.
Что делает функция create_function(), и почему её не стоит использовать?

​Функция create_function() в PHP позволяет динамически создавать анонимные функции из переданных строковых параметров, возвращая уникальное имя созданной функции. Однако начиная с версии PHP 7.2.0, эта функция была объявлена устаревшей, а в PHP 8.0.0 полностью удалена. ​


Почему не рекомендуется использовать create_function():

🔸 Проблемы безопасности: create_function() использует внутри себя функцию eval(), что может привести к выполнению произвольного кода, если входные данные не проверяются должным образом. Это открывает потенциальные уязвимости для внедрения вредоносного кода. ​

🔸 Низкая производительность и утечки памяти: Функции, созданные с помощью create_function(), являются глобальными и не могут быть освобождены, что может привести к повышенному потреблению памяти и снижению производительности. ​

🔸 Устаревание и удаление: Из-за вышеупомянутых проблем, начиная с PHP 7.2.0, функция create_function() была объявлена устаревшей, а в PHP 8.0.0 полностью удалена. Это означает, что её использование в современном коде приведёт к ошибкам. ​

Рекомендуемая альтернатива:

Вместо create_function() следует использовать анонимные функции или замыкания, которые были введены в PHP 5.3 и обеспечивают более безопасный и эффективный способ создания функций на лету. ​
Каковы основные компоненты Symfony и как они взаимодействуют друг с другом?

Symfony состоит из нескольких ключевых компонентов, которые взаимодействуют между собой, создавая надежный PHP-фреймворк. Компонент HttpFoundation заменяет глобальные переменные PHP объектно-ориентированным слоем, что позволяет проще работать с HTTP-запросами и ответами. Routing управляет генерацией и подбором URL, а EventDispatcher позволяет разделить код с помощью слушателей событий. DependencyInjection обеспечивает функциональность контейнера сервисов, способствуя многократному использованию и тестированию кода.

HttpKernel выступает в качестве ядра, используя другие компоненты для обработки запросов и генерации ответов. Он использует EventDispatcher для отправки нескольких событий во время обработки запроса, что позволяет внедрять пользовательское поведение в различных точках. Например, он отправляет событие 'kernel.request', которое может быть использовано Routing для сопоставления запроса с маршрутом и определения контроллера.

Компонент Twig — это шаблонизатор Symfony, взаимодействующий с HttpKernel для рендеринга представлений. Doctrine ORM взаимодействует с базой данных, обеспечивая персистентность данных.

Security обеспечивает аутентификацию и авторизацию, тесно взаимодействуя с HttpKernel и EventDispatcher. Translation помогает в интернационализации, а Validator обеспечивает соблюдение правил проверки объектов.
Что такое фильтр Блума?

Фильтр Блума — это вероятностная структура данных, которая используется для тестирования принадлежности элемента множеству. Он может давать ложные положительные результаты, но никогда не дает ложных отрицательных. Это значит, что если фильтр Блума говорит, что элемент принадлежит множеству, это может быть не так (ложное срабатывание), но если фильтр говорит, что элемент не принадлежит множеству, это точно так.

Подробнее в статье
Как использовать «PHPUnit» для написания модульных тестов для PHP-приложений, и каковы лучшие практики для этого?

PHPUnit — это широко используемый фреймворк для написания модульных тестов в PHP. Он позволяет разработчикам писать тестовые примеры, чтобы убедиться, что отдельные части кода работают так, как ожидается.

Для запуска тестов можно использовать команду phpunit. Лучшие практики модульного тестирования включают в себя написание независимых, повторяемых и легко поддерживаемых тестов, использование фикстур для создания тестовых данных, а также тестирование как положительных, так и отрицательных случаев.
В чем разница между include, require, include_once и require_once на уровне производительности?

​В PHP конструкции include, require, include_once и require_once используются для включения содержимого одного файла в другой. Основные различия между ними связаны с обработкой ошибок и повторным включением файлов. С точки зрения производительности различия минимальны, но некоторые нюансы стоит учитывать.​

Обзор конструкций:

include: Включает указанный файл. Если файл не найден, генерируется предупреждение (E_WARNING), и выполнение скрипта продолжается.​

require: Аналогично include, но при отсутствии файла вызывает фатальную ошибку (E_COMPILE_ERROR), останавливая выполнение скрипта.​

include_once: Действует как include, но проверяет, был ли файл уже включен ранее; если да, повторное включение не происходит.​

require_once: Работает аналогично require, с дополнительной проверкой на предыдущее включение файла, предотвращая повторное его подключение.​

Различия в производительности:

При однократном включении файлов различия в производительности между этими конструкциями незначительны и практически незаметны. Однако при многократном включении одного и того же файла в рамках одного запроса могут возникнуть некоторые отличия:​

include и require: Каждый раз при вызове происходит повторное включение и выполнение кода из указанного файла, что может привести к избыточным операциям и увеличению времени выполнения, особенно если файл содержит ресурсоемкий код.​

include_once и require_once: Перед включением файла проверяют, был ли он уже подключен ранее. Эта проверка требует дополнительных ресурсов, но предотвращает повторное выполнение кода из одного и того же файла, что может быть полезно для избежания ошибок, связанных с повторным объявлением функций, классов или переменных.​

Практические рекомендации:

🔸 Используйте require_once для подключения файлов, содержащих критически важный код (например, конфигурационные файлы или определения классов), чтобы гарантировать их однократное включение и избежать фатальных ошибок при их отсутствии.​

🔸 Для менее критичных файлов можно использовать include или include_once, в зависимости от необходимости проверки на повторное включение.​

🔸 Избегайте многократного включения одного и того же файла в рамках одного запроса без необходимости, чтобы минимизировать накладные расходы и потенциальные ошибки.
Что такое JIT? Как он работает?

JIT (Just-In-Time) компиляция представляет собой метод оптимизации выполнения кода, при котором код не компилируется заранее в процессе разработки, а только перед фактическим его выполнением во время работы программы. Это позволяет улучшить производительность программы за счет оптимизаций, которые могут быть применены с учетом конкретных условий выполнения.

Как это работает:

Исходный код на PHP: Начинаем с написания нашего кода на PHP, который является языком сценариев с динамической типизацией.

Компиляция в промежуточный код (opcode): Исходный код PHP компилируется в промежуточный байт-код, который представляет собой некий промежуточный представитель нашего кода, близкий к машинному коду, но все еще независимый от конкретной аппаратной платформы.

Исполнение байт-кода: Этот байт-код может быть интерпретирован непосредственно виртуальной машиной PHP (Zend VM), что происходит по умолчанию. Однако, в контексте JIT, в определенный момент, когда интерпретатор PHP обнаруживает, что некий участок кода используется достаточно часто, он может решить скомпилировать этот участок кода в машинный код «на лету».

JIT-компиляция: В процессе выполнения программы JIT-компилятор анализирует и оптимизирует участки кода, которые часто выполняются. Это позволяет улучшить производительность, так как оптимизации адаптированы к конкретному исполняющемуся коду и платформе.

Выполнение оптимизированного кода: После JIT-компиляции оптимизированный машинный код заменяет соответствующий участок байт-кода, что может значительно ускорить выполнение программы.

Этот подход позволяет более эффективно использовать ресурсы и достигнуть лучшей производительности в сравнении с чистым интерпретатором.
Как использовать принцип инверсии управления в Laravel? 

​Принцип инверсии управления (Inversion of Control, IoC) в Laravel реализуется с помощью сервис-контейнера, который управляет зависимостями между классами и их внедрением. Это позволяет создавать гибкую и тестируемую архитектуру приложений.

В нашем примере:

Определяется интерфейс PaymentGatewayInterface с методом charge, который должен быть реализован любым платежным шлюзом.

Класс StripePaymentGateway реализует этот интерфейс, предоставляя конкретную логику для обработки платежей через Stripe.

Класс OrderProcessor зависит от PaymentGatewayInterface. Вместо создания экземпляра конкретного платежного шлюза внутри себя, он получает его через конструктор, что соответствует принципу инверсии управления.

В сервис-контейнере Laravel регистрируется связь между PaymentGatewayInterface и его реализацией StripePaymentGateway с помощью метода $app->bind().

Создание экземпляра OrderProcessor осуществляется через сервис-контейнер с помощью метода $app->make(), который автоматически внедрит необходимые зависимости.

Использование IoC в Laravel позволяет:​

🔸 Разделить ответственность между классами, делая их более независимыми и удобными для тестирования.

🔸 Легко заменять реализации зависимостей, например, для использования другого платежного шлюза, достаточно зарегистрировать новую реализацию в контейнере.

🔸 Упрощает тестирование, позволяя подменять реальные реализации зависимостей на заглушки или моки.
Как можно выполнить PHP-код через eval() без явного использования eval()?

​Выполнение динамически сформированного PHP-кода без явного использования функции eval() возможно через альтернативные методы, каждый из которых имеет свои особенности и области применения. Ниже рассмотрены основные подходы:​

1. Использование функции create_function()

Ранее в PHP существовала функция create_function(), позволяющая создавать анонимные функции из строковых выражений. Однако начиная с PHP 7.2.0 эта функция объявлена устаревшей, а в PHP 8.0.0 удалена. Поэтому её использование не рекомендуется.​

2. Динамическое создание и подключение файлов

Другой подход заключается в создании временного PHP-файла с необходимым кодом и его последующем подключении с помощью конструкции include или require.​

Пример:

$code = '';$tempFile = tempnam(sys_get_temp_dir(), 'php');file_put_contents($tempFile, $code);include $tempFile;unlink($tempFile); // Удаляем временный файл после выполнения


Этот метод позволяет выполнить динамически созданный код, сохраняя его в отдельном файле. Однако он может быть менее эффективным из-за операций ввода-вывода и требует тщательной очистки временных файлов.​

3. Использование шаблонизаторов

Если цель заключается в динамическом формировании и отображении контента, рекомендуется использовать шаблонизаторы, такие как Twig или Smarty. Они обеспечивают безопасность и гибкость при работе с динамическими данными, избегая прямого выполнения кода.​

Пример с использованием Twig:
require_once '/path/to/vendor/autoload.php';$loader = new \Twig\Loader\ArrayLoader(['index' => 'Привет, {{ name }}!',]);$twig = new \Twig\Environment($loader);echo $twig->render('index', ['name' => 'мир']);
Опишите поведение при использовании traits с одинаковыми именами полей и / или методов?

При использовании traits с одинаковыми именами полей и/или методов, возникает конфликт имен, который не позволяет PHP однозначно определить, какое поле или метод должны использоваться.

Если в классе используется несколько traits, и в этих traits есть поля или методы с одинаковыми именами, возникает ошибка компиляции «trait method has not been applied», или «trait field has not been applied». По сути, PHP не знает, из какого trait следует взять поле или метод.

Для разрешения этого конфликта имен трейт может быть использован с использованием псевдонима. Например, можно использовать ключевое слово insteadof, чтобы указать, какой метод или поле должно использоваться

Также в PHP 7.2 и выше была добавлена возможность адаптировать методы или конфликтующие поля с помощью оператора as.
С помощью таких алиасов можно явным образом указать, какой метод или поле следует использовать из каждого трейта.
Как масштабировать PHP-приложение, чтобы оно выдерживало высокую нагрузку, и каковы некоторые связанные с этим проблемы?

Для масштабирования PHP-приложения можно использовать горизонтальное масштабирование, при котором несколько экземпляров приложения могут работать на разных серверах за балансировщиком нагрузки. Это поможет справиться с высокой нагрузкой. Одной из проблем является обеспечение согласованности данных в нескольких экземплярах приложения.

Другая задача — обеспечить эффективную связь между экземплярами приложения и гарантировать, что каждый экземпляр имеет достаточно ресурсов, чтобы справиться с нагрузкой. Для снижения нагрузки на серверы можно также использовать методы кэширования и оптимизации.
Влияет ли unset() на потребление памяти в PHP?

​В PHP функция unset() удаляет указанную переменную, разрывая связь между именем переменной и её данными. Однако это не гарантирует немедленного освобождения памяти, занятой этой переменной. PHP использует сборщик мусора, который освобождает память, когда это необходимо или когда завершается выполнение скрипта. ​
PHP

Присвоение переменной значения null также разрывает связь с её данными, но, в отличие от unset(), оставляет переменную в области видимости со значением null. Оба подхода сигнализируют сборщику мусора о возможности освобождения памяти, но фактическое время освобождения определяется внутренними механизмами PHP. ​

В большинстве случаев явное использование unset() или присвоение null не требуется, так как PHP автоматически управляет памятью. Однако в сценариях с длительно работающими скриптами или при обработке больших объемов данных явное освобождение памяти может быть полезным для предотвращения её исчерпания. ​

Важно отметить, что unset() удаляет только ссылку на данные. Если другие переменные ссылаются на те же данные, они останутся в памяти до тех пор, пока все ссылки не будут удалены. ​

Таким образом, хотя unset() может помочь в управлении памятью, его использование не всегда приводит к немедленному освобождению памяти. PHP полагается на сборщик мусора для эффективного управления ресурсами, и в большинстве случаев ручное вмешательство не требуется.
Что такое gap locks в MySQL?

Gap locks в MySQL — это блокировки, применяемые на диапазоне значений индексов таблицы, но не на конкретную запись. Они используются для решения проблем с возможными фантомными чтениями и уровнем изоляции транзакций.

Когда выполняется операция SELECT с условием по диапазону значений, MySQL устанавливает gap lock на промежуток (gap) между найденными итемами или между первым и последним итемами в результате выборки. Gap lock блокирует вставку или обновление других записей, которые попадают в этот промежуток, и предотвращает получение непоследовательных данных другими транзакциями.

Gap locks обеспечивают последовательность данных при выполнении поисковых запросов с условиями по диапазону значений в многопользовательской среде. Они помогают предотвратить возможные конфликты между транзакциями и гарантируют консистентность данных.

Однако, стоит отметить, что использование gap locks может увеличить вероятность блокировки и снизить производительность в высоконагруженных средах. Поэтому, при разработке приложений на PHP с использованием MySQL, важно тщательно оценить необходимость использования gap locks и продумать стратегию обработки блокировок для обеспечения оптимальной производительности и надежности системы.
Что такое type hinting, как работает, зачем нужен?

Type hinting в PHP — это возможность указывать ожидаемые или допустимые типы данных для параметров функций и функций-обратного вызова (callback). Он используется для определения типов аргументов функций и возвращаемых значений.

Type hinting выполняется при помощи объявления типа данных перед именем параметра функции или функции-обратного вызова.

В данном случае мы указываем, что параметры $a и $b должны быть целочисленного типа (int), а функция должна возвращать тоже целочисленное значение. Если будет передан несоответствующий тип данных, то PHP выдаст ошибку.

Type hinting в PHP имеет следующие преимущества и цели:

1. Увеличение надежности и безопасности кода: Type hinting позволяет контролировать типы данных, которые принимаются и возвращаются функциями, что может помочь предотвратить ошибки типизации и некорректное использование функций.

2. Улучшение понимания кода: Type hinting делает код более читаемым и понятным, особенно при работе в команде. Видя ожидаемые типы данных в объявлениях функций, другие разработчики легко могут понять, какие данные ожидаются для правильного использования функции.

3. Интеграция со средами разработки: Многие среды разработки, такие как PhpStorm, могут использовать информацию о типах данных для предоставления подсказок и автодополнения, что упрощает разработку и ускоряет процесс написания кода.

Однако, следует отметить, что type hinting в PHP не является строгой типизацией. Если передать аргумент, не соответствующий указанному типу, PHP будет пытаться выполнить приведение типов, что может привести к непредсказуемому поведению программы.
Чем опасен оператор @ в PHP?

​В PHP оператор @ используется для подавления сообщений об ошибках, возникающих при выполнении выражений. Однако его применение может привести к ряду проблем, особенно в контексте безопасности и отладки кода.​

Проблемы, связанные с использованием оператора @:

🔸 Подавление критических ошибок:

Оператор @ скрывает все сообщения об ошибках, включая критические. Это может привести к тому, что важные ошибки останутся незамеченными, что затруднит их обнаружение и исправление.​

🔸 Снижение производительности:

Использование @ может негативно влиять на производительность приложения. Даже если ошибки не возникают, PHP выполняет дополнительные операции для подавления сообщений, что может замедлить выполнение кода.​

🔸 Сложности в отладке:

Подавление ошибок затрудняет процесс отладки, поскольку разработчик не получает информации о возникших проблемах. Это может привести к длительному поиску причин некорректного поведения программы.​

Рекомендации по использованию:

🔹 Избегайте использования @:

Вместо подавления ошибок рекомендуется обрабатывать их с помощью конструкций try-catch или проверять условия перед выполнением потенциально проблемных операций.​

🔹 Настройка уровней отчетности об ошибках:

Используйте функции error_reporting() и set_error_handler() для управления выводом сообщений об ошибках и их обработкой. Это позволит более гибко контролировать поведение приложения при возникновении ошибок.​

В целом, использование оператора @ в PHP считается плохой практикой. Подавление ошибок может привести к пропуску критических проблем и усложнить процесс отладки. Рекомендуется явным образом обрабатывать возможные ошибки и исключения, что способствует повышению надежности и безопасности кода.
Как использовать конструктор запросов(query builder) для выполнения сложных запросов в Laravel?

Laravel предоставляет конструктор запросов, который позволяет строить SQL-запросы более бегло и читабельно, чем при написании исходного SQL. Конструктор запросов предоставляет ряд методов для построения запросов, включая select, where, join и orderBy.

В этом примере мы используем конструктор запросов, чтобы выбрать имя и номер телефона всех пользователей старше 25 лет, упорядоченные по имени. Мы соединяем таблицу contacts с таблицей users с помощью метода join и фильтруем результаты с помощью метода where.

Затем мы используем метод orderBy для сортировки результатов, а метод get — для получения результатов в виде коллекции.
Зачем нужно ключевое слово final?

Ключевое слово «final» в PHP используется для обозначения, что класс или метод не может быть изменен или унаследован в дочерних классах.

Если класс объявлен как final, то он не может быть унаследован, и не может быть основой для других классов.

Если метод объявлен как final, то он не может быть переопределен (перезаписан) в дочерних классах.

Если свойство объявлено как final, то его значение уже не может быть изменено, и оно становится константой