PHPerKaigi 2025

Nesnelerin Kalıtımı

Kalıtım iyi kurgulanmış bir programlama prensibidir ve PHP bu prensibi kendi nesne modelinde kullanır. Bu prensip çoğu sınıf ve nesnenin bir diğeri ile ilişkisini düzenler.

Örneğin, bir sınıf (extend ile) genişletildiğinde alt sınıf üst sınıftaki genel ve korunmuş yöntemlerin tümünü miras alır. Bir sınıf bu yöntemleri aynı isimdeki yöntemlerle geçersizleştirmedikçe bunlar özgün işlevselliklerini korur.

Bu, işlevselliği tanımlamak ve soyutlamak için ve paylaşılan işlevselliğin tamamını yeni baştan gerçeklemeksizin benzer nesnelerde ek işlevsellikleri gerçeklemek için kullanışlıdır.

Bir ebeveyn sınıfın private yöntemlerine bir çocuk sınıftan erişilemez. Sonuç olarak, normal miras alma kurallarına aldırmadan çocuk sınıfların bu private yöntemleri yeniden gerçeklemesi gerekebilir. PHP 8.0.0 öncesinde, private yöntemlere bunun yanında final ve static sınırlamaları da uygulanırdı. PHP 8.0.0 ve sonrasında zorunlu olan tek private yöntem kısıtlaması, private final kuruculardır, çünkü bu, yerleşik statik yöntemleri kullanırken kurucuyu "devre dışı" bırakmanın yaygın bir yoludur.

Yöntemlerin, özelliklerin ve sabitlerin görünürlüğü gevşetilebilir, örn. protected bir yöntem public olarak imlenebilir; ancak sınırlandırılamaz, örneğin bir public özellik, private olarak imlenebilir. Görünürlüğü kısıtlanabilen kurucular istisnadır, örn. bir public kurucu, bir çocuk sınıfta private olarak imlenebilir.

Bilginize:

Otomatik yükleme kullanılmadıkça, sınıflar kullanılmadan önce tanımlanmalıdır. Eğer bir sınıf diğerini genişletiyorsa ebeveyn sınıf çocuk sınıfın yapısı kurulmadan önce bildirilmiş olmalıdır. Bu kural diğer sınıfları ve arayüzleri miras alan tüm sınıflara uygulanır.

Bilginize:

Okunup yazılabilir bir özellik salt-okunur özellik vb. ile geçersiz kılınamaz.

<?php

class A {
public
int $prop;
}
class
B extends A {
// Geçersiz: oku-yaz özellik salt-okunur yapılamaz
public readonly int $prop;
}
?>

Örnek 1 Inheritance Example

<?php

class Foo
{
public function
printItem($string)
{
echo
'Foo: ' . $string . PHP_EOL;
}

public function
printPHP()
{
echo
'PHP is great.' . PHP_EOL;
}
}

class
Bar extends Foo
{
public function
printItem($string)
{
echo
'Bar: ' . $string . PHP_EOL;
}
}

$foo = new Foo();
$bar = new Bar();
$foo->printItem('baz'); // Çıktısı: 'Foo: baz'
$foo->printPHP(); // Çıktısı: 'PHP is great'
$bar->printItem('baz'); // Çıktısı: 'Bar: baz'
$bar->printPHP(); // Çıktısı: 'PHP is great'

?>

Dahili Sınıflar ve Dönüş Türü Uyumluluğu

PHP 8.1 öncesinde, dahili sınıfların vaya yöntemlerin çoğunda dönüş türü bildirimi yoktu ve bunları genişletirken dönüş türüne izin veriliyordu.

PHP 8.1.0 ve sonrasında, çoğu dahili yöntem dönüş türlerini "geçici olarak" bildirmeye başladı, bu durumda yöntemin dönüş türü, genişletilen ebeveyn ile uyumlu olmalıdır; aksi takdirde, bir kullanımdan kaldırma bildirimi çıktılanır. Açık bir dönüş türü bildiriminin olmayışı, ayrıca yapısal bir uyuşmazlık olarak kabul edilmekte ve dolayısıyla kullanımdan kaldırma bildirimiyle sonuçlanmaktadır.

PHP'nin sürümler arası uyumluluk endişeleri nedeniyle geçersiz kılma yöntemi için dönüş türü bildirilemezse, kullanımdan kaldırma bildirimini sessizleştirmek için ReturnTypeWillChange özniteliği eklenebilir.

Örnek 2 - Dönüş türünü bildirmeyen geçersiz kılma yöntemi

<?php
class MyDateTime extends DateTime
{
public function
modify(string $modifier) { return false; }
}

// "Deprecated: Return type of MyDateTime::modify(string $modifier) should either be compatible with DateTime::modify(string $modifier): DateTime|false, or the #[\ReturnTypeWillChange] attribute should be used to temporarily suppress the notice" as of PHP 8.1.0
// Kullanımı önerilmiyor: MyDateTime::modify(string $modifier) için dönüş türü ya DateTime::modify(string $modifier): DateTime|false ile uymlu olmalı ya da PHP 8.1.0 ve sonrasında uyarıyı susturmak için #[\ReturnTypeWillChange] özniteliği kullanılmış olmalıdır.
?>

Örnek 3 - Yanlış dönüş türü bildiren geçersiz kılma yöntemi

<?php
class MyDateTime extends DateTime
{
public function
modify(string $modifier): ?DateTime { return null; }
}

// "Deprecated: Return type of MyDateTime::modify(string $modifier): ?DateTime should either be compatible with DateTime::modify(string $modifier): DateTime|false, or the #[\ReturnTypeWillChange] attribute should be used to temporarily suppress the notice" as of PHP 8.1.0
// Kullanımı önerilmiyor: MyDateTime::modify(string $modifier): ?DateTime için dönüş türü ya DateTime::modify(string $modifier): DateTime|false ile uymlu olmalı ya da PHP 8.1.0 ve sonrasında uyarıyı susturmak için #[\ReturnTypeWillChange] özniteliği kullanılmış olmalıdır.
?>

Örnek 4 - Yanlış dönüş türü bildiren ama kullanımı önerilmiyor uyarısı vermeyen geçersiz kılma yöntemi

<?php
class MyDateTime extends DateTime
{
/**
* @return DateTime|false
*/
#[\ReturnTypeWillChange]
public function
modify(string $modifier) { return false; }
}

// Bir uyarı tetiklenmez
?>
add a note

User Contributed Notes 6 notes

up
214
jackdracona at msn dot com
14 years ago
Here is some clarification about PHP inheritance – there is a lot of bad information on the net. PHP does support Multi-level inheritance. (I tested it using version 5.2.9). It does not support multiple inheritance.

This means that you cannot have one class extend 2 other classes (see the extends keyword). However, you can have one class extend another, which extends another, and so on.

Example:

<?php
class A {
// more code here
}

class
B extends A {
// more code here
}

class
C extends B {
// more code here
}


$someObj = new A(); // no problems
$someOtherObj = new B(); // no problems
$lastObj = new C(); // still no problems

?>
up
100
Mohammad Istanbouly
7 years ago
I think the best way for beginners to understand inheritance is through a real example so here is a simple example I can gave to you

<?php

class Person
{
public
$name;
protected
$age;
private
$phone;

public function
talk(){
//Do stuff here
}

protected function
walk(){
//Do stuff here
}

private function
swim(){
//Do stuff here
}
}

class
Tom extends Person
{
/*Since Tom class extends Person class this means
that class Tom is a child class and class person is
the parent class and child class will inherit all public
and protected members(properties and methods) from
the parent class*/

/*So class Tom will have these properties and methods*/

//public $name;
//protected $age;
//public function talk(){}
//protected function walk(){}

//but it will not inherit the private members
//this is all what Object inheritance means
}
up
19
akashwebdev at gmail dot com
9 years ago
The Idea that multiple inheritence is not supported is correct but with tratits this can be reviewed.

for e.g.

<?php
trait custom
{
public function
hello()
{
echo
"hello";
}
}

trait
custom2
{
public function
hello()
{
echo
"hello2";
}
}

class
inheritsCustom
{
use
custom, custom2
{
custom2::hello insteadof custom;
}
}

$obj = new inheritsCustom();
$obj->hello();
?>
up
15
jarrod at squarecrow dot com
15 years ago
You can force a class to be strictly an inheritable class by using the "abstract" keyword. When you define a class with abstract, any attempt to instantiate a separate instance of it will result in a fatal error. This is useful for situations like a base class where it would be inherited by multiple child classes yet you want to restrict the ability to instantiate it by itself.

Example........

<?php

abstract class Cheese
{
//can ONLY be inherited by another class
}

class
Cheddar extends Cheese
{
}

$dinner = new Cheese; //fatal error
$lunch = new Cheddar; //works!

?>
up
18
strata_ranger at hotmail dot com
14 years ago
I was recently extending a PEAR class when I encountered a situation where I wanted to call a constructor two levels up the class hierarchy, ignoring the immediate parent. In such a case, you need to explicitly reference the class name using the :: operator.

Fortunately, just like using the 'parent' keyword PHP correctly recognizes that you are calling the function from a protected context inside the object's class hierarchy.

E.g:

<?php
class foo
{
public function
something()
{
echo
__CLASS__; // foo
var_dump($this);
}
}

class
foo_bar extends foo
{
public function
something()
{
echo
__CLASS__; // foo_bar
var_dump($this);
}
}

class
foo_bar_baz extends foo_bar
{
public function
something()
{
echo
__CLASS__; // foo_bar_baz
var_dump($this);
}

public function
call()
{
echo
self::something(); // self
echo parent::something(); // parent
echo foo::something(); // grandparent
}
}

error_reporting(-1);

$obj = new foo_bar_baz();
$obj->call();

// Output similar to:
// foo_bar_baz
// object(foo_bar_baz)[1]
// foo_bar
// object(foo_bar_baz)[1]
// foo
// object(foo_bar_baz)[1]

?>
up
0
ignacio at inek dot com dot ar
18 hours ago
In case you have a public readonly property in a class you need to extend, adding other properties, this can be a way to do it:

<?php

class A {
public function
__construct(
public readonly
int $prop
) {}
}

class
B extends A {
public function
__construct(
int $prop,
public readonly
int $prop2
) {
parent::__construct($prop);
}
}
?>
To Top