Да, PHP умирает... уже 20 лет, а зарплаты всё растут. Но готов ли ты к роли кофаундера? 🚀
Пока эксперты хоронят стек, Proglib App строит будущее EdTech-платформы. Это экосистема с квизами, курсами и ИИ-тьюторами. MVP в проде, юзеры на борту, и проекту нужен технический лидер, готовый взять на себя архитектуру.
Это шанс выйти за рамки привычного бэкенда и стать партнёром в амбициозном продукте.
🛠️ Стек (без legacy-боли):
React 18, TypeScript, Vite, Express 5, PostgreSQL, Drizzle ORM.
Твои задачи:
• Проектировать масштабируемую архитектуру и пилить фичи. • Рефакторить, писать тесты и делать качественное код-ревью. • Работать автономно в связке с основателем.
Идеальный мэтч, если:
• Уверенно владеешь TS, React и Node.js (или готов быстро пересесть). • Используешь Claude Code и Cursor для ускорения разработки. • Бонус: опыт создания ИИ-агентов и интерес к обучению.
Удалёнка, гибкий график, отсутствие бюрократии.
Готов делать продукт, которым будут пользоваться тысячи коллег? Пиши о себе и кидай GitHub 👇
Пока эксперты хоронят стек, Proglib App строит будущее EdTech-платформы. Это экосистема с квизами, курсами и ИИ-тьюторами. MVP в проде, юзеры на борту, и проекту нужен технический лидер, готовый взять на себя архитектуру.
Это шанс выйти за рамки привычного бэкенда и стать партнёром в амбициозном продукте.
🛠️ Стек (без legacy-боли):
React 18, TypeScript, Vite, Express 5, PostgreSQL, Drizzle ORM.
Твои задачи:
• Проектировать масштабируемую архитектуру и пилить фичи. • Рефакторить, писать тесты и делать качественное код-ревью. • Работать автономно в связке с основателем.
Идеальный мэтч, если:
• Уверенно владеешь TS, React и Node.js (или готов быстро пересесть). • Используешь Claude Code и Cursor для ускорения разработки. • Бонус: опыт создания ИИ-агентов и интерес к обучению.
Удалёнка, гибкий график, отсутствие бюрократии.
Готов делать продукт, которым будут пользоваться тысячи коллег? Пиши о себе и кидай GitHub 👇
❤4😁4🥱1
💡 Используй array_map + array_filter с null колбэком для мультимерного flatten
Не городи рекурсию там, где достаточно одной строки через SPL или array_merge(...array_map(...)):
Но вот где реальная магия — когда нужно flatten + filter за раз:
⚠️ Ловушки
— array_merge(...$matrix) упадёт с ошибкой на пустом массиве [].
— Работает только для одного уровня вложенности.
📦 В продакшене это спасает при обработке сгруппированных результатов из БД, когда ORM отдаёт Collection[] и нужно быстро схлопнуть в плоский список.
Библиотека пхпшника
#vardump
Не городи рекурсию там, где достаточно одной строки через SPL или array_merge(...array_map(...)):
$matrix = [[1, 2, 3], [4, 5], [6]];
// ❌ Цикл или рекурсия
$result = [];
foreach ($matrix as $row) {
foreach ($row as $item) {
$result[] = $item;
}
}
// ✅ Одна строка
$result = array_merge(...$matrix);
// [1, 2, 3, 4, 5, 6]
Но вот где реальная магия — когда нужно flatten + filter за раз:
$data = [[1, 0, 2], [null, 3], [false, 4, '']];
$result = array_values(
array_filter(
array_merge(...$data)
)
);
// [1, 2, 3, 4]
— array_merge(...$matrix) упадёт с ошибкой на пустом массиве [].
— Работает только для одного уровня вложенности.
📦 В продакшене это спасает при обработке сгруппированных результатов из БД, когда ORM отдаёт Collection[] и нужно быстро схлопнуть в плоский список.
Библиотека пхпшника
#vardump
Please open Telegram to view this post
VIEW IN TELEGRAM
👍6❤2🔥1😁1🥱1
⏳ Как устроена типизация в PHP 8.x?
PHP долго был языком с динамической типизацией и репутацией «всё приведётся само». Начиная с PHP 7, а особенно с версий 8.0–8.3, система типов стала по-настоящему выразительной.
🔍 Что появилось и зачем
🔹 Union Types (8.0)
Тип может быть одним из нескольких:
Раньше это документировали только в PHPDoc и надеялись на лучшее.
🔹 mixed, never, void
▪️ mixed — явно говорит "любой тип", в отличие от отсутствия типа
▪️ never — функция никогда не вернёт управление (бросает исключение или завершает процесс)
▪️ void — функция ничего не возвращает
🔹 Intersection Types (8.1)
Значение должно реализовывать все указанные интерфейсы одновременно:
Полезно при работе с коллекциями и DTO, которые соответствуют нескольким контрактам.
🔹 Enums (8.1)
Наконец-то нативные перечисления вместо констант в классах:
Теперь нельзя передать произвольную строку туда, где ожидается Status.
🔹 Readonly properties (8.1) и readonly классы (8.2)
В 8.2 можно пометить весь класс как readonly — все свойства автоматически становятся иммутабельными. Отлично подходит для Value Objects и DTO.
🔹 DNF Types — Disjunctive Normal Form (8.2)
Комбинация union и intersection:
Тип читается как: "реализует оба интерфейса, либо null".
🔹 typed class constants (8.3)
До 8.3 константы классов не имели типов — теперь контракт полный.
⚠️ Нюансы
— declare(strict_types=1) обязателен, иначе PHP будет тихо приводить типы и "5" пройдёт там, где ожидается int
— Intersection types не работают с примитивами только с интерфейсами и классами
— readonly нельзя применить к static свойствам
— Enums не поддерживают наследование
Библиотека пхпшника
PHP долго был языком с динамической типизацией и репутацией «всё приведётся само». Начиная с PHP 7, а особенно с версий 8.0–8.3, система типов стала по-настоящему выразительной.
🔹 Union Types (8.0)
Тип может быть одним из нескольких:
function process(int|string $id): User|null {
// ...
}Раньше это документировали только в PHPDoc и надеялись на лучшее.
🔹 mixed, never, void
▪️ mixed — явно говорит "любой тип", в отличие от отсутствия типа
▪️ never — функция никогда не вернёт управление (бросает исключение или завершает процесс)
▪️ void — функция ничего не возвращает
function redirect(string $url): never {
header("Location: $url");
exit();
}🔹 Intersection Types (8.1)
Значение должно реализовывать все указанные интерфейсы одновременно:
function process(Serializable&Countable $data): void {
// ...
}Полезно при работе с коллекциями и DTO, которые соответствуют нескольким контрактам.
🔹 Enums (8.1)
Наконец-то нативные перечисления вместо констант в классах:
enum Status: string {
case Active = 'active';
case Inactive = 'inactive';
}
function setStatus(Status $status): void { ... }Теперь нельзя передать произвольную строку туда, где ожидается Status.
🔹 Readonly properties (8.1) и readonly классы (8.2)
class User {
public function __construct(
public readonly int $id,
public readonly string $name,
) {}
}В 8.2 можно пометить весь класс как readonly — все свойства автоматически становятся иммутабельными. Отлично подходит для Value Objects и DTO.
🔹 DNF Types — Disjunctive Normal Form (8.2)
Комбинация union и intersection:
function handle(Countable&Iterator|null $collection): void {
// ...
}Тип читается как: "реализует оба интерфейса, либо null".
🔹 typed class constants (8.3)
class Config {
const string VERSION = '1.0.0';
const int MAX_RETRIES = 3;
}До 8.3 константы классов не имели типов — теперь контракт полный.
— declare(strict_types=1) обязателен, иначе PHP будет тихо приводить типы и "5" пройдёт там, где ожидается int
— Intersection types не работают с примитивами только с интерфейсами и классами
— readonly нельзя применить к static свойствам
— Enums не поддерживают наследование
Библиотека пхпшника
Please open Telegram to view this post
VIEW IN TELEGRAM
👍8❤3🔥2
Forwarded from Библиотека собеса по PHP | вопросы с собеседований
Fiber — примитив для кооперативной многозадачности. Позволяет приостановить выполнение функции в произвольной точке через Fiber::suspend() и возобновить позже, передав значение обратно.
Главное отличие от async/await: Fibers не заражают сигнатуры. В JS как только где-то await — вся цепочка выше должна быть async. В PHP Fiber::suspend() можно вызвать на любой глубине стека, не меняя сигнатуры вызывающих функций.
Второе отличие — Fibers это низкоуровневый примитив, event loop в них не встроен. Сами по себе они не дают параллелизма — это строительный блок, поверх которого ReactPHP, Revolt и Amp строят настоящую неблокирующую конкурентность.
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥8👍2❤1
Иногда пишу нейронке «спасибо»
А вы как? Благодарите после удачного ответа или общаетесь с ИИ как с микроволновкой?
#междусобойчик
Please open Telegram to view this post
VIEW IN TELEGRAM
👍8😁5🔥1
🔥 PDO vs MySQLi
Каждый рано или поздно задаёт себе вопрос:
Чаще всего ответ → «так исторически сложилось». Но давай разберём более делательно.
🔹 MySQLi — быстрый, но привязанный
Работает только с MySQL/MariaDB. Зато даёт нативные async-запросы через mysqli_poll(), что редкость, но иногда критично.
Синтаксис bind_param с типами — это боль при большом числе параметров.
🔹 PDO — абстракция с характером
Поддерживает 12+ драйверов: MySQL, PostgreSQL, SQLite, MSSQL, и др. Именованные плейсхолдеры — читаемость на уровне:
Одна нотация везде. Смена БД без переписывания логики запросов.
📊 Что реально важно на проекте
Смена БД когда-либо → PDO (без вариантов)
Legacy MySQL-проект → MySQLi уже там, не трогай
Нужен fetchAll в объекты → PDO::FETCH_CLASS чище
Производительность → разница < 1%, забудь
💡 Вердикт
MySQLi — это не "плохо". Это инструмент под конкретный стек.
Но если ты стартуешь проект с нуля или пишешь библиотеку, то лучше PDO. Любой ORM (Doctrine, Eloquent) под капотом использует именно его.
🐸 Библиотека пхпшника
#элементарный_выбор
Каждый рано или поздно задаёт себе вопрос:
«А почему мы вообще используем именно это?»
Чаще всего ответ → «так исторически сложилось». Но давай разберём более делательно.
🔹 MySQLi — быстрый, но привязанный
Работает только с MySQL/MariaDB. Зато даёт нативные async-запросы через mysqli_poll(), что редкость, но иногда критично.
$stmt = $mysqli->prepare("SELECT * FROM users WHERE id = ?");
$stmt->bind_param("i", $id);
$stmt->execute();Синтаксис bind_param с типами — это боль при большом числе параметров.
🔹 PDO — абстракция с характером
Поддерживает 12+ драйверов: MySQL, PostgreSQL, SQLite, MSSQL, и др. Именованные плейсхолдеры — читаемость на уровне:
$stmt = $pdo->prepare("SELECT * FROM users WHERE id = :id AND role = :role");
$stmt->execute(['id' => $id, 'role' => $role]);Одна нотация везде. Смена БД без переписывания логики запросов.
📊 Что реально важно на проекте
Смена БД когда-либо → PDO (без вариантов)
Legacy MySQL-проект → MySQLi уже там, не трогай
Нужен fetchAll в объекты → PDO::FETCH_CLASS чище
Производительность → разница < 1%, забудь
MySQLi — это не "плохо". Это инструмент под конкретный стек.
Но если ты стартуешь проект с нуля или пишешь библиотеку, то лучше PDO. Любой ORM (Doctrine, Eloquent) под капотом использует именно его.
#элементарный_выбор
Please open Telegram to view this post
VIEW IN TELEGRAM
👍4🔥2❤1😢1
⌨️ Топ-вакансий по PHP за неделю
PHP (Laravel) backend разработчик — от 80 000 до 130 000 ₽ — удалёнка
Middle Fullstack Developer (Laravel + Vue.js) (Contractor, Remote) — от 2500 до 3800 $ — удалёнка
Senior / Lead PHP (Symfony) Engineer — от 3000 до 7000 $ — удалёнка
➡️ Еще больше топовых вакансий — в нашем канале PHP Jobs
PHP (Laravel) backend разработчик — от 80 000 до 130 000 ₽ — удалёнка
Middle Fullstack Developer (Laravel + Vue.js) (Contractor, Remote) — от 2500 до 3800 $ — удалёнка
Senior / Lead PHP (Symfony) Engineer — от 3000 до 7000 $ — удалёнка
Please open Telegram to view this post
VIEW IN TELEGRAM
❤1👍1🔥1
🔧 Фишка PHP
match с true — замена громоздкому if-elseif
Читается сверху вниз, возвращает значение, не требует break.
💡 match использует строгое сравнение (===), а при отсутствии default бросает UnhandledMatchError. Всегда добавляй default если входные данные непредсказуемы.
🐸 Библиотека пхпшника
match с true — замена громоздкому if-elseif
// ❌ Было
if ($score >= 90) {
$grade = 'A';
} elseif ($score >= 75) {
$grade = 'B';
} elseif ($score >= 60) {
$grade = 'C';
} else {
$grade = 'F';
}
// ✅ Стало
$grade = match(true) {
$score >= 90 => 'A',
$score >= 75 => 'B',
$score >= 60 => 'C',
default => 'F',
};
Читается сверху вниз, возвращает значение, не требует break.
Please open Telegram to view this post
VIEW IN TELEGRAM
👍19❤3🔥3🥱3
This media is not supported in your browser
VIEW IN TELEGRAM
😁2🥱2
📅 Старт курса — 20 апреля.
Если хотите разобраться, как строить управляемые агентные системы:
P.S.
Please open Telegram to view this post
VIEW IN TELEGRAM
🐘 PHP для начинающих (и напоминание для опытных)
🔹 Eloquent Relationships или почему связи это не страшно
Большинство новичков понимают модели. Один класс — одна таблица, всё логично.
Страшно становится, когда таблицы нужно связывать между собой. В голове сразу всплывают JOIN-ы из курса по SQL, и хочется закрыть ноутбук.
Eloquent это скрывает. Ты просто описываешь словами, как модели связаны друг с другом.
Всё. Laravel сам догадается, что posts.user_id ссылается на users.id — по соглашению об именовании. Теперь можно писать читаемый код:
Никаких JOIN-ов вручную. Никакого WHERE user_id = ?. Ты работаешь с объектами, а не с сырым SQL.
Три связи, которые покрывают 90% случаев
▪️ hasMany — у пользователя много постов
▪️ belongsTo — пост принадлежит пользователю
▪️ belongsToMany — пост имеет много тегов, тег стоит на многих постах (промежуточная таблица)
Связи — это просто способ сказать Laravel: «вот как мои таблицы соотносятся». Дальше он разберётся сам.
Библиотека пхпшника
#php_азбука
🔹 Eloquent Relationships или почему связи это не страшно
Большинство новичков понимают модели. Один класс — одна таблица, всё логично.
Страшно становится, когда таблицы нужно связывать между собой. В голове сразу всплывают JOIN-ы из курса по SQL, и хочется закрыть ноутбук.
Eloquent это скрывает. Ты просто описываешь словами, как модели связаны друг с другом.
// User.php
public function posts(): HasMany
{
return $this->hasMany(Post::class);
}
// Post.php
public function user(): BelongsTo
{
return $this->belongsTo(User::class);
}
Всё. Laravel сам догадается, что posts.user_id ссылается на users.id — по соглашению об именовании. Теперь можно писать читаемый код:
// Все посты пользователя
$user->posts;
// Автор конкретного поста
$post->user->name;
// Посты только опубликованные
$user->posts()->where('published', true)->get();
Никаких JOIN-ов вручную. Никакого WHERE user_id = ?. Ты работаешь с объектами, а не с сырым SQL.
Три связи, которые покрывают 90% случаев
▪️ hasMany — у пользователя много постов
▪️ belongsTo — пост принадлежит пользователю
▪️ belongsToMany — пост имеет много тегов, тег стоит на многих постах (промежуточная таблица)
Связи — это просто способ сказать Laravel: «вот как мои таблицы соотносятся». Дальше он разберётся сам.
Библиотека пхпшника
#php_азбука
👍5🔥1
Не городи microtime(true) в начале index.php, когда PHP сделал это за тебя:
// ❌ Часто вижу
define('APP_START', microtime(true));
// ... где-то в конце
$elapsed = microtime(true) - APP_START;
// ✅ Уже есть
$elapsed = microtime(true) - $_SERVER['REQUEST_TIME_FLOAT'];
Разница принципиальная. APP_START фиксирует момент после автозагрузки, DI-контейнера и прочего bootstrap'а. REQUEST_TIME_FLOAT — реальный старт запроса на уровне PHP-FPM.
// Логируй реальное время ответа в footer или middleware
header('X-Response-Time: ' . round(
(microtime(true) - $_SERVER['REQUEST_TIME_FLOAT']) * 1000, 2
) . 'ms');
— В CLI REQUEST_TIME_FLOAT есть, но равен моменту запуска скрипта.
— Не путай с REQUEST_TIME (int, секунды) — для замеров он бесполезен.
Библиотека пхпшника
#vardump
Please open Telegram to view this post
VIEW IN TELEGRAM
👍10🔥4❤3
🔥 Named arguments — удобство, которое ломает рефакторинг
Имя параметра теперь публичный контракт. Rector, IDE-рефакторинг, переименование в наследнике — всё это теперь breaking change.
💬 А вы юзаете named arguments в публичном API или только внутри?
Библиотека пхпшника
#междусобойчик
// Переименовал параметр — и где-то тихо упало
function createUser(string $name, bool $isActive): void {}
createUser(name: 'Ivan', active: true); // Fatal
Имя параметра теперь публичный контракт. Rector, IDE-рефакторинг, переименование в наследнике — всё это теперь breaking change.
Библиотека пхпшника
#междусобойчик
Please open Telegram to view this post
VIEW IN TELEGRAM
😁5👍1🔥1
Forwarded from Библиотека собеса по PHP | вопросы с собеседований
Классическая ловушка с утечкой состояния 👇
📦 Задание
Написали систему обработки платёжных webhook'ов. В логах периодически замечают, что у одной транзакции записывается merchant_id другой. Баг плавающий, воспроизводится только под нагрузкой. Найдите проблему и исправьте:
class MerchantContext
{
private static ?string $currentMerchantId = null;
public static function set(string $id): void
{
self::$currentMerchantId = $id;
}
public static function get(): ?string
{
return self::$currentMerchantId;
}
public static function clear(): void
{
self::$currentMerchantId = null;
}
}
class WebhookProcessor
{
public function handle(WebhookPayload $payload): void
{
MerchantContext::set($payload->merchantId);
$fiber = new Fiber(function () use ($payload): void {
$result = $this->validateSignature($payload);
// симулируем I/O — запрос к внешнему сервису
Fiber::suspend();
$this->auditLogger->log(
'Webhook processed for: ' . MerchantContext::get(),
['amount' => $payload->amount]
);
});
$fiber->start();
// ... другие файберы запускаются здесь
$fiber->resume();
}
}
🔹 Задачи
— Объяснить, почему MerchantContext::get() после resume() может вернуть merchant_id другого запроса
— Исправить так, чтобы контекст не терялся при переключении между файберами
Ставьте → 🔥 если нравится формат. Если нет → 🌚
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥19😁2❤1👍1🤔1
Простая команда на случай, когда надо быстро и в удобном формате прочитать CSV-файл в терминале:
$ cat inventory.csv | column -t -s,
Флаг
-s указывает на использование запятых в качестве разделителей, а -t форматирует выходные данные в чистую таблицу.📍 Навигация: Вакансии • Задачи • Вопросы с собеса
Please open Telegram to view this post
VIEW IN TELEGRAM
👍16🔥2❤1
💥 Открытый вебинар | ИИ-агенты в продакшене: от хайпа к деньгам
Агенты уже везде. Но мало кто признаётся, сколько денег сжёг на бесконечных циклах, галлюцинациях в RAG и отсутствии мониторинга.
Полина Полунина, руководитель AI-направления Альфа-Банка, расскажет честно:
▪️ Чем агент отличается от «просто GPT с промптом» и когда бизнесу достаточно обычного LLM
▪️ 3 реальных кейса из корпоративной среды: что взлетело, а что нет
▪️ Live-демо работающего агента
▪️ ТОП-5 граблей, на которые наступают команды при внедрении
⏱️ 10 марта в 19:00 (МСК)
🎁 Участники получат промокод на скидку на самый полный курс по ИИ-агентам
👉 Регистрируйся
Агенты уже везде. Но мало кто признаётся, сколько денег сжёг на бесконечных циклах, галлюцинациях в RAG и отсутствии мониторинга.
Полина Полунина, руководитель AI-направления Альфа-Банка, расскажет честно:
▪️ Чем агент отличается от «просто GPT с промптом» и когда бизнесу достаточно обычного LLM
▪️ 3 реальных кейса из корпоративной среды: что взлетело, а что нет
▪️ Live-демо работающего агента
▪️ ТОП-5 граблей, на которые наступают команды при внедрении
⏱️ 10 марта в 19:00 (МСК)
🎁 Участники получат промокод на скидку на самый полный курс по ИИ-агентам
👉 Регистрируйся
🥱2❤1