PHP Matrices
PHP Matrices
Es bastante común que algunas fracciones decimales simples como 0.1 o 0.7 no puedan ser
convertidas a su representación binaria interna sin perder un poco de precisión. Esto puede llevar a
resultados confusos: por ejemplo, floor((0.1+0.7)*10) usualmente devolverá 7 en lugar del
esperado 8 ya que el resultado de la representación interna es en realidad algo como
7.9999999999....
Esto se encuentra relacionado al hecho de que es imposible expresar de forma exacta algunas
fracciones en notación decimal con un número finito de dígitos. Por ejemplo, 1/3 en forma
decimal se convierte en 0.3333333. . ..
Así que nunca confíe en resultados de números flotantes hasta el último dígito, y nunca compare
números de punto flotante para conocer si son ñalentes. Si realmente necesita una mejor precisión,
es buena idea que use las funciones matemáticas de precisión arbitraria o las funciones gmp en su
lugar.
Conversión a flotante
Para más información sobre cuándo y cómo son convertidas las cadenas a flotantes, vea la sección
titulada Conversión de cadenas a números. Para valores de otros tipos, la conversión es la misma
que si el valor hubiese sido convertido a entero y luego a flotante. Vea la sección Conversión a
entero para más información.
Cadenas
Un valor string es una serie de caracteres. En PHP, un caracter es lo mismo que un byte, es decir,
hay exactamente 256 tipos de caracteres diferentes. Esto implica también que PHP no tiene soporte
nativo de Unicode. Vea utf8_encode() y utf8_decode() para conocer sobre el soporte Unicode.
Nota: El que una cadena se haga muy grande no es un problema. PHP no impone límite
práctico alguno sobre el tamaño de las cadenas, así que no hay ninguna razón para
preocuparse sobre las cadenas largas.
Sintaxis
• comillas simples
• comillas dobles
• sintaxis heredoc
Comillas simples
La forma más simple de especificar una cadena sencilla es rodearla de comillas simples (el caracter
').
Para especificar una comilla sencilla literal, necesita escaparla con una barra invertida (\), como en
muchos otros lenguajes. Si una barra invertida necesita aparecer antes de una comilla sencilla o al
final de la cadena, necesitará doblarla. Note que si intenta escapar cualquier otro caracter, ¡la barra
invertida será impresa también! De modo que, por lo general, no hay necesidad de escapar la barra
invertida misma.
Nota: En PHP 3, se generará una advertencia de nivel E_NOTICE cuando esto ocurra.
Nota: A diferencia de las otras dos sintaxis, las variables y secuencias de escape para
caracteres especiales no serán expandidas cuando ocurren al interior de cadenas entre
comillas sencillas.
<?php
echo 'esta es una cadena simple';
Comillas dobles
Si la cadena se encuentra rodeada de comillas dobles ("), PHP entiende más secuencias de escape
para caracteres especiales:
secuencia significado
\n alimentación de línea (LF o 0x0A (10) en ASCII)
\r retorno de carro (CR o 0x0D (13) en ASCII)
\t tabulación horizontal (HT o 0x09 (9) en ASCII)
\\ barra invertida
\$ signo de dólar
\" comilla-doble
la secuencia de caracteres que coincide con la expresión regular es un caracter
\[0-7]{1,3}
en notación octal
\x[0-9A-Fa-f] la secuencia de caracteres que coincide con la expresión regular es un caracter
{1,2} en notación hexadecimal
Nuevamente, si intenta escapar cualquier otro caracter, ¡la barra invertida será impresa también!
Pero la característica más importante de las cadenas entre comillas dobles es el hecho de que los
nombres de variables serán expandidos. Vea procesamiento de cadenas para más detalles.
Heredoc
Otra forma de delimitar cadenas es mediante el uso de la sintaxis heredoc ("<<<"). Debe indicarse
un identificador después de la secuencia <<<, luego la cadena, y luego el mismo identificador para
cerrar la cita.
Aviso
Es muy importante notar que la línea con el identificador de cierre no contenga otros caracteres,
excepto quizás por un punto-y-coma (;). Esto quiere decir en especial que el identificador no debe
usar sangría, y no debe haber espacios o tabuladores antes o después del punto-y-coma. Es
importante también notar que el primer caracter antes del identificador de cierre debe ser un salto
de línea, tal y como lo defina su sistema operativo. Esto quiere decir \r en Macintosh, por ejemplo.
function foo()
{
$this->foo = 'Foo';
$this->bar = array('Bar1', 'Bar2', 'Bar3');
}
}
echo <<<FIN
Mi nombre es "$nombre". Estoy imprimiendo algo de $foo->foo.
Ahora, estoy imprimiendo algo de {$foo->bar[1]}.
Esto debería imprimir una letra 'A' mayúscula: \x41
FIN;
?>
Nota: El soporte heredoc fue agregado en PHP 4.
Procesamiento de variables
Cuando una cadena es especificada en comillas dobles o al interior de un bloque heredoc, las
variables son interpretadas en su interior.
Existen dos tipos de sintaxis: una simple y una compleja. La sintaxis simple es la más común y
conveniente. Esta ofrece una forma de interpretar una variable, un valor array, o una propiedad de
un object.
La sintaxis compleja fue introducida en PHP 4, y puede reconocerse por las llaves que rodean la
expresión.
Sintaxis simple
<?php
$cerveza = 'Heineken';
echo "El sabor de varias $cerveza's es excelente"; // funciona, "'" no es un caracter valido para
echo "Tomó algunas $cervezas"; // no funciona, 's' es un caracter valido para nombres de
echo "Tomó algunas ${cerveza}s"; // funciona
echo "Tomó algunas {$cerveza}s"; // funciona
?>
De forma similar, puede hacer que un índice de un array o una propiedad de un object sean
interpretados. En el caso de los índices de matrices, el corchete cuadrado de cierre (]) marca el final
del índice. Para las propiedades de objetos, se aplican las mismas reglas de las variables simples,
aunque con las propiedades de objetos no existe un truco como el que existe con las variables.
<?php
// Estos ejemplos son especificos al uso de matrices al interior de
// cadenas. Cuando se encuentre por fuera de una cadena, siempre rodee
// de comillas las claves tipo cadena de su matriz, y no use
// {llaves} por fuera de cadenas tampoco.
// Funciona pero note que esto trabaja de forma diferente por fuera de
// cadenas entre comillas
echo "Un banano es $frutas[banano].";
// Funciona
echo "Un banano es {$frutas['banano']}.";
// Funciona, pero PHP musca una constante llamada banano primero, como
// se describe mas adelante.
echo "Un banano es {$frutas[banano]}.";
// Funciona
echo "Un banano es " . $frutas['banano'] . ".";
// Funciona
echo "Este cuadro tiene $cuadro->ancho metros de ancho.";
Esta no es llamada compleja porque la sintaxis sea compleja, sino porque es posible incluir
expresiones complejas de esta forma.
De hecho, de esta forma puede incluir cualquier valor que sea parte del espacio de nombres al
interior de cadenas. Simplemente escriba la expresión en la misma forma que lo haría si se
encontrara por fuera de una cadena, y luego la ubica entre { y }. Ya que no es posible escapar '{',
esta sintaxis será reconocida únicamente cuando el caracter $ se encuentra inmediatamente después
de {. (Use "{\$" o "\{$" para obtener una secuencia literal "{$"). Algunos ejemplos para aclarar el
asunto:
<?php
// Mostremos todos los errores
error_reporting(E_ALL);
$genial = 'fantástico';
// Funciona
echo "Este cuadro tiene {$cuadro->ancho}00 centímetros de ancho.";
// Funciona
echo "Esto funciona: {$matriz[4][3]}";
// Esto esta mal por la misma razon por la que $foo[bar] esta mal por
// fuera de una cadena. En otras palabras, aun funciona pero ya que
// PHP busca primero una constante llamada foo, genera un error de
// nivel E_NOTICE (constante indefinida).
echo "Esto esta mal: {$matriz[foo][3]}";
// Funciona.
echo "Esto funciona: " . $arr['foo'][3];
Los caracteres al interior de una cadena pueden ser consultados y modificados al especificar el
desplazamiento, comenzando en cero, del caracter deseado después de la cadena entre llaves.
Nota: Para efectos de compatibilidad con versiones anteriores, aun puede usar corchetes
tipo matriz para el mismo propósito. Sin embargo, esta sintaxis es obsoleta a partir de
PHP 4.
?>
Funciones y operadores útiles
Las cadenas pueden ser concatenadas usando el operador '.' (punto). Note que el operador '+'
(adición) no funciona para este propósito. Por favor refiérase a la sección Operadores de cadena
para más información.
Vea la sección de funciones de cadena para consultar funciones de uso general, o las funciones de
expresiones regulares para búsquedas y reemplazos avanzados (en dos sabores: Perl y POSIX
extendido).
Existen también funciones para cadenas tipo URL, y funciones para encriptar/descifrar cadenas
(mcrypt y mhash).
Finalmente, si aun no ha encontrado lo que busca, vea también las funciones de tipo de caracter.
Conversión a cadena
Es posible convertir un valor a una cadena usando el moldeamiento (string), o la función strval().
La conversión a cadena se realiza automáticamente para usted en el contexto de una expresión
cuando se necesita una cadena. Esto ocurre cuando usa las funciones echo() o print(), o cuando
compara el valor de una variable con una cadena. El contenido de las secciones del manual sobre
Tipos y Manipulación de Tipos ayudan a aclarar este hecho. Vea también settype().
Un valor boolean TRUE es convertido a la cadena "1", el valor FALSE se representa como "" (una
cadena vacía). De esta forma, usted puede convertir de ida y vuelta entre valores booleanos y de
cadena.
Un número integer o de punto flotante (float) es convertido a una cadena que representa el número
con sus dígitos (incluyendo la parte del exponente para los números de punto flotante).
Las matrices son siempre convertidas a la cadena "Array", de modo que no puede volcar los
contenidos de un valor array con echo() o print() para ver lo que se encuentra en su interior. Para
ver un elemento, usted tendría que hacer algo como echo $arr['foo']. Vea más adelante algunos
consejos sobre el volcado/vista del contenido completo.
Los objetos son convertidos siempre a la cadena "Object". Si quisiera imprimir los valores de
variables miembro de un object para efectos de depuración, lea los parágrafos siguientes. Si quiere
conocer el nombre de clase del cual un objeto dado es instancia, use get_class().
Los recursos son siempre convertidos a cadenas con la estructura "Resource id #1" en donde 1 es el
número único del valor resource asignado por PHP durante tiempo de ejecución. Si quisiera
obtener el tipo del recurso, use get_resource_type().
Como puede apreciar, el imprimir matrices, objetos o recursos no le ofroce información útil sobre
los valores mismos. Consulte las funciones print_r() y var_dump() para conocer mejores formas
de imprimir valores para depuración.
También puede convertir valores PHP a cadenas y almacenarlas permanentemente. Este método es
conocido como seriación, y puede ser efectuado con la función serialize(). También puede seriar
valores PHP a estructuras XML, si cuenta con soporte WDDX en su configuración de PHP.
La cadena será evaluada como un float si contiene cualquier caracter entre '.', 'e', o 'E'. De otra
forma, evaluará como un entero.
El valor es dado por la porción inicial de la cadena. Si la cadena comienza con datos numéricos
válidos, éstos serán el valor usado. De lo contrario, el valor será 0 (cero). Un signo opcional es
considerado un dato numérico válido, seguido por uno o más dígitos (que pueden contener un punto
decimal), seguidos por un exponente opcional. El exponente es una 'e' o 'E' seguida de uno o más
dígitos.
<?php
$foo = 1 + "10.5"; // $foo es flotante (11.5)
$foo = 1 + "-1.3e3"; // $foo es flotante (-1299)
$foo = 1 + "bob-1.3e3"; // $foo es entero (1)
$foo = 1 + "bob3"; // $foo es entero (1)
$foo = 1 + "10 Cerditos"; // $foo es entero (11)
$foo = 4 + "10.2 Cerditos"; // $foo es flotante (14.2)
$foo = "10.0 cerdos " + 1; // $foo es flotante (11)
$foo = "10.0 cerdos " + 1.0; // $foo es flotante (11)
?>
Para más información sobre esta conversión, vea la página del manual Unix sobre strtod(3).
Si quisiera probar cualquiera de los ejemplos presentados en esta sección, puede cortar y pegar los
ejemplos e insertar la siguiente línea para verificar por sí mismo lo que está sucediendo:
<?php
echo "\$foo==$foo; tipo es " . gettype ($foo) . "<br />\n";
?>
No espere obtener el código de un caractar convirtiéndolo a un entero (como lo haría en C, por
ejemplo). Use las funciones ord() y chr() para convertir entre códigos de caracter y caracteres.
Matrices
Una matriz en PHP es en realidad un mapa ordenado. Un mapa es un tipo de datos que asocia
valores con claves. Este tipo es optimizado en varias formas, de modo que puede usarlo como una
matriz real, o una lista (vector), tabla asociativa (caso particular de implementación de un mapa),
diccionario, colección, pila, cola y probablemente más. Ya que puede tener otra matriz PHP como
valor, es realmente fácil simular árboles.
Una explicación sobre tales estructuras de datos se encuentra por fuera del propósito de este manual,
pero encontrará al menos un ejemplo de cada uno de ellos. Para más información, le referimos a
literatura externa sobre este amplio tema.
Sintaxis
Un array puede ser creado por la construcción de lenguaje array(). Ésta toma un cierto número de
parejas clave => valor separadas con coma.
<?php
$matriz = array("foo" => "bar", 12 => true);
<?php
$matriz = array("unamatriz" => array(6 => 5, 13 => 9, "a" => 42));
echo $matriz["unamatriz"][6]; // 5
echo $matriz["unamatriz"][13]; // 9
echo $matriz["unamatriz"]["a"]; // 42
?>
Si no especifica una clave para un valor dado, entonces es usado el máximo de los índices enteros, y
la nueva clave será ese valor máximo + 1. Si especifica una clave que ya tiene un valor asignado,
ése valor será sobrescrito.
<?php
// Esta matriz es la misma que ...
array(5 => 43, 32, 56, "b" => 12);
// ...esta matriz
array(5 => 43, 6 => 32, 7 => 56, "b" => 12);
?>
Aviso
A partir de PHP 4.3.0, el comportamiento de generación de índices descrito ha cambiado. Ahora,
si agrega un elemento a una matriz cuya clave máxima actual es un valor negativo, entonces la
siguiente clave creada será cero (0). Anteriormente, el nuevo índice hubiera sido establecido a la
clave mayor existente + 1, al igual que con los índices positivos.
Al usar TRUE como clave, el valor será evaluado al integer 1. Al usar FALSE como clave, el valor
será evaluado al integer 0. Al usar NULL como clave, el valor será evaluado a una cadena vacía. El
uso de una cadena vacía como clave creará (o reemplazará) una clave con la cadena vacía y su valor;
no es lo mismo que usar corchetes vacíos.
No es posible usar matrices u objetos como claves. Al hacerlo se producirá una advertencia: Illegal
offset type.
Creación/modificación con sintaxis de corchetes cuadrados
Esto es posible al asignar valores a la matriz al mismo tiempo que se especifica la clave entre
corchetes. También es posible omitir la clave, agregar una pareja vacía de corchetes ("[]") al
nombre de la variable en ese caso.
$matriz[clave] = valor;
$matriz[] = valor;
// clave puede ser un integer o string
// valor puede ser cualquier valor
Si $matriz no existe aun, ésta será creada. De modo que esta es también una forma alternativa de
especificar una matriz. Para modificar un cierto valor, simplemente asigne un nuevo valor a un
elemento especificado con su clave. Si desea remover una pareja clave/valor, necesita eliminarla
mediante unset().
<?php
$matriz = array(5 => 1, 12 => 2);
Aviso
A partir de PHP 4.3.0, el comportamiento de generación de índices descrito ha cambiado. Ahora,
si agrega un elemento al final de una matriz en la que la clave máxima actual es negativa, la
siguiente clave creada será cero (0). Anteriormente, el nuevo índice hubiera sido definido como la
mayor clave + 1, al igual que ocurre con los índices positivos.
Note que la clave entera máxima usada para este caso no necesita existir actualmente en
la matriz. Tan solo debe haber existido en la matriz en algún punto desde que la matriz
haya sido re-indexada. El siguiente ejemplo ilustra este caso:
<?php
// Crear una matriz simple.
$matriz = array(1, 2, 3, 4, 5);
print_r($matriz);
// Re-indexar:
$matriz = array_values($matriz);
$matriz[] = 7;
print_r($matriz);
?>
El anterior ejemplo produciría la siguiente salida:
Array
(
[0] => 1
[1] => 2
[2] => 3
[3] => 4
[4] => 5
)
Array
(
)
Array
(
[5] => 6
)
Array
(
[0] => 6
[1] => 7
)
Funciones útiles
Existe un buen número de funciones útiles para trabajar con matrices. Consulte la sección funciones
de matrices.
$b = array_values($a);
// Ahora $b es array(0 => 'uno', 1 =>'tres')
?>
La estructura de control foreach existe específicamente para las matrices. Ésta provee una manera
fácil de recorrer una matriz.
Siempre deben usarse comillas alrededor de un índice de matriz tipo cadena literal. Por ejemplo, use
$foo['bar'] y no $foo[bar]. ¿Pero qué está mal en $foo[bar]? Es posible que haya visto la siguiente
sintaxis en scripts viejos:
<?php
$foo[bar] = 'enemigo';
echo $foo[bar];
// etc
?>
Esto está mal, pero funciona. Entonces, ¿porqué está mal? La razón es que este código tiene una
constante indefinida (bar) en lugar de una cadena ('bar' - note las comillas), y puede que en el futuro
PHP defina constantes que, desafortunadamente para su código, tengan el mismo nombre. Funciona
porque PHP automáticamente convierte una cadena pura (una cadena sin comillas que no
corresponda con símbolo conocido alguno) en una cadena que contiene la cadena pura. Por ejemplo,
si no se ha definido una constante llamada bar, entonces PHP reemplazará su valor por la cadena
'bar' y usará ésta última.
Nota: Esto no quiere decir que siempre haya que usar comillas en la clave. No querrá
usar comillas con claves que sean constantes o variables, ya que en tal caso PHP no
podrá interpretar sus valores.
<?php
error_reporting(E_ALL);
ini_set('display_errors', true);
ini_set('html_errors', false);
// Matriz simple:
$matriz = array(1, 2);
$conteo = count($matriz);
for ($i = 0; $i < $conteo; $i++) {
echo "\nRevisando $i: \n";
echo "Mal: " . $matriz['$i'] . "\n";
echo "Bien: " . $matriz[$i] . "\n";
echo "Mal: {$matriz['$i']}\n";
echo "Bien: {$matriz[$i]}\n";
}
?>
Nota: La salida del anterior fragmento es:
Revisando 0:
Notice: Undefined index: $i in /path/to/script.html on line 9
Mal:
Bien: 1
Notice: Undefined index: $i in /path/to/script.html on line 11
Mal:
Bien: 1
Revisando 1:
Notice: Undefined index: $i in /path/to/script.html on line 9
Mal:
Bien: 2
Notice: Undefined index: $i in /path/to/script.html on line 11
Mal:
Bien: 2
Más ejemplos para demostrar este hecho:
<?php
// Mostrar todos los errores
error_reporting(E_ALL);
// Correcto
print $matriz['fruta']; // manzana
print $matriz['vegetal']; // zanahoria
Tal y como se indica en la sección de sintaxis, debe existir una expresión entre los corchetes
cuadrados ('[' y ']'). Eso quiere decir que puede escribir cosas como esta:
<?php
echo $matriz[alguna_funcion($bar)];
?>
Este es un ejemplo del uso de un valor devuelto por una función como índice de matriz. PHP
también conoce las constantes, tal y como ha podido apreciar aquellas E_* antes.
<?php
$descripciones_de_error[E_ERROR] = "Un error fatal ha ocurrido";
$descripciones_de_error[E_WARNING] = "PHP produjo una advertencia";
$descripciones_de_error[E_NOTICE] = "Esta es una noticia informal";
?>
Note que E_ERROR es también un identificador válido, asi como bar en el primer ejemplo. Pero el
último ejemplo es ñalente a escribir:
<?php
$descripciones_de_error[1] = "Un error fatal ha ocurrido";
$descripciones_de_error[2] = "PHP produjo una advertencia";
$descripciones_de_error[8] = "Esta es una noticia informal";
?>
ya que E_ERROR es igual a 1, etc.
Tal y como lo hemos explicado en los anteriores ejemplos, $foo[bar] aun funciona pero está mal.
Funciona, porque debido a su sintaxis, se espera que bar sea una expresión constante. Sin embargo,
en este caso no existe una constante con el nombre bar. PHP asume ahora que usted quiso decir bar
literalmente, como la cadena "bar", pero que olvidó escribir las comillas.
En algún momento en el futuro, el equipo de PHP puede querer usar otra constante o palabra clave,
o puede que usted introduzca otra constante en su aplicación, y entonces se ve en problemas. Por
ejemplo, en este momento no puede usar las palabras empty y default de esta forma, ya que son
palabras clave reservadas especiales.
Conversión a matriz
Para cualquiera de los tipos: integer, float, string, boolean y resource, si convierte un valor a un
array, obtiene una matriz con un elemento (con índice 0), el cual es el valor escalar con el que
inició.
Si convierte un object a una matriz, obtiene las propiedades (variables miembro) de ese objeto
como los elementos de la matriz. Las claves son los nombres de las variables miembro.
Ejemplos
El tipo matriz en PHP es bastante versátil, así que aquí se presentan algunos ejemplos que
demuestran el poder completo de las matrices.
<?php
// esto
$a = array( 'color' => 'rojo',
'sabor' => 'dulce',
'forma' => 'redonda',
'nombre' => 'manzana',
4 // la clave sera 0
);
$b[] = 'a';
$b[] = 'b';
$b[] = 'c';
// resultara en la matriz array(0 => 'a' , 1 => 'b' , 2 => 'c'),
// o simplemente array('a', 'b', 'c')
?>
Ejemplo 11-5. Uso de array()
<?php
// Array como mapa de propiedades
$mapa = array( 'version' => 4,
'SO' => 'Linux',
'idioma => 'ingles',
'etiquetas_cortas' => true
);
// matriz vacia
$vacio = array();
?>
Ejemplo 11-6. Colección
<?php
$colores = array('rojo', 'azul', 'verde', 'amarillo');
?>
Esto producirá la salida:
¿Le gusta el rojo?
¿Le gusta el azul?
¿Le gusta el verde?
¿Le gusta el amarillo?
Note que actualmente no es posible cambiar los valores de la matriz directamente en un ciclo de ese
tipo. Una solución parcial es la siguiente:
Ejemplo 11-7. Colección
<?php
foreach ($colores as $clave => $color) {
// no funciona:
//$color = strtoupper($color);
// funciona:
$colores[$clave] = strtoupper($color);
}
print_r($colores);
?>
Esto genera la salida:
Array
(
[0] => ROJO
[1] => AZUL
[2] => VERDE
[3] => AMARILLO
)
Este ejemplo crea una matriz con base uno.
Ejemplo 11-8. Índice con base 1
<?php
$primercuarto = array(1 => 'Enero', 'Febrero', 'Marzo');
print_r($primercuarto);
?>
Esto imprime:
Array
(
[1] => 'Enero'
[2] => 'Febrero'
[3] => 'Marzo'
)
Ejemplo 11-9. Llenado de una matriz
<?php
// llenar una matriz con todos los items de un directorio
$gestor = opendir('.');
while (false !== ($archivo = readdir($gestor))) {
$archivos[] = $archivo;
}
closedir($gestor);
?>
Las matrices son ordenadas. Puede también cambiar el orden usando varias funciones de
ordenamiento. Vea la sección sobre funciones de matrices para más información. Puede contar el
número de items en una matriz usando la función count().
<?php
$matriz1 = array(2, 3);
$matriz2 = $matriz1;
$matriz2[] = 4; // $matriz2 cambia,
// $matriz1 sigue siendo array(2, 3)
$matriz3 = &$matriz1;
$matriz3[] = 4; // ahora $matriz1 y $matriz3 son iguales
?>
Objetos
Inicialización de Objetos
Para inicializar un objeto, use la sentencia new, lo que instancia el objeto a una variable.