PHP / Magento Dev Blog

  • Publikacje
  • O autorze
  • Kontakt

PHP 8.0 – premiera i pierwsze wrażenia z nowych funkcji

by Henryk Tews / czwartek, 26 listopada 2020 / Opublikowano w PHP


PHP 8.0 wyszło oficjalnie 26 listopada 2020. Pisałem o zapowiedziach w lipcu – teraz gdy mam już finalną wersję, czas na pierwsze praktyczne wrażenia. Sprawdzam co działa tak jak obiecywano, co zaskoczyło i na co uważać przy migracji starszych projektów.

Match expression – używam od razu

Match wszedł do mojego kodu niemal natychmiast. Nie tylko dlatego że jest krótszy od switch – przede wszystkim dlatego że rzuca UnhandledMatchError gdy żaden arm nie pasuje. To eliminuje ciche błędy gdzie switch nic nie robił bo zapomniałeś o jednym przypadku:

<?php

declare(strict_types=1);

// Stary kod - switch milczał przy nieznanym statusie
function getStatusLabel(int $status): string
{
    switch ($status) {
        case 1: return 'Aktywny';
        case 2: return 'Nieaktywny';
        // zapomniano o 3 - switch zwraca null, PHP ostrzega, kod idzie dalej
    }
}

// PHP 8.0 - match rzuca UnhandledMatchError przy nieznanym statusie
function getStatusLabel(int $status): string
{
    return match($status) {
        1 => 'Aktywny',
        2 => 'Nieaktywny',
        3 => 'Oczekuje na weryfikację',
        default => throw new \InvalidArgumentException("Nieznany status: {$status}"),
    };
}

Szczególnie przydatny przy mapowaniu stałych Magento – status zamówienia, widoczność produktu, typ atrybutu.

Named Arguments w praktyce – czytelność kodu

Named arguments okazały się bardziej przydatne niż myślałem, ale nie wszędzie gdzie spodziewałem. Największy zysk przy wbudowanych funkcjach PHP z wieloma opcjonalnymi parametrami:

<?php

// Kto pamięta kolejność parametrów array_slice?
$page = array_slice($items, offset: ($currentPage - 1) * $pageSize, length: $pageSize);

// str_contains, str_starts_with, str_ends_with - nowe w PHP 8.0, super czytelne
$sku = 'MG-RED-XL-42';

if (str_starts_with($sku, 'MG-')) {
    echo 'Produkt Magento';
}

if (str_ends_with($sku, '-42')) {
    echo 'Rozmiar 42';
}

if (str_contains($sku, '-RED-')) {
    echo 'Kolor: czerwony';
}

// Zastępuje: strpos($sku, '-RED-') !== false
// Zdecydowanie czytelniejsze

str_contains, str_starts_with, str_ends_with – niedocenione nowości

Przy okazji PHP 8.0 dostaliśmy trzy funkcje, które wyczyszczą połowę warunków z strpos() !== false w istniejącym kodzie. To nie jest wielka filozofia, ale poprawia czytelność każdego dnia:

<?php

// Stare podejście - nieczytelne
if (strpos($url, '/admin') === 0) { /* ... */ }
if (strpos($email, '@example.com') !== false) { /* ... */ }
if (substr($filename, -4) === '.php') { /* ... */ }

// PHP 8.0 - intencja jasna od razu
if (str_starts_with($url, '/admin')) { /* ... */ }
if (str_contains($email, '@example.com')) { /* ... */ }
if (str_ends_with($filename, '.php')) { /* ... */ }

Constructor Promotion w Magento 2 – ostrożnie

Constructor promotion wygląda świetnie w czystym PHP 8.0. W Magento 2 jest jeden haczyk: narzędzia generujące kod (szczególnie starsze wersje setup:di:compile) mogą mieć problem z analizą konstruktorów używających promotion. Bezpieczne podejście to używać promotion w nowych klasach które nie są częścią DI container (value objects, DTO), a w klasach wstrzykiwanych przez Magento trzymać się starego stylu do czasu gdy projekt oficjalnie przejdzie na Magento 2.4.x z pełnym wsparciem PHP 8.0:

<?php

declare(strict_types=1);

// Bezpieczne - DTO/Value Object poza DI
class OrderExportDto
{
    public function __construct(
        public readonly int $orderId,
        public readonly string $status,
        public readonly float $grandTotal,
        public readonly \DateTimeImmutable $createdAt
    ) {}
}

// Ostrożnie - klasa wstrzykiwana przez Magento DI
// Zostaw stary styl dla bezpieczeństwa przy starszych wersjach Magento
class OrderExportService
{
    private OrderRepositoryInterface $orderRepository;
    private \Psr\Log\LoggerInterface $logger;

    public function __construct(
        OrderRepositoryInterface $orderRepository,
        \Psr\Log\LoggerInterface $logger
    ) {
        $this->orderRepository = $orderRepository;
        $this->logger          = $logger;
    }
}

Nullsafe operator – rzeczywiste użycie

<?php

// Realne użycie w Magento - łańcuchy metod które mogą zwrócić null
$countryName = $order
    ?->getShippingAddress()
    ?->getCountryId();

$customerGroupCode = $quote
    ?->getCustomer()
    ?->getGroupId();

// Uwaga - nullsafe nie zastąpi sprawdzenia wyjątków
// getById() rzuca NoSuchEntityException, nie zwraca null
// Tutaj nullsafe nie pomoże - potrzebujesz try/catch
try {
    $product = $productRepository->getById(42);
    $price   = $product->getPrice();
} catch (\Magento\Framework\Exception\NoSuchEntityException $e) {
    $price = 0.0;
}

Fibers – zapowiedź na PHP 8.1

PHP 8.1 (planowany na koniec 2021) zapowiada Fibers – lekkie korutyny umożliwiające asynchroniczne wykonanie kodu bez zewnętrznych bibliotek jak ReactPHP. To fundamentalna zmiana dla PHP jako języka serwerowego, choć dla typowych aplikacji Magento wpływ będzie ograniczony.

Podsumowanie

PHP 8.0 to najlepsza wersja PHP w historii – nie ma co do tego wątpliwości. Match, named arguments, str_contains i nullsafe operator wchodzą do kodu od razu. Union types i constructor promotion wymagają więcej przemyślenia przy integracji z Magento. Migracja z PHP 7.4 jest bezbolesna jeśli projekt nie używa przestarzałych funkcji – uruchom php -l i PHP Compatibility Checker przed aktualizacją.

About Henryk Tews

Co możesz przeczytać następne

AI-assisted optymalizacja SQL – LLM + EXPLAIN + Blackfire, 5x przyspieszenie
Dijkstra – najkrótsza ścieżka z wagami, SplMinHeap, zastosowania w e-commerce
PHP 8.1 preview – enumy, readonly properties, intersection types, fibers
  • Publikacje
  • O autorze
  • Kontakt

© 2026 Created by

GÓRA
Zarządzaj zgodą
Aby zapewnić jak najlepsze wrażenia, korzystamy z technologii, takich jak pliki cookie, do przechowywania i/lub uzyskiwania dostępu do informacji o urządzeniu. Zgoda na te technologie pozwoli nam przetwarzać dane, takie jak zachowanie podczas przeglądania lub unikalne identyfikatory na tej stronie. Brak wyrażenia zgody lub wycofanie zgody może niekorzystnie wpłynąć na niektóre cechy i funkcje.
Funkcjonalne Zawsze aktywne
Przechowywanie lub dostęp do danych technicznych jest ściśle konieczny do uzasadnionego celu umożliwienia korzystania z konkretnej usługi wyraźnie żądanej przez subskrybenta lub użytkownika, lub wyłącznie w celu przeprowadzenia transmisji komunikatu przez sieć łączności elektronicznej.
Preferencje
Przechowywanie lub dostęp techniczny jest niezbędny do uzasadnionego celu przechowywania preferencji, o które nie prosi subskrybent lub użytkownik.
Statystyka
Przechowywanie techniczne lub dostęp, który jest używany wyłącznie do celów statystycznych. Przechowywanie techniczne lub dostęp, który jest używany wyłącznie do anonimowych celów statystycznych. Bez wezwania do sądu, dobrowolnego podporządkowania się dostawcy usług internetowych lub dodatkowych zapisów od strony trzeciej, informacje przechowywane lub pobierane wyłącznie w tym celu zwykle nie mogą być wykorzystywane do identyfikacji użytkownika.
Marketing
Przechowywanie lub dostęp techniczny jest wymagany do tworzenia profili użytkowników w celu wysyłania reklam lub śledzenia użytkownika na stronie internetowej lub na kilku stronach internetowych w podobnych celach marketingowych.
  • Zarządzaj opcjami
  • Zarządzaj serwisami
  • Zarządzaj {vendor_count} dostawcami
  • Przeczytaj więcej o tych celach
Zobacz preferencje
  • {title}
  • {title}
  • {title}