| 
        php.net | support | documentation | report a bug | advanced search | search howto | statistics | random bug | login | 
  [2007-05-21 23:06 UTC] ralph at smashlabs dot com
 Description:
------------
When an interface is a parent of a class, the interface is implying a specific profile for a given function (here its __get) even though it is not defined in the interface itself.  This is making it impossible to overload and/or change the profile of the function in a descendant class.
Reproduce code:
---------------
The following will produce this notice:
Strict Standards: Declaration of Z_Concrete::__get() should be compatible with that of Z_Abstract::__get() in xxx on line 16
<?php
error_reporting(E_ALL | E_STRICT);
interface Z_Interface
{ }
abstract class Z_Abstract implements Z_Interface
{
  public function __get($name)
  {
    return;
  }
}
class Z_Concrete extends Z_Abstract
{
  public function & __get($name)
  {
    return null;
  }
}
$t = new Z_Concrete();
This code will not produce a notice:
<?php
error_reporting(E_ALL | E_STRICT);
abstract class Z_Abstract
{
  public function __get($name)
  {
    return;
  }
}
class Z_Concrete extends Z_Abstract
{
  public function & __get($name)
  {
    return null;
  }
}
$t = new Z_Concrete();
The only difference is that the former has an interface as a parent, the latter does not.
Expected result:
----------------
No E_STRICT notice raised
Actual result:
--------------
E_STRICT notice raised.
PatchesPull RequestsHistoryAllCommentsChangesGit/SVN commits             
             | 
    |||||||||||||||||||||||||||||||||||||
            
                 
                Copyright © 2001-2025 The PHP GroupAll rights reserved.  | 
        Last updated: Tue Nov 04 14:00:01 2025 UTC | 
I am sorry, perhaps I should rephrase (I was rushed when I wrote the original bug report) and I don't think I was especially clear. In general, we are speaking strictly of Method Overriding. PHP currently allows concrete classes to override methods in abstract classes as demonstrated by this code: webdeveloper@webdevelopment ~/tmp $ cat test_method_override.php <?php error_reporting(E_ALL | E_STRICT); class Z_Abstract { public function someFunc($one, $two) { } } class Z_Concrete extends Z_Abstract { public function & someFunc($uno, $dos, $tres) { } } $o = new Z_Concrete(); webdeveloper@webdevelopment ~/tmp $ php test_method_override.php webdeveloper@webdevelopment ~/tmp $ As you can see, no E_STRICT was triggered, thus allowing the concrete class to override the method from the abstract class. On the other hand, when adding an _empty_ interface to the superclass hierarchy, the behavior changes. The simple inclusion of an interface now (albeit empty) seemingly shouldn't add any complexities to method overriding for methods outside the scope of the interface itself. I have done a bit more homework, and it seems like not alot of IS_A inheritance literature exists to describe this problem, let alone justify the behavior we are seeing in practice.. See code: webdeveloper@webdevelopment ~/tmp $ cat test_method_override2.php <?php error_reporting(E_ALL | E_STRICT); interface Z_Interface { } class Z_Abstract implements Z_Interface { public function someFunc($one, $two) { } } class Z_Concrete extends Z_Abstract { public function & someFunc($uno, $dos, $tres) { } } $o = new Z_Concrete(); webdeveloper@webdevelopment ~/tmp $ php test_method_override2.php Strict standards: Declaration of Z_Concrete::someFunc() should be compatible with that of Z_Abstract::someFunc() in /home/webdeveloper/tmp/test_method_override2.php on line 14 At the very least, I looks like one of the two is broken (At least in the spirit of OOP as it relates to polymorphism and overriding methods.)Should this not be possible without errors or notices? I am overloading an overridden method. If memory serves me, I am able to do this in other OO languages (Java, .NET/C#). class BaseClass { public function TestFunction ($parameter) { var_dump($parameter); } } class FooClass extends BaseClass { public function TestFunction () { parent::TestFunction('This is Foo'); } } class BarClass extends BaseClass { public function TestFunction () { parent::TestFunction('This is Bar'); } } $test = new FooClass(); $test->TestFunction(); $test = new BarClass(); $test->TestFunction(); Instead: Strict Standards: Declaration of FooClass::TestFunction() should be compatible with that of BaseClass::TestFunction() Strict Standards: Declaration of BarClass::TestFunction() should be compatible with that of BaseClass::TestFunction()