Diferencia entre revisiones de «Plantilla:PHP/Sobrecarga»

De WikiEducator
Saltar a: navegación, buscar
(Página creada con «===Sobrecarga=== Un concepto muy importante y básico en la programación orientada a objetos. La sobrecarga es una concreción del principio de '''polimorfismo''' {{MMF_...»)
 
 
(32 revisiones intermedias por el mismo usuario no mostrado)
Línea 1: Línea 1:
===Sobrecarga===
+
<div id=parrafo>
Un concepto muy importante y básico en la programación orientada a objetos.
+
 
 +
=== Sobrecarga ===
 +
Es un concepto muy importante y básico en la programación orientada a objetos.
 +
La sobrecarga es una concreción del principio de '''polimorfismo'''.
 +
 
 +
{{MRM_Definicion|Title=Polimorfismo|
 +
* Podemos tener varios métodos con el mismo nombre, pero con diferente número de parámetros o con parámetros de distinto tipo. 
 +
* En tiempo de ejecución se ejecutará uno u otro en función de los parámetros reales que pasemos en la invocación del método.
 +
}}
 +
 
 +
* Sin embargo, este aspecto en PHP no es del todo intuitivo: no existe la sobrecarga como la entendemos en otros lenguajes. 
 +
* No obstante, disponemos de técnicas para poder ''simular'' la sobrecarga. 
 +
* En muchos casos resulta fundamental, especialmente al sobrecargar el '''constructor''' de una clase. 
 +
* Para simular la sobrecarga en '''PHP''', aprovechamos que una variable que no tenga valor se considera del tipo '''null'''. 
 +
* Lo veremos con una serie de ejemplos para dejar claro este concepto.
 +
 
 +
* Tomamos como ejemplo una función:
  
La sobrecarga es una concreción del principio de '''polimorfismo'''
 
{{MMF_Definicion|Title=polimorfismo|
 
*Podemos tener varios métodos con el mimos nombre, pero diferente número de parámetros o con parámetros de difernte tipo
 
*El tiempo de ejecución se ejecutará uno u otro en función de los parámetros reales que pasemos en la invocación del método}}
 
*Sin embargo este aspecto en php no es del todo intuitivo. No existe la sobrecarga como la entendemos en otros lenguajes.
 
*No obstante tenemos técnicas para poder simular la sobrecarga.
 
*Muchas veces es fundamental. Especialmente importante a la hora de sobrecargar el constructor de la clase.
 
*Vamos a ver como podemos implementar la sobrecarga en php
 
*Para simular la sobrecarga jugamos con el concepto de que una variable que no tenga valor se considera de tipo null
 
*A coninuación una serie de ejemplos para dejar claro este concepto
 
*Tomamos como ejemplo una función
 
 
{{MRM_Ejemplo|
 
{{MRM_Ejemplo|
 
<source lang=php>
 
<source lang=php>
function verTipoParametros($a,$b,$c){
+
function verTipoParametros($a = null, $b = null, $c = null){
     echo "Primer parámetro ";
+
     echo "Primer parámetro: ";
 
     var_dump($a);
 
     var_dump($a);
     echo "Segundo parámetro ";
+
     echo "Segundo parámetro: ";
 
     var_dump($b);
 
     var_dump($b);
     echo "Tercer parámetro ";
+
     echo "Tercer parámetro: ";
 
     var_dump($c);
 
     var_dump($c);
 
}
 
}
 
</source>
 
</source>
}}<br />
+
}}
 +
 
 +
<br />
 +
 
 +
* Ahora la invocamos de diferentes maneras y observamos el resultado:
  
*Ahora la invocamos de diferente manera y vemos el resultado
+
{{MRM_Ejemplo|Title=Invocar sin parámetros reales|
{{MRM_Ejemplo|Title=Invocar sin parámetros reales
+
 
<source lang=php>
 
<source lang=php>
echo "Invocando a <strong>verTipoParametros ()</strong><hr / />";
+
echo "Invocando a <strong>verTipoParametros()</strong><hr />";
verTipoParametros ();
+
verTipoParametros();
}
+
 
</source>
 
</source>
  
*A pesar de que tiene tres parámetros, la invacamos sin parámetros
+
* A pesar de que tiene tres parámetros, la invocamos sin argumentos. 
*El resultado será que cada parámetro al ejecutar la función será de tipo null con valor null (es un tipo válido en php
+
* El resultado será que cada parámetro, al ejecutar la función, será de tipo '''null''' con valor '''null''' (es un tipo válido en PHP).
[[Imagen:funcionSinParametrosReales.png]]
+
[[Archivo:funcionSinParametrosReales.png]]
 
}}
 
}}
  
{{MRM_Ejemplo|Title=Invocar con 1 parámetro real
+
{{MRM_Ejemplo|Title=Invocar con 1 parámetro real|
 
<source lang=php>
 
<source lang=php>
  ...
+
echo "Invocando a <strong>verTipoParametros(5)</strong><hr />";
  echo "Invocando a <strong>verTipoParametros (5)</strong><hr />";
+
verTipoParametros(5);
  verTipoParametros (5);
+
  ...
+
 
</source>
 
</source>
*En esta caso invocamos con un solo parámetro de tipo entero
+
* En este caso invocamos con un solo parámetro de tipo entero.
 
+
[[Archivo:funcion1ParametrosReal.png]]
*Podemos ver el resultado
+
[[Imagen:funcion1ParametrosReal.png]]
+
 
}}
 
}}
  
 
+
{{MRM_Ejemplo|Title=Invocar con 2 parámetros reales|
{{MRM_Ejemplo|Title=Invocar con 3 parámetros reales
+
 
<source lang=php>
 
<source lang=php>
  ...
+
echo "Invocando a <strong>verTipoParametros(5,7)</strong><hr />";
echo "Invocando a <strong>verTipoParametros (5,7)</strong><hr />";
+
verTipoParametros(5,7);
verTipoParametros (5,7);
+
  ...
+
 
</source>
 
</source>
*En esta caso invocamos con dos parámetros de tipo entero
+
* En este caso invocamos con dos parámetros de tipo entero
*Al igual que en caso anterior los parámetros en la función serían 3, dos de ellos con valor de tipo entero, y el tercero con valor y tipo null
+
* Los parámetros dentro de la función serán tres: dos con valor entero y el tercero con valor y tipo '''null'''.
*Podemos ver el resultado
+
[[Archivo:funcion2ParametrosReal.png]]
[[Imagen:funcion2ParametrosReal.png]]
+
 
}}
 
}}
{{MRM_Ejemplo|Title=Invocar con 3 parámetro reales uno de ellos un array
+
 
 +
{{MRM_Ejemplo|Title=Invocar con 3 parámetros reales|
 
<source lang=php>
 
<source lang=php>
  ...
+
echo "Invocando a <strong>verTipoParametros('pedro',5,9)</strong><hr />";
echo "Invocando a <strong>verTipoParametros ([1,4,'maria'],true,'sonia')</strong><hr />";
+
verTipoParametros('pedro',5,9);
verTipoParametros ([1,4,'maria'],true,'sonia');
+
  ...
+
 
</source>
 
</source>
*Ahora igualemnte pasamos tres parámetros, pero uno de ellos es un array
+
* En este caso pasamos tres parámetros: el primero de tipo string y los otros dos de tipo entero.
*Podemos ver el resultado
+
[[Archivo:funcion3ParametrosReal.png]]
[[Imagen:funcionParametroRealArray.png]]
+
 
}}
 
}}
  
 
+
{{MRM_Ejemplo|Title=Invocar con 3 parámetros reales, uno de ellos un '''array'''|
{{MRM_Ejemplo|Title=Invocar con 2 parámetro real
+
 
<source lang=php>
 
<source lang=php>
  ...
+
echo "Invocando a <strong>verTipoParametros([1,4,'maría'], true, 'sonia')</strong><hr />";
echo "Invocando a <strong>verTipoParametros ('pedro',5,9)</strong><hr />";
+
verTipoParametros([1,4,'maría'], true, 'sonia');
verTipoParametros ('pedro',5,9);
+
  ...
+
 
</source>
 
</source>
*En esta caso invocamos con tres parámetros de diferente tipo
+
* Ahora igualmente pasamos tres parámetros, pero uno de ellos es un array.
{{Tipo|PHP lenguaje de tipado dinámico, vemos como el tipo puede cambiar en cada invocación}}
+
[[Archivo:funcionParametroRealArray.png]]
*Ahora ya tenemos tres valores diferentes a null en los parámetros reales
+
*Podemos ver el resultado
+
[[Imagen:funcion3ParametrosReal.png]]
+
 
}}
 
}}
  
 +
==== Sobrecargando el constructor ====
 +
* Usando esta forma de trabajar, vamos a ''sobrecargar'' el constructor de una clase. 
 +
* Tomemos como ejemplo una clase '''Racional'''. Un número racional es un objeto que tiene numerador y denominador.
 +
[[Archivo:claseRacional.png]]
  
 +
* Queremos permitir crear el objeto de distintas formas:
 +
<source lang=php>
 +
$r1 = new Racional("8/5"); /* 8/5 */
 +
$r2 = new Racional(5,4);  /* 5/4 */
 +
$r3 = new Racional(5);    /* 5/1 */
 +
$r4 = new Racional();      /* 1/1 */
 +
</source>
  
 +
* Necesitamos que el constructor pueda responder a todas las situaciones. 
 +
* Aplicando los conceptos vistos, solo tenemos que comprobar '''de qué tipo''' son los parámetros.
 +
{{Tip|Recordar que ''null'' también es un tipo.}}
  
 +
* Vemos que podemos tener 0, 1 o 2 parámetros. 
 +
Por lo tanto, el constructor tendrá dos parámetros opcionales.
  
 
 
*Supongamos que en el ejemplo anterior quisiérmos poder instanciar de la siguiente manera los racionales
 
 
<source lang=php>
 
<source lang=php>
$a = new racional ("8/5");/*  8/5  */
+
public function __construct($num = 1, $den = 1) {
$b = new racional (5,4);  /*  5/6  */
+
    ...
$c = new racional (5);    /*  5/1  */
+
$d = new racional ();    /*  1/1  */
+
</source>
+
*Para esta situación tenemos diferentes estrategias. En este tema vamos a analizar 2:
+
#sobrecargar en el constructor
+
#sobrecargar con '''''métodos mágicos'''''
+
;Sobrecargar en el constructor
+
*Una sencila es ir viendo de qué tipo son los parámetros
+
*Vemos que podemos tener 0, 1 o 2 paŕametros
+
*Por lo tanto la función constructora tendrá que tener 3 parámetros inicializados por si no le pasamos valores
+
<source lang=php>
+
public function __construct($cadenaOrNum=null, $den = null) {
+
....
+
 
}
 
}
 
</source>
 
</source>
{{MRM_Actividad|Title=Sobrecargar el constructor de un racional|
 
Siguiendo el ejemplo establecido anteriormente realiza un constructor que permita instanciar un objeto de la clase racional de la siguiente manera
 
<source lang=php>
 
$a = new racional ("8/5");/*  8/5  */
 
$b = new racional (5,4);  /*  5/6  */
 
$c = new racional (5);    /*  5/1  */
 
$d = new racional ();    /*  1/1  */
 
</source>
 
}}
 
<!--
 
  
Vamos a simular la sobre carga de la clase racional
+
* Especificamos el código de cómo se podría implementar:
  
 
<source lang=php>
 
<source lang=php>
<!--
+
class Racional {
<?php
+
 
/**
+
* Description of racional
+
*
+
* @author manolo
+
*/
+
class racional {
+
    //put your code here
+
    private $numRacional;
+
 
     private $num;
 
     private $num;
 
     private $den;
 
     private $den;
     public function __construct($cadenaOrNum=null, $den = null) {
+
 
         if (is_int($cadenaOrNum)){
+
     public function __construct($num = 1, $den = 1) {
            if (isset($den))
+
         // opciones:
                $this->racionalNumDen($cadenaOrNum, $den );
+
        // new Racional()      => 1/1
             else
+
        // new Racional(5)     => 5/1
                $this->racionalNum($cadenaOrNum );
+
        // new Racional("5/2") => 5/2
 +
        // new Racional(5,2)    => 5/2
 +
        // Otra situación: no se instancia correctamente
 +
 
 +
        if (is_string($num)) {
 +
            $numero = explode("/", $num);
 +
             $num = $numero[0];
 +
            $den = $numero[1];
 
         }
 
         }
         else{
+
         $this->num = $num;  
            if (isset($cadenaOrNum))
+
        $this->den = $den;
                $this->racionalCadena($cadenaOrNum);
+
    }
            else
+
                $this->racionalVacio();
+
        }
+
    }
+
   
+
    private function racionalNumDen($num, $den){
+
        $this->numRacional=$num."/".$den;
+
        $this->num=$num;
+
        $this->den=$den;
+
        $this->convierteNum();
+
    }
+
    private function racionalNum($num){
+
        $this->numRacional=$num."/1";
+
        $this->num=$num;
+
        $this->den=1;
+
        $this->convierteNum();
+
    }
+
    private function racionalCadena($cadenaOrNum){
+
        $this->numRacional=$cadenaOrNum;
+
        $pos = strpos($cadenaOrNum,"/");
+
       
+
        $num=substr($cadenaOrNum,0,$pos);
+
        $den=substr($cadenaOrNum,$pos+1);
+
        $this->num=$num;
+
        $this->den=$den;
+
    }
+
    private function racionalVacio(){
+
        $this->numRacional="1/1";
+
        $this->num=1;
+
        $this->den=1;
+
    }
+
    private function convierteNum(){
+
        settype($this->num, int);
+
        settype($this->den, int);
+
    }
+
   
+
       
+
  
     public function visualiza(){
+
    /* Método para visualizar el objeto como cadena de caracteres */
         return $this->numRacional;
+
     public function __toString() {
 +
         return ($this->num . "/" . $this->den);
 
     }
 
     }
     public function valorRacional(){
+
 
         return $this->num/$this->den;
+
    // Métodos privados para asignar valores según la forma de invocación
 +
     private function racionalNum($num) {
 +
         $this->num = $num;
 +
        $this->den = 1;
 
     }
 
     }
     private function simplifica(){
+
 
          
+
     /**
 +
    * @param string $num número racional del tipo "a/b"
 +
    * Hay muchas formas de poder descomponer esa cadena en dos números.
 +
    */
 +
    public function racionalCadena($num) {
 +
         $partes = explode("/", $num);
 +
        $this->num = (int) $partes[0];
 +
        $this->den = (int) $partes[1];
 
     }
 
     }
     private function suma(racional $num){
+
 
          
+
     public function racionalVacio() {
 +
         $this->num = 1;
 +
        $this->den = 1;
 
     }
 
     }
  
 +
    /* En este caso, si los valores son incorrectos, asigno el racional 1/1 */
 +
    public function racionalNumDen($num, $den) {
 +
        if (is_numeric($num) && is_numeric($den)) {
 +
            $this->num = $num;
 +
            $this->den = $den;
 +
        } else {
 +
            $this->num = 1;
 +
            $this->den = 1;
 +
        }
 +
    }
 
}
 
}
 
$a=new racional("8/5");
 
$b=new racional("6/4");
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
echo '<br>racional de $b: '. $b->visualiza();
 
echo '<br>valor de $b: '. $b->valorRacional();
 
echo"<hr/>";
 
$a=new racional();
 
$b=new racional();
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
echo '<br>racional de $b: '. $b->visualiza();
 
echo '<br>valor de $b: '. $b->valorRacional();
 
echo"<hr/>";
 
$a=new racional(8);
 
$b=new racional(4);
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
echo '<br>racional de $b: '. $b->visualiza();
 
echo '<br>valor de $b: '. $b->valorRacional();
 
echo"<hr/>";
 
$a=new racional(124,6);
 
$b=new racional(7,123);
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
echo '<br>racional de $b: '. $b->visualiza();
 
echo '<br>valor de $b: '. $b->valorRacional();
 
 
?>
 
 
 
</source>
 
</source>
  
}}
+
;Esta sería una posibilidad, pero podemos crear un código más compacto. 
 +
En la siguiente versión inicializamos los parámetros con un valor por defecto (<span class="r">1</span>). 
 +
Posteriormente, si el primer parámetro es una cadena, se divide en numerador y denominador. 
 +
Hecho esto, solo queda asignar los valores a los atributos.
  
*Código implementado en clase
 
<!--
 
 
<source lang=php>
 
<source lang=php>
<?php
+
public function __construct($num = 1, $den = 1){
 
+
     if (is_string($num)){
/**
+
         $numero = explode("/", $num);
* Description of racional
+
         $num = $numero[0];
*
+
         $den = $numero[1];
* @author alumno
+
*/
+
class racional {
+
    //put your code here
+
    private $num;
+
    private $den;
+
    private $numRacional;
+
    public function __construct($cadenaOrNum=null, $den = null) {
+
        if (is_null($cadenaOrNum))
+
          $this->racionalVacio();   
+
        else
+
          if (!(is_null($den)))
+
              $this->racionalNumDen($cadenaOrNum, $den );
+
          else
+
              if (is_int($cadenaOrNum))
+
                  $this->racionalNum($cadenaOrNum);
+
              else
+
                  $this->racionalCadena($cadenaOrNum);
+
     }
+
   
+
    //Sobrecargamos el método asigna de modo que se pueda implementar de las siguientes maneras
+
    /*asigna("6/4");
+
      asigna();
+
      asigna(8);
+
      asigna(124, 6);
+
    * */
+
 
+
    public function __call($metodo, $argumentos) {
+
        if ($metodo = "asigna"){
+
            switch (count($argumentos)){
+
                case 0:
+
                    $this->racionalVacio();
+
                    break;
+
                case 2:
+
                    $this->racionalNumDen($argumentos[0], $argumentos[1]);
+
                    break;
+
                case 1:
+
                    if (is_int($argumentos[0]))
+
                        $this->racionalNum($argumentos[0]);
+
                    else
+
                        $this->racionalCadena($argumentos[0]);
+
                    break;
+
            }
+
           
+
           
+
           
+
           
+
        }
+
       
+
    }
+
    // caso $a = racional(8,7);
+
    private function racionalNumDen($num, $den ){
+
        $this->num=$num;
+
        $this->den=$den;
+
        $this->numRacional="$num/$den";
+
       
+
    }
+
// caso $a = racional(8);
+
    private function racionalNum($num ){
+
         $this->num=$num;
+
        $this->den=1;
+
        $this->numRacional="$num/1";
+
       
+
    }
+
// caso $a = racional("8/7");
+
    private function racionalCadena($cadena){
+
        $pos = strpos($cadena, "/");
+
        $num = substr($cadena,0, $pos);
+
         $den = substr($cadena ,$pos+1 );
+
       
+
        settype($num, "int");
+
        settype($den, "int");
+
       
+
        $this->num= $num;
+
         $this->den= $den;
+
        $this->numRacional=$cadena;
+
    }
+
// caso $a = racional(); y se inicia con 1/1
+
    private function racionalVacio(){
+
        $this->num=1;
+
        $this->den=1;
+
        $this->numRacional="1/1";
+
       
+
       
+
    }
+
   
+
    public function visualiza(){
+
        echo "<br/>valor del numerador ".$this->num;
+
        echo "<br/>valor del denominador ".$this->den;
+
        echo "<br/>valor del racional ".$this->numRacional;
+
        echo "<br/>valor numerico ".($this->num/$this->den);
+
 
     }
 
     }
 +
    $this->num = $num;
 +
    $this->den = $den;
 
}
 
}
 +
</source>
  
 +
* Probamos este constructor con el siguiente código:
  
 +
<source lang=php>
 +
$a = new Racional();
 +
$b = new Racional(5);
 +
$c = new Racional(5,6);
 +
$d = new Racional("6/6");
  
//-------------------------
+
echo "Valor del racional \$a = $a <br />";
 +
echo "Valor del racional \$b = $b <br />";
 +
echo "Valor del racional \$c = $c <br />";
 +
echo "Valor del racional \$d = $d <br />";
 +
</source>
  
 +
* Mostrando los siguientes resultados:
  
$a= new racional();
+
Valor del racional $a <nowiki>=</nowiki> 1/1 
$b= new racional(5);
+
Valor del racional $b <nowiki>=</nowiki> 5/1 
$c= new racional(5,6);
+
Valor del racional $c <nowiki>=</nowiki> 5/6
$d= new racional("6/6");
+
Valor del racional $d <nowiki>=</nowiki> 6/6
  
$a->visualiza();
+
==== Sobrecarga con __call(...) ====
$b->visualiza();
+
* Otra forma de lograr un comportamiento similar es usando el método mágico '''__call($funcion, $parametros)'''.
$c->visualiza();
+
{{Tip|El método mágico __call(...) se ejecuta cuando invocamos un método que no existe en la clase.}}
$d->visualiza();
+
  
$a->asigna(6000,545);
+
* Queremos usar un método llamado '''asigna()''' que nos permita cambiar el valor de un racional. 
$a->visualiza();
+
* La forma de aportar el nuevo valor será la misma que la de construir el objeto.
$a->asigna("8458/55154");
+
$a->visualiza();
+
$a->asigna(85547);
+
$a->visualiza();
+
$a->asigna();
+
$a->visualiza();
+
?>
+
  
 
 
</source>
 
-->
 
;métodos mágicos y la sobrecarga
 
*Hemos visto __construct y __destruct
 
*Vamos a analizar un método mágico para implementar la sobrecarga y exponer su funcionamiento
 
*Ël método '''''__call($nombreFuncion, $argumentos)'''' es un método mágico que se invoca cuando intentamos acceder a un método de la clase y este método no existe
 
*Volvamos a nuestro racional
 
*Supongamos que los métodos de sobrecarga que queremos son los usados en el constructor __construct
 
 
<source lang=php>
 
<source lang=php>
// caso $a = racional(8,7);
+
$r1 = new Racional();     // construye el objeto 1/1
racionalNumDen($cadenaOrNum, $den );
+
$r1->asigna("6/4");       // ahora el objeto vale 6/4
// caso $a = racional(8);
+
$r1->asigna();             // ahora el objeto vale 1/1
racionalNum($cadenaOrNum );
+
$r1->asigna(8);           // ahora el objeto vale 8/1
// caso $a = racional("8/7");
+
$r1->asigna(124, 6);       // ahora el objeto vale 124/6
racionalCadena($cadenaOrNum);
+
// caso $a = racional(); y se inicia con 1/1
+
racionalVacio();
+
 
</source>
 
</source>
*Pero imagenimos que nos gustaría que todos tuvieran el mismo nombre, por ejemplo crea().
 
*Vamos a usar el método __call para ello
 
*Este método recibe dos argumentos
 
# $nombreFuncion es el nombre del método que se invoca
 
# $argumentos es un array con los argumentos de la función
 
{{MRM_Actividad| vamos a sobrecargar el método asigna() de la clase racional de modo que accepte sobrecarga
 
}}
 
<!--
 
<source lang=php>
 
  
$a=new racional();
+
* La forma de proceder será usando los métodos privados creados anteriormente. 
 +
* El siguiente código implementa la solución:
  
$a->asigna("6/4");
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
 
echo"<hr/>";
 
$a->asigna();
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
 
echo"<hr/>";
 
$a->asigna(8);
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
 
echo"<hr/>";
 
$a->asigna(124, 6);
 
echo '<br>racional de $a: '. $a->visualiza();
 
echo '<br>valor de $a: '. $a->valorRacional();
 
</source>
 
<!--
 
 
<source lang=php>
 
<source lang=php>
<?php
+
public function __call($metodo, $argumentos) {
 
+
    if ($metodo == "asigna") {
/*
+
        switch (count($argumentos)) {
* To change this license header, choose License Headers in Project Properties.
+
            case 0:
* To change this template file, choose Tools | Templates
+
                $this->racionalVacio();
* and open the template in the editor.
+
                break;
*/
+
            case 2:
 
+
                 $this->racionalNumDen($argumentos[0], $argumentos[1]);
/**
+
                break;
* Description of racional
+
            case 1:
*
+
                if (is_int($argumentos[0])) {
* @author manolo
+
                    $this->racionalNum($argumentos[0]);
*/
+
                } else {
class racional {
+
                    $this->racionalCadena($argumentos[0]);
    //put your code here
+
                }
    private $numRacional;
+
                 break;
    private $num;
+
    private $den;
+
    public function __construct($argumento) {
+
       
+
    }
+
    public function __call($name, $argumentos) {
+
        if ($name=="asigna"){
+
            switch (count($argumentos)){
+
                case 0:
+
                    $this->racionalVacio();
+
                    break;
+
                 case 1: //Aquí tenemos dos casos hay que ver el tipo
+
                    if (is_int($argumentos[0]))
+
                        $this->racionalNum($argumentos[0]);
+
                    else
+
                        $this->racionalCadena($argumentos[0]);
+
                   
+
                    break;
+
                 case 2:
+
                    $this->racionalNumDen($argumentos[0],$argumentos[1]);
+
                    break;
+
            }
+
           
+
 
         }
 
         }
   
 
    }
 
   
 
    private function racionalNumDen($num, $den){
 
        $this->numRacional=$num."/".$den;
 
        $this->num=$num;
 
        $this->den=$den;
 
        $this->convierteNum();
 
    }
 
    private function racionalNum($num){
 
        $this->numRacional=$num."/1";
 
        $this->num=$num;
 
        $this->den=1;
 
        $this->convierteNum();
 
    }
 
    private function racionalCadena($cadenaOrNum){
 
        $this->numRacional=$cadenaOrNum;
 
        $pos = strpos($cadenaOrNum,"/");
 
       
 
        $num=substr($cadenaOrNum,0,$pos);
 
        $den=substr($cadenaOrNum,$pos+1);
 
        $this->num=$num;
 
        $this->den=$den;
 
    }
 
    private function racionalVacio(){
 
        $this->numRacional="1/1";
 
        $this->num=1;
 
        $this->den=1;
 
    }
 
    private function convierteNum(){
 
        settype($this->num, int);
 
        settype($this->den, int);
 
    }
 
   
 
       
 
 
    public function visualiza(){
 
        return $this->numRacional;
 
 
     }
 
     }
    public function valorRacional(){
 
        return $this->num/$this->den;
 
    }
 
    private function simplifica(){
 
       
 
    }
 
    private function suma(racional $num){
 
       
 
    }
 
 
 
}
 
}
 +
</source>
  
$a=new racional();
+
</div>
$b=new racional();
+
$a->asigna("8/5");
+
$b->asigna("6/4");
+
echo '<br>racional de $a: '. $a->visualiza();
+
echo '<br>valor de $a: '. $a->valorRacional();
+
echo '<br>racional de $b: '. $b->visualiza();
+
echo '<br>valor de $b: '. $b->valorRacional();
+
echo"<hr/>";
+
$a->asigna();
+
$a->asigna();
+
echo '<br>racional de $a: '. $a->visualiza();
+
echo '<br>valor de $a: '. $a->valorRacional();
+
echo '<br>racional de $b: '. $b->visualiza();
+
echo '<br>valor de $b: '. $b->valorRacional();
+
echo"<hr/>";
+
$a->asigna(8);
+
$b->asigna(4);
+
echo '<br>racional de $a: '. $a->visualiza();
+
echo '<br>valor de $a: '. $a->valorRacional();
+
echo '<br>racional de $b: '. $b->visualiza();
+
echo '<br>valor de $b: '. $b->valorRacional();
+
echo"<hr/>";
+
$a->asigna(124, 6);
+
$b->asigna(124, 6);
+
echo '<br>racional de $a: '. $a->visualiza();
+
echo '<br>valor de $a: '. $a->valorRacional();
+
echo '<br>racional de $b: '. $b->visualiza();
+
echo '<br>valor de $b: '. $b->valorRacional();
+
 
+
?>
+
 
+
</source>
+
}}
+
-->
+
==::==
+

Última revisión de 05:47 7 nov 2025

Sobrecarga

Es un concepto muy importante y básico en la programación orientada a objetos. La sobrecarga es una concreción del principio de polimorfismo.



Icon define.gif
Polimorfismo
  • Podemos tener varios métodos con el mismo nombre, pero con diferente número de parámetros o con parámetros de distinto tipo.
  • En tiempo de ejecución se ejecutará uno u otro en función de los parámetros reales que pasemos en la invocación del método.


  • Sin embargo, este aspecto en PHP no es del todo intuitivo: no existe la sobrecarga como la entendemos en otros lenguajes.
  • No obstante, disponemos de técnicas para poder simular la sobrecarga.
  • En muchos casos resulta fundamental, especialmente al sobrecargar el constructor de una clase.
  • Para simular la sobrecarga en PHP, aprovechamos que una variable que no tenga valor se considera del tipo null.
  • Lo veremos con una serie de ejemplos para dejar claro este concepto.
  • Tomamos como ejemplo una función:



Icon casestudy.gif
Ejemplo
function verTipoParametros($a = null, $b = null, $c = null){
    echo "Primer parámetro: ";
    var_dump($a);
    echo "Segundo parámetro: ";
    var_dump($b);
    echo "Tercer parámetro: ";
    var_dump($c);
}




  • Ahora la invocamos de diferentes maneras y observamos el resultado:



Icon casestudy.gif
Invocar sin parámetros reales
echo "Invocando a <strong>verTipoParametros()</strong><hr />";
verTipoParametros();
  • A pesar de que tiene tres parámetros, la invocamos sin argumentos.
  • El resultado será que cada parámetro, al ejecutar la función, será de tipo null con valor null (es un tipo válido en PHP).

FuncionSinParametrosReales.png




Icon casestudy.gif
Invocar con 1 parámetro real
echo "Invocando a <strong>verTipoParametros(5)</strong><hr />";
verTipoParametros(5);
  • En este caso invocamos con un solo parámetro de tipo entero.

Funcion1ParametrosReal.png




Icon casestudy.gif
Invocar con 2 parámetros reales
echo "Invocando a <strong>verTipoParametros(5,7)</strong><hr />";
verTipoParametros(5,7);
  • En este caso invocamos con dos parámetros de tipo entero.
  • Los parámetros dentro de la función serán tres: dos con valor entero y el tercero con valor y tipo null.

Funcion2ParametrosReal.png




Icon casestudy.gif
Invocar con 3 parámetros reales
echo "Invocando a <strong>verTipoParametros('pedro',5,9)</strong><hr />";
verTipoParametros('pedro',5,9);
  • En este caso pasamos tres parámetros: el primero de tipo string y los otros dos de tipo entero.

Funcion3ParametrosReal.png




Icon casestudy.gif
Invocar con 3 parámetros reales, uno de ellos un array
echo "Invocando a <strong>verTipoParametros([1,4,'maría'], true, 'sonia')</strong><hr />";
verTipoParametros([1,4,'maría'], true, 'sonia');
  • Ahora igualmente pasamos tres parámetros, pero uno de ellos es un array.

FuncionParametroRealArray.png



Sobrecargando el constructor

  • Usando esta forma de trabajar, vamos a sobrecargar el constructor de una clase.
  • Tomemos como ejemplo una clase Racional. Un número racional es un objeto que tiene numerador y denominador.

ClaseRacional.png

  • Queremos permitir crear el objeto de distintas formas:
$r1 = new Racional("8/5"); /* 8/5 */
$r2 = new Racional(5,4);   /* 5/4 */
$r3 = new Racional(5);     /* 5/1 */
$r4 = new Racional();      /* 1/1 */
  • Necesitamos que el constructor pueda responder a todas las situaciones.
  • Aplicando los conceptos vistos, solo tenemos que comprobar de qué tipo son los parámetros.
Icon present.gif
Tip: Recordar que null también es un tipo.


  • Vemos que podemos tener 0, 1 o 2 parámetros.

Por lo tanto, el constructor tendrá dos parámetros opcionales.

public function __construct($num = 1, $den = 1) {
    ...
}
  • Especificamos el código de cómo se podría implementar:
class Racional {
 
    private $num;
    private $den;
 
    public function __construct($num = 1, $den = 1) {
        // opciones:
        // new Racional()       => 1/1
        // new Racional(5)      => 5/1
        // new Racional("5/2")  => 5/2
        // new Racional(5,2)    => 5/2
        // Otra situación: no se instancia correctamente
 
        if (is_string($num)) {
            $numero = explode("/", $num);
            $num = $numero[0];
            $den = $numero[1];
        }
        $this->num = $num; 
        $this->den = $den;
    }
 
    /* Método para visualizar el objeto como cadena de caracteres */
    public function __toString() {
        return ($this->num . "/" . $this->den);
    }
 
    // Métodos privados para asignar valores según la forma de invocación
    private function racionalNum($num) {
        $this->num = $num;
        $this->den = 1;
    }
 
    /**
     * @param string $num número racional del tipo "a/b"
     * Hay muchas formas de poder descomponer esa cadena en dos números.
     */
    public function racionalCadena($num) {
        $partes = explode("/", $num);
        $this->num = (int) $partes[0];
        $this->den = (int) $partes[1];
    }
 
    public function racionalVacio() {
        $this->num = 1;
        $this->den = 1;
    }
 
    /* En este caso, si los valores son incorrectos, asigno el racional 1/1 */
    public function racionalNumDen($num, $den) {
        if (is_numeric($num) && is_numeric($den)) {
            $this->num = $num;
            $this->den = $den;
        } else {
            $this->num = 1;
            $this->den = 1;
        }
    }
}
Esta sería una posibilidad, pero podemos crear un código más compacto.

En la siguiente versión inicializamos los parámetros con un valor por defecto (1). Posteriormente, si el primer parámetro es una cadena, se divide en numerador y denominador. Hecho esto, solo queda asignar los valores a los atributos.

public function __construct($num = 1, $den = 1){
    if (is_string($num)){
        $numero = explode("/", $num);
        $num = $numero[0];
        $den = $numero[1];
    }
    $this->num = $num;
    $this->den = $den;
}
  • Probamos este constructor con el siguiente código:
$a = new Racional();
$b = new Racional(5);
$c = new Racional(5,6);
$d = new Racional("6/6");
 
echo "Valor del racional \$a = $a <br />";
echo "Valor del racional \$b = $b <br />";
echo "Valor del racional \$c = $c <br />";
echo "Valor del racional \$d = $d <br />";
  • Mostrando los siguientes resultados:
Valor del racional $a = 1/1  
Valor del racional $b = 5/1  
Valor del racional $c = 5/6  
Valor del racional $d = 6/6  

Sobrecarga con __call(...)

  • Otra forma de lograr un comportamiento similar es usando el método mágico __call($funcion, $parametros).
Icon present.gif
Tip: El método mágico __call(...) se ejecuta cuando invocamos un método que no existe en la clase.


  • Queremos usar un método llamado asigna() que nos permita cambiar el valor de un racional.
  • La forma de aportar el nuevo valor será la misma que la de construir el objeto.
$r1 = new Racional();      // construye el objeto 1/1
$r1->asigna("6/4");        // ahora el objeto vale 6/4
$r1->asigna();             // ahora el objeto vale 1/1
$r1->asigna(8);            // ahora el objeto vale 8/1
$r1->asigna(124, 6);       // ahora el objeto vale 124/6
  • La forma de proceder será usando los métodos privados creados anteriormente.
  • El siguiente código implementa la solución:
public function __call($metodo, $argumentos) {
    if ($metodo == "asigna") {
        switch (count($argumentos)) {
            case 0:
                $this->racionalVacio();
                break;
            case 2:
                $this->racionalNumDen($argumentos[0], $argumentos[1]);
                break;
            case 1:
                if (is_int($argumentos[0])) {
                    $this->racionalNum($argumentos[0]);
                } else {
                    $this->racionalCadena($argumentos[0]);
                }
                break;
        }
    }
}