views:

85

answers:

8

Here's the objective. I have a PHP class and there are one or two of its methods that I would like to override with my own. As I understand OOP (in PHP and in general) I could write a child class that extends it and overrides the functionality of the methods in question.

However, I was wondering if this is the best way of achieving this task and if this is a proper use for child classes or if there is something better in PHP for what I'm trying to do.

+2  A: 

Yes, that is the best way to do it. The idea of extending a class is to provide extra or more specific functionality.

icio
+1  A: 

yes. this exactly what you use inheritance for.

A good principle though, is make sure your new inheriting class "is-a" base class.

Like Human is-a Mammal. Don't do Alien extends Human just because Alien does a lot of stuff Humans do.

seanmonstar
Yeah well pointed out. I remember years ago, when I first started OOP in php, getting myself into a bit of a mess by doing exactly what you describe.These days I find that, even on a complex project, my classes are all pretty short and pretty simple. The complexity comes then from how you combine the classes. If your brain is being fried trying to write some massive method cos you need to deal with lots of cases or variations, then you probably wanna review how your classes are structured
Addsy
+1  A: 

So, are you asking if overriding methods (which requires extending the superclass) is the best way to override methods? Yes.

That is the most important use of subclasses. In fact, polymorphism is at the heart of OOP. The other use is to provide new properties/methods, but that usually won't be enough.

Artefacto
A: 

Sounds like you're on the right track.

For the methods you want to override, you'd probably want to:

  • make sure you're not changing the intent of the method
  • know whether you have to call the base class's method within, and when
John at CashCommons
A: 

The only other way, would be to create a generic "superclass"...

class SuperClass {
    protected $obj = null;
    protected $overrides = array();

    public function __construct($obj) {
        if (!is_object($obj)) {
            throw new InvalidArgumentException('Argument is not an object');
        }
        $this->obj = $obj;
    }

    public function __call($method, $args) {
        $method = strtolower($method);
        if (isset($this->overrides[$method])) {
            array_unshift($args, $this);
            return call_user_func_array($this->overrides[$method], $args);
        } elseif (is_callable(array($this->obj, $method))) {
            return call_user_func_array(array($this->obj, $method), $args);
        } else {
            throw new BadMethodCallException('Invalid Method Called');
        }
    }

    public function __get($var) {
        return isset($this->obj->$var) ? $this->obj->$var : null;
    }

    public function __set($var, $value) {
        $this->obj->$var = $value;
    }

    public function addOverride($method, $callback) {
        $this->overrides[strtolower($method)] = $callback;
    }
}

It's not always the best solution, but it's possible that some situations exist to use something like that. It will let you "add" and "override" methods to any object at run time.

The better generic solution is to simply extend the class in a child class... But the above "superclass" does have some uses...

ircmaxell
+1  A: 
class oldClass {
  public function methodToOverride() {
    echo 'oh hai';
  }
}

class childClass extends oldClass {
  public function methodToOverride($arg, $arg2) {
    // your custom code
    echo 'hai world ' . $arg . ' ' . $arg2;

    // if you need to still call the parent class
    parent::methodToOverride();
  }
}

$child = new childClass();
$child->methodToOverride('nom', 'nom');
cballou
harsh -1 for a correct answer. nom noms were for effect.
cballou
Just so you know, that wasn't me man and that goes for anyone else who took the time to answer a n00b's (such as myself) question: I appreciate anyone's efforts to read my question and give me an answer and I would never give anyone a thumbs down or -1.
A: 

Another route I haven't seen mentioned here is to question whether it wouldn't be better to extract an abstract class and have the two classes extend that. Of course you would need to be able to alter the code of the first class for that (e.i. if the class came from an open source library you might refrain from changing the code).

koen
A: 

you are on the right track

ovais.tariq