mirror of
https://github.com/MariaDB/server.git
synced 2025-01-19 13:32:33 +01:00
1644 lines
70 KiB
Text
1644 lines
70 KiB
Text
|
8 Tutorial MySQL
|
|||
|
=======================
|
|||
|
|
|||
|
Este cap<61><70>tulo ofrece un tutorial de introducci<63><69>n a MySQL, mostrando c<><63>mo usar el programa cliente
|
|||
|
mysql para crear y usar una simple base de datos. mysql (al que algunas veces nos referimos como
|
|||
|
"monitor terminal" o simplemente "monitor") es un programa interactivo que te permite conectarte a
|
|||
|
un servidor MySQL, ejecutar consultas y observar los resultados. mysql puede ser usado tambi<62><69>n en
|
|||
|
modo batch: escribes tus consultas en un fichero de texto, para despu<70><75>s pedirle a mysql que
|
|||
|
ejecute el contenido del fichero. Se cubren aqu<71><75> esas dos formas de usar de usar mysql.
|
|||
|
|
|||
|
Para ver una lista de opciones proporcionadas por mysql, l<><6C>nzalo con las opci<63><69>n --help :
|
|||
|
|
|||
|
shell> mysql --help
|
|||
|
|
|||
|
Este cap<61><70>tulo asume que mysql est<73><74> instalado en tu m<><6D>quina, y que hay disponible un servidor al que
|
|||
|
te puedes conectar. Si esto no es as<61><73>, contacta con tu administrador MySQL. (Si el administrador
|
|||
|
eres t<><74>, necesitar<61><72>s consultar otra secci<63><69>n de este manual).
|
|||
|
|
|||
|
El cap<61><70>tulo describe el proceso completo de configurar y usar una base de datos. Si est<73><74>s interesado
|
|||
|
s<EFBFBD><EFBFBD>lo en acceder una base de datos ya existente, querr<72><72>s saltar las secciones que describen c<><63>mo
|
|||
|
crear la base de datos y las tablas que la contienen.
|
|||
|
|
|||
|
Dado que este cap<61><70>tulo es un tutorial b<><62>sico, se dejar<61><72>n en el tintero muchos
|
|||
|
detalles. Consulta las secciones relevantes del manual para m<><6D>s informaci<63><69>n sobre los temas
|
|||
|
aqu<EFBFBD><EFBFBD> cubiertos.
|
|||
|
|
|||
|
|
|||
|
8.1 Conectando y desconectando del servidor
|
|||
|
=============================================
|
|||
|
|
|||
|
|
|||
|
Para conectarse al servidor, generalmente necesitar<61><72>s facilitar un nombre de usuario MySQL cuando
|
|||
|
lances el cliente mysql y, lo m<><6D>s probable, tambi<62><69>n un password. Si el servidor se est<73><74> ejecutando
|
|||
|
en una m<><6D>quina distinta a la que est<73><74>s conectado, necesitar<61><72>s especificar tambi<62><69>n un nombre de
|
|||
|
host. Contacta con tu administrador para averiguar qu<71><75> par<61><72>metros de conexi<78><69>n necesitas usar para
|
|||
|
conectar (es decir, qu<71><75> host, nombre de usuario y password usar). Una vez que conozcas los
|
|||
|
par<EFBFBD><EFBFBD>metros adecuados, deber<65><72>s ser capaz de conectar de la siguiente forma:
|
|||
|
|
|||
|
shell> mysql -h host -u user -p
|
|||
|
Enter password: *******
|
|||
|
Welcome to the MySQL monitor. Commands end with ; or \g.
|
|||
|
Your MySQL connection id is 459 to server version: 3.22.20a-log
|
|||
|
|
|||
|
Type 'help' for help.
|
|||
|
|
|||
|
mysql>
|
|||
|
|
|||
|
El prompt te indica que mysql ya est<73><74> listo para la introducci<63><69>n de comandos.
|
|||
|
|
|||
|
Algunas instalaciones MySQL permiten a los usuarios conectarse como usuarios "anonymous" (sin
|
|||
|
nombre) al servidor ejecut<75><74>ndose en el host local. Si este es el caso en tu m<><6D>quina, deber<65><72>as ser
|
|||
|
capaz de conectar a ese servidor invocando mysql sin ninguna opci<63><69>n:
|
|||
|
|
|||
|
shell> mysql
|
|||
|
|
|||
|
Una vez que hayas conectado con <20><>xito, puedes desconectarte en cualquier momento tecleando QUIT en
|
|||
|
el prompt mysql> :
|
|||
|
|
|||
|
mysql> QUIT
|
|||
|
Bye
|
|||
|
|
|||
|
Tambi<EFBFBD><EFBFBD>n puedes desconectar tecleando control-D.
|
|||
|
|
|||
|
La mayor parte de los ejemplos en las siguientes secciones asumen que est<73><74>s conectado al
|
|||
|
servidor. Lo indicar<61><72>n por el prompt mysql>
|
|||
|
|
|||
|
|
|||
|
8.2 Haciendo consultas
|
|||
|
========================
|
|||
|
|
|||
|
Aseg<EFBFBD><EFBFBD>rate de que est<73><74>s conectado al servidor, como se ha discutido en secciones anteriores. El
|
|||
|
hacerlo no implica que tengas seleccionada ninguna base de datos con la que trabajar, pero est<73><74>
|
|||
|
bien. En este punto, es m<><6D>s importante averiguar un poco sobre c<><63>mo lanzar consultas que lanzarse
|
|||
|
directamente a la creaci<63><69>n de tablas, cargar datos en ellas y recuperar los datos de las
|
|||
|
mismas. Esta secci<63><69>n describe los principios b<><62>sicos de la entrada de comandos, usando varias
|
|||
|
consultas que puedes probar para familiarizarte con la forma de trabajo de mysql.
|
|||
|
|
|||
|
Aqu<EFBFBD><EFBFBD> presentamos un comando simple que pide al servidor que nos diga su n<><6E>mero de versi<73><69>n y fecha
|
|||
|
actual. Tecl<63><6C>alo como se muestra a continuaci<63><69>n siguiendo el prompt mysql> y pulsa la tecla RETURN:
|
|||
|
|
|||
|
|
|||
|
mysql> SELECT VERSION(), CURRENT_DATE;
|
|||
|
+-----------+--------------+
|
|||
|
| version() | CURRENT_DATE |
|
|||
|
+-----------+--------------+
|
|||
|
| 3.22.23b | 2000-01-05 |
|
|||
|
+-----------+--------------+
|
|||
|
1 row in set (0.06 sec)
|
|||
|
|
|||
|
mysql>
|
|||
|
|
|||
|
Esta consulta ilustra muchas cosas sobre mysql:
|
|||
|
|
|||
|
* Un comando consiste normalmente de una sentencia SQL seguida por un punto y coma. (Existen algunas
|
|||
|
excepciones donde no es necesario el punto y coma. QUIT, mencionado m<><6D>s adelante, es una de
|
|||
|
ellas. Conoceremos otras m<><6D>s adelante.)
|
|||
|
|
|||
|
* Cuando lanzas un comando, mysql lo env<6E><76>a al servidor para su ejecuci<63><69>n y muestra los resultados,
|
|||
|
despu<70><75>s imprime otro mysql> para indicar que est<73><74> listo para otro comando.
|
|||
|
|
|||
|
* mysql muestra la salida de una consulta como una tabla (filas y columnas). La primera fila
|
|||
|
contiene etiquetas para las columnas. Las siguientes filas son el resultado de la
|
|||
|
consulta. Normalmente, las etiquetas de las columnas son los nombres de las columnas que has
|
|||
|
obtenido de la base de datos. Si pides el valor de una expresi<73><69>n en vez de una columna de una
|
|||
|
tabla (como en el ejemplo anterior), mysql etiqueta la columna usando la propia expresi<73><69>n.
|
|||
|
|
|||
|
* mysql muestra el n<><6E>mero de filas que se han dado como resultado, y cu<63><75>nto tiempo llev<65><76> la
|
|||
|
ejecuci<63><69>n de la consulta, lo que te da una idea aproximada del rendimiento del servidor. Estos
|
|||
|
valores son imprecisos porque representan tiempo real (no tiempo de CPU o m<><6D>quina), y porque est<73><74>n
|
|||
|
afectados por factores como la carga del servidor y la latencia de la red. (Por cuestiones de
|
|||
|
brevedad, la l<><6C>nea "rows in set" no se mostrar<61><72> en los ejemplos posteriores de este cap<61><70>tulo.)
|
|||
|
|
|||
|
Las palabras clave pueden ser tecleadas en cualquier combinaci<63><69>n may<61><79>scula/min<69><6E>scula. Las siguientes
|
|||
|
consultas son equivalentes:
|
|||
|
|
|||
|
mysql> SELECT VERSION(), CURRENT_DATE;
|
|||
|
mysql> select version(), current_date;
|
|||
|
mysql> SeLeCt vErSiOn(), current_DATE;
|
|||
|
|
|||
|
He aqu<71><75> otra consulta. Demuestra que puedes usar mysql como una calculadora sencilla:
|
|||
|
|
|||
|
mysql> SELECT SIN(PI()/4), (4+1)*5;
|
|||
|
+-------------+---------+
|
|||
|
| SIN(PI()/4) | (4+1)*5 |
|
|||
|
+-------------+---------+
|
|||
|
| 0.707107 | 25 |
|
|||
|
+-------------+---------+
|
|||
|
|
|||
|
Los comandos vistos hasta aqu<71><75> han sido relativamente cortos, sentencias de una sola l<><6C>nea. Tambi<62><69>n puedes
|
|||
|
insertar m<><6D>ltiples sentencias en una sola l<><6C>nea. Simplemente, termina cada una con un punto y coma:
|
|||
|
|
|||
|
mysql> SELECT VERSION(); SELECT NOW();
|
|||
|
|
|||
|
+-----------+
|
|||
|
| version() |
|
|||
|
+-----------+
|
|||
|
| 3.22.23b |
|
|||
|
+-----------+
|
|||
|
|
|||
|
+---------------------+
|
|||
|
| NOW() |
|
|||
|
+---------------------+
|
|||
|
| 2000-01-05 17:33:16 |
|
|||
|
+---------------------+
|
|||
|
|
|||
|
Un comando no necesita ser dado todo en una s<><73>la l<><6C>nea, as<61><73> pues, los comandos largos que requieran
|
|||
|
varias lineas no son un problema. mysql determina cuando termina tu sentencia buscando el punto y
|
|||
|
coma final, no buscando el final de la l<><6C>nea de entrada. (En otras palabras, mysql acepta entrada de
|
|||
|
libre formato: recoleta las l<><6C>neas de entrada pero no las ejecutahasta que vea el punto y coma.)
|
|||
|
|
|||
|
Aqu<EFBFBD><EFBFBD> tenemos un simple ejemplo de m<><6D>ltiples l<><6C>neas:
|
|||
|
|
|||
|
mysql> SELECT
|
|||
|
-> USER()
|
|||
|
-> ,
|
|||
|
-> CURRENT_DATE;
|
|||
|
+----------------+--------------+
|
|||
|
| USER() | CURRENT_DATE |
|
|||
|
+----------------+--------------+
|
|||
|
| root@localhost | 2000-01-05 |
|
|||
|
+----------------+--------------+
|
|||
|
|
|||
|
En este ejemplo, observa como cambia el prompt de mysql> a -> una vez que has insertado la primera
|
|||
|
l<EFBFBD><EFBFBD>nea de una consulta multi-l<><6C>nea. Esta es la forma en que mysql indica que no ha encontrado una
|
|||
|
sentencia completa y que est<73><74> esperando por el resto. El prompt es tu amigo, dado que ofrece una
|
|||
|
retroalimentaci<EFBFBD><EFBFBD>n (feedback) significativa. Si usas ese feedback, siempre sabr<62><72>s a qu<71><75> est<73><74>
|
|||
|
esperando mysql.
|
|||
|
|
|||
|
Si decides que no quieres ejecutar un comando que est<73><74> en proceso de introducci<63><69>n, puedes cancelarlo
|
|||
|
tecleando \c :
|
|||
|
|
|||
|
mysql> SELECT
|
|||
|
-> USER
|
|||
|
-> \c
|
|||
|
mysql>
|
|||
|
|
|||
|
Observa aqu<71><75> tambi<62><69>n el prompt. Ha vuelto a mysql> tras haber tecleado \c, ofreciendo un feedback
|
|||
|
que indica que mysql est<73><74> listo para un nuevo comando.
|
|||
|
|
|||
|
La siguiente tabla muestra cada uno de los prompts que puedes ver y resume qu<71><75> es lo que significan
|
|||
|
y el estado en el que se encontrar<61><72> mysql:
|
|||
|
|
|||
|
Prompt Significado
|
|||
|
mysql> Listo para un nuevo comando
|
|||
|
-> Esperando una nueva l<><6C>nea de una consulta multi-l<><6C>nea
|
|||
|
'> Esperando la siguiente l<><6C>nea, se ha insertado una l<><6C>nea que comienza con (')
|
|||
|
"> Esperando la siguiente l<><6C>nea, se ha insertado una l<><6C>nea que comienza con (")
|
|||
|
|
|||
|
Las sentencias multi-l<><6C>nea ocurren com<6F><6D>nmente "por accidente" cuando intentas lanzar un comando en
|
|||
|
una <20><>nica l<><6C>nea, pero olvidas el punto y coma del final. En este caso, mysql espera m<><6D>s entrada:
|
|||
|
|
|||
|
mysql> SELECT USER()
|
|||
|
->
|
|||
|
|
|||
|
Si esto es lo que te ocurre (crees que has introducido una sentencia pero la <20><>nica respuesta es un
|
|||
|
prompt como ->), lo m<><6D>s probable es que mysql est<73><74> esperando por el punto y coma. Si no observas qu<71><75>
|
|||
|
es lo que te dice el prompt, podr<64><72>as quedarte esperando un buen rato antes de enterarte de qu<71><75> es lo
|
|||
|
que sucede. Introduce un punto y coma para completar la sentencia, y mysql la ejecutar<61><72>:
|
|||
|
|
|||
|
mysql> SELECT USER()
|
|||
|
-> ;
|
|||
|
+----------------+
|
|||
|
| USER() |
|
|||
|
+----------------+
|
|||
|
| root@localhost |
|
|||
|
+----------------+
|
|||
|
|
|||
|
Los prompts '> y "> ocurren durante la recogida de strings. En MySQL, puedes escribir strings
|
|||
|
encerrados por comillas simples (') o dobles (") (por ejemplo, 'hola' o "adios"), y mysql te permite
|
|||
|
introducir tambi<62><69>n strings que se cortan en m<><6D>ltiples l<><6C>neas. Cuando veas un prompt como '> <20><> ">,
|
|||
|
significa que has introducido una l<><6C>nea que conten<65><6E>a un string que comenzaba por (') o ("), pero que
|
|||
|
no has introducido a<><61>n la comilla (simple o doble) de cierre. Esto est<73><74> bien si realmente estabas
|
|||
|
introduciendo un string multi-l<><6C>nea, pero no es lo m<><6D>s normal. Lo que s<><73> es m<><6D>s normal, es que los
|
|||
|
prompts '> <20><> "> indiquen que te has olvidado del caracter de cierre " <20><> '. Por ejemplo:
|
|||
|
|
|||
|
mysql> SELECT * FROM mi_tabla WHERE nombre ="Garc<72><63>a AND edad < 30;
|
|||
|
">
|
|||
|
|
|||
|
Si tecleas esta sentencia SELECT, despu<70><75>s pulsas ENTER y esperas por el resultado, no suceder<65><72>
|
|||
|
nada. En lugar de preocuparte, "<22><>por qu<71><75> tarda tanto esta consulta?", observa la pista que te ofrece
|
|||
|
el prompt "> . Esto te indica que mysql espera ver el resto de un string que a<><61>n no ha
|
|||
|
terminado. (<28><>Ves el error en la sentencia? La cadena "Garc<72><63>a ha perdido las comillas de cierre.)
|
|||
|
|
|||
|
Llegados a este punto, <20><>qu<71><75> puedes hacer?. Lo m<><6D>s f<><66>cil es cancelar el comando. Sin embargo, no
|
|||
|
puedes teclear simplemente \c en este ejemplo, dado que mysql <20><>lo interpretar<61><72> como parte del string
|
|||
|
que est<73><74> leyendo! En vez de eso, introduce las comillas de cierre (para que mysql sepa que ya has
|
|||
|
terminado de introducir el string), y despu<70><75>s teclea \c :
|
|||
|
|
|||
|
mysql> SELECT * FROM mi_tabla WHERE nombre ="Garc<72><63>a AND edad < 30;
|
|||
|
"> "\c
|
|||
|
mysql>
|
|||
|
|
|||
|
El prompt vuelve a cambiar a mysql>, indicando que mysql est<73><74> listo para un nuevo comando.
|
|||
|
|
|||
|
Es importante saber qu<71><75> significan los prompts '> y ">, dado que si introduces por error un string
|
|||
|
sin cerrar, cualquier otra l<><6C>nea que introduzcas ser<65><72>n ignoradas por mysql - <20><>incluyendo una l<><6C>nea
|
|||
|
que contenga QUIT! Esto puede ser bastante confuso, especialmente si no sabes que debes introducir
|
|||
|
la comilla de cierre antes de poder cancelar el comando actual.
|
|||
|
|
|||
|
8.3 Creando y usando una base de datos
|
|||
|
==========================================
|
|||
|
|
|||
|
Ahora que sabes como introducir comandos, es hora de acceder a la base de datos.
|
|||
|
|
|||
|
Supon que tienes varias mascotas en tu casa (tu peque<75><65>o "zoo") y que te gustar<61><72>a llevar un control
|
|||
|
de varios tipos de informaci<63><69>n sobre estos animales. Puedes hacerlo creando tablas que guarden tus
|
|||
|
datos y cargandolas con la informaci<63><69>n deseada. Despu<70><75>s puedes responder a diferentes series de
|
|||
|
preguntas sobre tus animales extrayendo los datos de las tablas. Esta secci<63><69>n explica c<><63>mo hacer
|
|||
|
todo esto:
|
|||
|
|
|||
|
* C<><43>mo crear una base de datos
|
|||
|
* C<><43>mo crear una tabla
|
|||
|
* C<><43>mo cargar los datos en la tabla
|
|||
|
* C<><43>mo extraer informaci<63><69>n de la tabla de varias maneras
|
|||
|
* C<><43>mo usar m<><6D>ltiples tablas
|
|||
|
|
|||
|
La base de datos del zoo ser<65><72> simple (deliberadamente), pero no es dif<69><66>cil pensar en situaciones del
|
|||
|
mundo real en las que se pudiera utilizar una base de datos similar. Por ejemplo, se podr<64><72>a usar una base
|
|||
|
de datos como <20><>sta en una granja para llevar un control del ganado, o por un veterinario para
|
|||
|
controlar el historial de sus pacientes.
|
|||
|
|
|||
|
Usa la sentencia SHOW para averiguar qu<71><75> bases de datos existen actualmente en el servidor:
|
|||
|
|
|||
|
mysql> SHOW DATABASES;
|
|||
|
+----------+
|
|||
|
| Database |
|
|||
|
+----------+
|
|||
|
| mysql |
|
|||
|
| test |
|
|||
|
+----------+
|
|||
|
|
|||
|
Probablemente, la lista de las bases de datos ser<65><72> diferente en tu m<><6D>quina, pero las bases de datos
|
|||
|
mysql y test es probable que se encuentren en esa lista. Se requiere la base de datos mysql pues
|
|||
|
describe los privilegios de acceso de los usuarios. La base de datos test se ofrece como campo de
|
|||
|
pruebas para que los usuarios prueben ah<61><68> sus teor<6F><72>as.
|
|||
|
|
|||
|
Si la base de datos test existe, intenta acceder a ella:
|
|||
|
|
|||
|
mysql> USE test
|
|||
|
Database changed
|
|||
|
|
|||
|
Observa que USE, como QUIT, no requiere un punto y coma. (Puedes terminar este tipo de sentencias
|
|||
|
con un punto y coma si quieres, pero no es necesario.) La sentencia USE es especial en otro sentido,
|
|||
|
tambi<EFBFBD><EFBFBD>n: debe ser tecleada en una sola l<><6C>nea.
|
|||
|
|
|||
|
Puedes usar la base de datos test (si tienes acceso a ella) para los ejemplos que siguen, pero
|
|||
|
cualquier cosa que crees en dicha base de datos puede ser eliminada por cualquiera que tenga acceso
|
|||
|
a ella. Por esta raz<61><7A>n, deber<65><72>as pedir a tu administrador MySQL permisos para usar una base de datos
|
|||
|
propia. Suponte que le quieres llamar zoo. El administrador necesitar<61><72> ejecutar entonces la
|
|||
|
siguiente orden:
|
|||
|
|
|||
|
mysql> GRANT ALL ON zoo.* TO tu_nombre;
|
|||
|
|
|||
|
donde tu_nombre es el nombre de usuario MySQL que tengas asignado.
|
|||
|
|
|||
|
ejemplo:
|
|||
|
|
|||
|
mysql> GRANT ALL ON zoo.* TO chessy@localhost;
|
|||
|
Query OK, 0 rows affected (0.08 sec)
|
|||
|
|
|||
|
|
|||
|
8.3.1 Creando y seleccionando una base de datos
|
|||
|
==================================================
|
|||
|
|
|||
|
Si el administrador cre<72><65> la base de datos para t<><74> cuando te configur<75><72> los permisos, puedes comenzar
|
|||
|
a usarla. En otro caso, deber<65><72>s crearla t<><74> mismo:
|
|||
|
|
|||
|
[chessy@bishito chessy]$ mysql -u chessy
|
|||
|
Welcome to the MySQL monitor. Commands end with ; or \g.
|
|||
|
Your MySQL connection id is 6 to server version: 3.22.23b
|
|||
|
|
|||
|
Type 'help' for help.
|
|||
|
|
|||
|
mysql> CREATE DATABASE zoo;
|
|||
|
Query OK, 1 row affected (0.02 sec)
|
|||
|
|
|||
|
|
|||
|
Bajo Unix, los nombres de bases de datos son sensibles a las may<61><79>sculas/min<69><6E>sculas (a diferencia de
|
|||
|
los comandos SQL), as<61><73> que deber<65><72>s referirte siempre a tu base de datos con el nombre zoo, no como
|
|||
|
Zoo, ZOO o cualquier otra variante. Es es as<61><73> tambi<62><69>n para las tablas. (Bajo Windows, esta
|
|||
|
restricci<EFBFBD><EFBFBD>n desaparece, aunque deber<65><72>as referirte a las bases de datos y a las tablas usando la
|
|||
|
misma sintaxis en tus consultas.)
|
|||
|
|
|||
|
Crear una base de datos no la selecciona para su uso, debes hacerlo expl<70><6C>citamente. Para hacer que
|
|||
|
la base de datos zoo sea tu base de datos de trabajo, usa el comando:
|
|||
|
|
|||
|
mysql> USE zoo;
|
|||
|
Database changed
|
|||
|
|
|||
|
Tu base de datos s<><73>lo necesita ser creada una vez, pero debes seleccionarla para usarla cada vez que
|
|||
|
comiences una sesi<73><69>n mysql. Puedes hacerlo lanzando un comando USE como se ha visto en el
|
|||
|
ejemplo. Alternativamente, puedes seleccionar la base de datos desde la l<><6C>nea de comandos cuando
|
|||
|
lanzas mysql. Simplemente especifica su nombre tras los par<61><72>metros de conexi<78><69>n que hayas
|
|||
|
escrito. Por ejemplo:
|
|||
|
|
|||
|
shell> mysql -h host -u user -p zoo
|
|||
|
Enter password: ********
|
|||
|
|
|||
|
Observa que en la l<><6C>nea de comandos del ejemplo, zoo no es tu password. Si quieres introducir tu
|
|||
|
password como par<61><72>metro en la l<><6C>nea de comandos tras la opci<63><69>n -p, debes hacerlo sin teclear un
|
|||
|
espacio en blanco intermedio (es decir, como -pmi_password, no como -p mi_password). Sin embargo, no
|
|||
|
es recomendable poner tu password en la l<><6C>nea de comandos, pues hacerlo lo expone a posibles
|
|||
|
miradas de otros usuarios conectados a tu m<><6D>quina.
|
|||
|
|
|||
|
8.3.2 Creando una tabla
|
|||
|
============================
|
|||
|
|
|||
|
Crear una tabla es la parte f<><66>cil, pero hasta este momento est<73><74> vac<61><63>a, como te dice la orden SHOW
|
|||
|
TABLES:
|
|||
|
|
|||
|
mysql> SHOW TABLES;
|
|||
|
Empty set (0.00 sec)
|
|||
|
|
|||
|
La parte m<><6D>s dura consiste en decidir cual va a ser la estructura de tu base de datos: qu<71><75> tablas
|
|||
|
necesitar<EFBFBD><EFBFBD>s, y qu<71><75> columnas tendr<64><72> cada una de ellas.
|
|||
|
|
|||
|
Querr<EFBFBD><EFBFBD>s seguramente una tabla que contenga un registro por cada una de tus mascotas. Esta tabla
|
|||
|
puede llamarse mascotas, y deber<65><72>a contener, como m<><6D>nimo, el nombre de cada animal. Dado que el
|
|||
|
nombre por s<><73> solo no es muy interesante, la tabla deber<65><72>a contener otra informaci<63><69>n. Por ejemplo,
|
|||
|
si m<><6D>s de una persona de tu familia tiene mascotas, probablemente quieras listar el propietario de
|
|||
|
cada animal. Tambi<62><69>n querr<72><72>s guardar informaci<63><69>n descriptiva b<><62>sica como puede ser la especie y el
|
|||
|
sexo de cada mascota.
|
|||
|
|
|||
|
<EFBFBD><EFBFBD>Qu<EFBFBD><EFBFBD> pasa con la edad? Podr<64><72>a ser de inter<65><72>s, pero no es una buena cosa a guardar en una base de
|
|||
|
datos. La edad cambia a medida que pasa el tiempo, lo que significa que tendr<64><72>s que actualizar tus
|
|||
|
registros a menudo. En vez de eso, es mejor almacenar un valor fijo como la edad de
|
|||
|
nacimiento. Despu<70><75>s, cada vez que necesites saber la edad, puedes calcularla como la diferencia
|
|||
|
entre la fecha actual y la fecha de nacimiento. MySQL ofrece funciones para realizar c<><63>lculos
|
|||
|
aritm<EFBFBD><EFBFBD>ticos entre fechas, por lo que esto no es dif<69><66>cil. Almacenar la fecha de nacimiento en lugar
|
|||
|
de la edad tiene tambi<62><69>n otras ventajas:
|
|||
|
|
|||
|
* Puedes usar la base de datos para generar recordatorios de cumplea<65><61>os de mascotas. (Si crees que
|
|||
|
este tipo de consulta es algo tonta, observa que es la misma pregunta que necesitar<61><72>s hacer en el
|
|||
|
contexto de una base de datos de un negocio para identificar clientes a los que pronto necesitar<61><72>s
|
|||
|
mandar un saludo por su cumplea<65><61>os, para ese toque personal asistido por ordenador :-)
|
|||
|
|
|||
|
* Puedes calcular la edad en relaci<63><69>n a fechas distintas a la fecha actual. Por ejemplo, si
|
|||
|
almacenas la fecha de muerte en la base de datos, puedes calcular f<><66>cilmente lo vieja que era una
|
|||
|
mascota cuando muri<72><69>.
|
|||
|
|
|||
|
Seguramente puedas pensar en otros tipos de informaci<63><69>n que ser<65><72>a <20><>til en la tabla mascota, pero los
|
|||
|
identificados hasta ahora son suficientes por el momento: nombre, propietarios, especie, sexo, fecha
|
|||
|
de nacimiento y muerte.
|
|||
|
|
|||
|
Usa una sentencia CREATE TABLE para especificar la estructura de tu tabla:
|
|||
|
|
|||
|
mysql> CREATE TABLE mascota (nombre VARCHAR(20), propietario VARCHAR(20),
|
|||
|
-> especie VARCHAR(20), sexo CHAR(1), nacimiento DATE, muerte DATE);
|
|||
|
|
|||
|
VARCHAR es una buena elecci<63><69>n para las columnas nombre, propietario y especie dado que los valores
|
|||
|
de estas columnas variar<61><72>n su longitud. Las longitudes de estas columnas no necesitan ser iguales, y
|
|||
|
no necesitan ser 20. Puedes elegir cualquier longitud entre 1 y 255, cualquiera que te parezca
|
|||
|
razonable. (Si realizar una elecci<63><69>n pobre y resulta que m<><6D>s adelante necesitas un campo mayor,
|
|||
|
MySQL ofrece una sentencia ALTER TABLE.)
|
|||
|
|
|||
|
El sexo del animal puede ser representado en una variedad de formas, por ejemplo, "m" y "f", o
|
|||
|
quiz<EFBFBD><EFBFBD>s "masculino" y "femenino". Es m<><6D>s simple usar un <20><>nico caracter, "m" <20><> "f".
|
|||
|
|
|||
|
El uso del tipo de datos DATE para las columnas de nacimiento y muerte es una opci<63><69>n bastante
|
|||
|
obvia.
|
|||
|
|
|||
|
Ahora que ya has creado una tabla, SHOW TABLES deber<65><72>a producir alguna salida:
|
|||
|
|
|||
|
mysql> SHOW TABLES;
|
|||
|
+---------------+
|
|||
|
| Tables in zoo |
|
|||
|
+---------------+
|
|||
|
| mascota |
|
|||
|
+---------------+
|
|||
|
|
|||
|
Para verificar que tu tabla fue creada de la forma que esperabas, usa una sentencia DESCRIBE:
|
|||
|
|
|||
|
mysql> DESCRIBE mascota;
|
|||
|
+-------------+-------------+------+-----+---------+-------+
|
|||
|
| Field | Type | Null | Key | Default | Extra |
|
|||
|
+-------------+-------------+------+-----+---------+-------+
|
|||
|
| nombre | varchar(20) | YES | | NULL | |
|
|||
|
| propietario | varchar(20) | YES | | NULL | |
|
|||
|
| especie | varchar(20) | YES | | NULL | |
|
|||
|
| sexo | char(1) | YES | | NULL | |
|
|||
|
| nacimiento | date | YES | | NULL | |
|
|||
|
| muerte | date | YES | | NULL | |
|
|||
|
+-------------+-------------+------+-----+---------+-------+
|
|||
|
|
|||
|
Puedes usar DESCRIBE en cualquier momento, por ejemplo, si olvidas los nombres de las columnas de tu
|
|||
|
tabla o a qu<71><75> tipo de datos pertenecen.
|
|||
|
|
|||
|
8.3.3 Cargando datos en una tabla
|
|||
|
=====================================
|
|||
|
|
|||
|
Una vez creada tu tabla, necesitas poblarla. Las sentencias LOAD DATA e INSERT son <20><>tiles para esto.
|
|||
|
|
|||
|
Suponte que tus registros de mascotas pueden ser descritos como se muestra m<><6D>s abajo. (Observa que
|
|||
|
MySQL espera que las fechas se introduzcan en formato AAAA-MM-DD; esto podr<64><72>a ser diferente a lo que
|
|||
|
est<EFBFBD><EFBFBD>s acostumbrado.)
|
|||
|
|
|||
|
nombre propietario especie sexo nacimiento muerte
|
|||
|
Fluffy Harold gato f 1993-02-04
|
|||
|
Claws Gwen gato m 1994-03-17
|
|||
|
Buffy Harold perro f 1989-05-13
|
|||
|
Fang Benny perro m 1990-08-27
|
|||
|
Bowser Diane perro m 1998-08-31 1995-07-29
|
|||
|
Chirpy Gwen p<><70>jaro f 1998-09-11
|
|||
|
Whistler Gwen p<><70>jaro 1997-12-09
|
|||
|
Slim Benny serpiente m 1996-04-29
|
|||
|
|
|||
|
Dado que est<73><74>s comenzando con una tabla vac<61><63>a, una forma sencilla de poblarla consiste en crear un
|
|||
|
fichero de texto conteniendo una fila para cada uno de tus animales, y despu<70><75>s cargar el contenido del
|
|||
|
fichero en la tabla con una sola sentencia.
|
|||
|
|
|||
|
Puedes crear un fichero de texto "mascota.txt" conteniendo un registro por l<><6C>nea, con valores separados
|
|||
|
por tabuladores, y dados en el orden en el que las columnas fueron listadas en la sentencia CREATE
|
|||
|
TABLE. Para valores perdidos (como sexos desconocidos, o fechas de muerte de animales que a<><61>n est<73><74>n
|
|||
|
vivos), puedes usar valores NULL. Para representar estos en tu fichero de texto, use \N. Por
|
|||
|
ejemplo, el registro para Whistler el p<><70>jaro ser<65><72>a algo como esto (donde el espacio en blanco entre
|
|||
|
valores es un simple caracter de tabulaci<63><69>n):
|
|||
|
|
|||
|
Whistler Gwen p<><70>jaro \N 1997-12-09 \N
|
|||
|
|
|||
|
Para cargar el fichero de texto "mascota.txt" en la tabla mascota, usa este comando:
|
|||
|
|
|||
|
mysql> LOAD DATA LOCAL INFILE "mascota.txt" INTO TABLE mascota;
|
|||
|
|
|||
|
Puedes especificar el valor de separaci<63><69>n de columna y el marcador de final de l<><6C>nea expl<70><6C>citamente
|
|||
|
en la sentencia LOAD DATA si lo deseas, pero por defecto equivalen a TAB y LF (intro). Estos valores
|
|||
|
por defecto son suficientes para que la sentencia que lee el fichero "mascota.txt" funcione
|
|||
|
correctamente.
|
|||
|
|
|||
|
Cuando quieras a<><61>adir nuevos registros uno a uno, la sentencia INSERT es muy <20><>til. En su forma m<><6D>s
|
|||
|
simple, ofreces valores para cada columna, en el orden en el que las columnas fueron listadas en la
|
|||
|
sentencia CREATE TABLE. Sup<75><70>n que Diane consige un nuevo hamster llamado Puffball. Podr<64><72>as a<><61>adir un
|
|||
|
nuevo registro usando una sentencia INSERT como esta:
|
|||
|
|
|||
|
mysql> INSERT INTO mascota
|
|||
|
-> VALUES ('Puffball','Diane','hamster','f','1999-03-30',NULL);
|
|||
|
|
|||
|
Observa que los valores string y fecha se espefican encerrados entre comillas. Observa tambi<62><69>n que,
|
|||
|
con INSERT, puedes insertar NULL directamente para representar un valor perdido. No usamos \N como
|
|||
|
hac<EFBFBD><EFBFBD>amos con LOAD DATA.
|
|||
|
|
|||
|
De este ejemplo, deber<65><72>as ser capaz de ver que hubiera dido mucho m<><6D>s costoso teclear todos los
|
|||
|
datos necesarios en la tabla mascota con sentencias INSERT que hacerlo como lo hemos hecho con una
|
|||
|
<EFBFBD><EFBFBD>nica sentencia LOAD DATA.
|
|||
|
|
|||
|
|
|||
|
8.3.4 Extrayendo informaci<63><69>n de una tabla
|
|||
|
===============================================
|
|||
|
|
|||
|
|
|||
|
La sentencia SELECT se usa para recabar informaci<63><69>n de una tabla. La forma
|
|||
|
general de la sentencia es:
|
|||
|
|
|||
|
SELECT qu<71><75>_seleccionar
|
|||
|
FROM de_qu<71><75>_tabla
|
|||
|
WHERE condiciones_a_satisfacer
|
|||
|
|
|||
|
qu<EFBFBD><EFBFBD>_seleccionar indica qu<71><75> es lo que quieres seleccionar. Puede ser una lista de
|
|||
|
columnas, o * para indicar "todas las columnas". de_qu<71><75>_tabla indica la tabla de
|
|||
|
la que quieres extraer datos. La cla<6C><61>sula WHERE es opcional. Si est<73><74> presente,
|
|||
|
condiciones_a_satisfacer especifica las codiciones que las filas deben cumplir
|
|||
|
para estar presentes en el resultado de la selecci<63><69>n.
|
|||
|
|
|||
|
8.3.4.1 Seleccionando todos los datos
|
|||
|
=======================================
|
|||
|
|
|||
|
La forma m<><6D>s simplede SELECT recoge toda la informaci<63><69>n de una tabla:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota;
|
|||
|
+----------+-------------+-----------+------+------------+------------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+----------+-------------+-----------+------+------------+------------+
|
|||
|
| Bluffy | Harold | gato | f | 1993-02-04 | NULL |
|
|||
|
| Claws | Gwen | gato | m | 1994-03-17 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
| Fang | Benny | perro | m | 1990-08-27 | NULL |
|
|||
|
| Bowser | Diane | perro | m | 1998-08-31 | 1995-07-29 |
|
|||
|
| Chirpy | Gwen | p<><70>jaro | f | 1998-09-11 | NULL |
|
|||
|
| Whistler | Gwen | p<><70>jaro | NULL | 1997-12-09 | NULL |
|
|||
|
| Slim | Benny | serpiente | m | 1996-04-29 | NULL |
|
|||
|
| Puffball | Diane | hamster | f | 1999-03-30 | NULL |
|
|||
|
+----------+-------------+-----------+------+------------+------------+
|
|||
|
|
|||
|
Esta forma de SELECT es <20><>til si quieres revisar tu tabla al completo, por
|
|||
|
ejemplo, tras haberla cargado con tu conjunto inicial de datos. Como suele
|
|||
|
suceder, la salida ya muestra un error en tu fichero de datos: Bowser <20><>parece
|
|||
|
haber nacido tras su muerte! Consultando tus papeles originales sobre el
|
|||
|
pedigree del perro, descubres que la fecha correcta de nacimiento es 1989, no
|
|||
|
1998.
|
|||
|
|
|||
|
Existen al menos un par de maneras de arreglar esto:
|
|||
|
|
|||
|
* Edita el fichero "mascota.txt" para corregir el error, despu<70><75>s vac<61><63>a la tabla
|
|||
|
y vuelve a cargarla usando DELETE y LOAD DATA:
|
|||
|
|
|||
|
mysql> DELETE from mascota;
|
|||
|
mysql> LOAD DATA LOCAL INFILE "mascota.txt" INTO TABLE mascota;
|
|||
|
|
|||
|
Sin embargo, si haces esto, debes re-escribir el registro para Puffball.
|
|||
|
|
|||
|
* Arreglar s<><73>lo el registro err<72><72>neo con la sentencia UPDATE:
|
|||
|
|
|||
|
mysql> UPDATE mascota SET nacimiento="1989-08-31" WHERE nombre="Bowser";
|
|||
|
|
|||
|
Como se muestra m<><6D>s arriba, es f<><66>cil recuperar el cuerpo de una data. Pero
|
|||
|
t<EFBFBD><EFBFBD>picamente no querr<72><72>s hacer eso, en particular cuando la tabla sea muy
|
|||
|
larga. Generalmente, estar<61><72>s m<><6D>s interesado en responder a una pregunta en
|
|||
|
particular, en cuyo caso deber<65><72>s especificar algunas restricciones en la
|
|||
|
informaci<EFBFBD><EFBFBD>n que deseas. Veamos algunas consultas de selecci<63><69>n en t<><74>rminos de
|
|||
|
preguntas sobre tus mascotas que se deben responder.
|
|||
|
|
|||
|
8.3.4.2 Seleccionando filas en particular
|
|||
|
=============================================
|
|||
|
|
|||
|
Puedes seleccionar s<><73>lo filas en particular de tu tabla. Por ejemplo, si quieres
|
|||
|
verificar el cambio que has realizado a la fecha de nacimiento de Bowser,
|
|||
|
selecciona el registro de Bowser de la siguiente forma:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre="Bowser";
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
| Bowser | Diane | perro | m | 1989-08-31 | 1995-07-29 |
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
|
|||
|
La salida confirma que el a<><61>o est<73><74> correctamente registrado como 1989, no 1998.
|
|||
|
|
|||
|
Las comparaciones de cadenas de texto son normalmente insensibles a las
|
|||
|
may<EFBFBD><EFBFBD>sculas/min<69><6E>sculas, por lo que puedes especificar el nombre como "bowser",
|
|||
|
"BOWSER", etc. El resultado de la consulta ser<65><72> el mismo.
|
|||
|
|
|||
|
Puedes especificar condiciones en cualquier columna, no s<><73>lo el nombre. Por
|
|||
|
ejemplo, si quisieras saber qu<71><75> animales nacieron a partir de 1998, examina la
|
|||
|
columna nacimiento:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nacimiento >= "1998-1-1";
|
|||
|
+----------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+----------+-------------+---------+------+------------+--------+
|
|||
|
| Chirpy | Gwen | p<><70>jaro | f | 1998-09-11 | NULL |
|
|||
|
| Puffball | Diane | hamster | f | 1999-03-30 | NULL |
|
|||
|
+----------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
Puedes combinar condiciones, por ejemplo, para localizar los perros hembra:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE especie="perro" AND sexo="f";
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
La consulta anterior usa el operador l<><6C>gico AND. Existe tambi<62><69>n un operador OR:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE especie="serpiente" OR especie="p<><70>jaro";
|
|||
|
+----------+-------------+-----------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+----------+-------------+-----------+------+------------+--------+
|
|||
|
| Chirpy | Gwen | p<><70>jaro | f | 1998-09-11 | NULL |
|
|||
|
| Whistler | Gwen | p<><70>jaro | NULL | 1997-12-09 | NULL |
|
|||
|
| Slim | Benny | serpiente | m | 1996-04-29 | NULL |
|
|||
|
+----------+-------------+-----------+------+------------+--------+
|
|||
|
|
|||
|
AND y OR pueden entremezclarse. Si lo haces, es una buena idea el utilizar
|
|||
|
par<EFBFBD><EFBFBD>ntesis para indicar c<><63>mo deber<65><72>an agruparse las condiciones:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE (especie="gato" AND sexo="m")
|
|||
|
-> OR (especie="perro" AND sexo="f");
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| Claws | Gwen | gato | m | 1994-03-17 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
8.3.4.3 Seleccionando columnas en particular
|
|||
|
===================================================
|
|||
|
|
|||
|
Si no quieres ver filas completas de tu tabla, simplemente nombra las columnas
|
|||
|
en las cuales est<73><74>s interesado, separadas por comas. Por ejemplo, si quieres
|
|||
|
saber cu<63><75>ndo nacieron tus animales, selecciona las columnas nombre y nacimiento:
|
|||
|
|
|||
|
mysql> SELECT nombre, nacimiento FROM mascota;
|
|||
|
+----------+------------+
|
|||
|
| nombre | nacimiento |
|
|||
|
+----------+------------+
|
|||
|
| Bluffy | 1993-02-04 |
|
|||
|
| Claws | 1994-03-17 |
|
|||
|
| Buffy | 1989-05-13 |
|
|||
|
| Fang | 1990-08-27 |
|
|||
|
| Bowser | 1989-08-31 |
|
|||
|
| Chirpy | 1998-09-11 |
|
|||
|
| Whistler | 1997-12-09 |
|
|||
|
| Slim | 1996-04-29 |
|
|||
|
| Puffball | 1999-03-30 |
|
|||
|
+----------+------------+
|
|||
|
|
|||
|
Para averiguar qui<75><69>n posee mascotas, usa esta consulta:
|
|||
|
|
|||
|
mysql> SELECT propietario FROM mascota;
|
|||
|
+-------------+
|
|||
|
| propietario |
|
|||
|
+-------------+
|
|||
|
| Harold |
|
|||
|
| Gwen |
|
|||
|
| Harold |
|
|||
|
| Benny |
|
|||
|
| Diane |
|
|||
|
| Gwen |
|
|||
|
| Gwen |
|
|||
|
| Benny |
|
|||
|
| Diane |
|
|||
|
+-------------+
|
|||
|
|
|||
|
Sin embargo, observa que la consulta simplemente obtiene el campo propietario de
|
|||
|
cada registro, y algunos de ellos aparecen m<><6D>s de una vez. Para minimizar la
|
|||
|
salida, obt<62><74>n cada registro de salida <20><>nico una sola vez a<><61>adiendo la palabra
|
|||
|
reservada DISTINCT:
|
|||
|
|
|||
|
mysql> SELECT DISTINCT propietario FROM mascota;
|
|||
|
+-------------+
|
|||
|
| propietario |
|
|||
|
+-------------+
|
|||
|
| Benny |
|
|||
|
| Diane |
|
|||
|
| Gwen |
|
|||
|
| Harold |
|
|||
|
+-------------+
|
|||
|
|
|||
|
Puedes usar una cla<6C><61>sula WHERE para combinar la selecci<63><69>n de filas con la
|
|||
|
selecci<EFBFBD><EFBFBD>n de columnas. Por ejemplo, para conseguir s<><73>lo las fechas de nacimiento
|
|||
|
de perros y gatos, usa esta consulta:
|
|||
|
|
|||
|
mysql> SELECT nombre, especie, nacimiento FROM mascota
|
|||
|
-> WHERE especie = "perro" OR especie = "gato";
|
|||
|
+--------+---------+------------+
|
|||
|
| nombre | especie | nacimiento |
|
|||
|
+--------+---------+------------+
|
|||
|
| Bluffy | gato | 1993-02-04 |
|
|||
|
| Claws | gato | 1994-03-17 |
|
|||
|
| Buffy | perro | 1989-05-13 |
|
|||
|
| Fang | perro | 1990-08-27 |
|
|||
|
| Bowser | perro | 1989-08-31 |
|
|||
|
+--------+---------+------------+
|
|||
|
|
|||
|
8.3.4.4 Ordenando filas
|
|||
|
===========================
|
|||
|
|
|||
|
Tal vez hayas observado que en los ejemplos anteriores las filas del resultado
|
|||
|
se muestran sin ning<6E><67>n tipo de orden en particular. Sin embargo, a menudo es m<><6D>s
|
|||
|
f<EFBFBD><EFBFBD>cil de examinar la salida de una consulta cuando las filas est<73><74>n ordenadas de
|
|||
|
alguna manera en particular. Para ordenar un resultado, usa la cla<6C><61>sula ORDER
|
|||
|
BY.
|
|||
|
|
|||
|
Aqu<EFBFBD><EFBFBD> mostramos las fechas de nacimiento de los animales, ordenadas por fecha:
|
|||
|
|
|||
|
mysql> SELECT nombre, nacimiento FROM mascota ORDER BY nacimiento;
|
|||
|
+----------+------------+
|
|||
|
| nombre | nacimiento |
|
|||
|
+----------+------------+
|
|||
|
| Buffy | 1989-05-13 |
|
|||
|
| Bowser | 1989-08-31 |
|
|||
|
| Fang | 1990-08-27 |
|
|||
|
| Bluffy | 1993-02-04 |
|
|||
|
| Claws | 1994-03-17 |
|
|||
|
| Slim | 1996-04-29 |
|
|||
|
| Whistler | 1997-12-09 |
|
|||
|
| Chirpy | 1998-09-11 |
|
|||
|
| Puffball | 1999-03-30 |
|
|||
|
+----------+------------+
|
|||
|
|
|||
|
Para ordenar de forma inversa, a<><61>ade la palabra reservada DESC (descendente) al
|
|||
|
nombre de la columna por la que est<73><74>s ordenando:
|
|||
|
|
|||
|
mysql> SELECT nombre, nacimiento FROM mascota ORDER BY nacimiento DESC;
|
|||
|
+----------+------------+
|
|||
|
| nombre | nacimiento |
|
|||
|
+----------+------------+
|
|||
|
| Puffball | 1999-03-30 |
|
|||
|
| Chirpy | 1998-09-11 |
|
|||
|
| Whistler | 1997-12-09 |
|
|||
|
| Slim | 1996-04-29 |
|
|||
|
| Claws | 1994-03-17 |
|
|||
|
| Bluffy | 1993-02-04 |
|
|||
|
| Fang | 1990-08-27 |
|
|||
|
| Bowser | 1989-08-31 |
|
|||
|
| Buffy | 1989-05-13 |
|
|||
|
+----------+------------+
|
|||
|
|
|||
|
Puedes ordenar por m<><6D>ltiples columnas. Por ejemplo, para ordenar por tipo de
|
|||
|
animal, despu<70><75>s por fecha de nacimiento dentro del mismo tipo de animal estando
|
|||
|
los animales m<><6D>s j<><6A>venes primero, usa la siguiente consulta:
|
|||
|
|
|||
|
mysql> SELECT nombre, especie, nacimiento FROM mascota ORDER BY especie, nacimiento DESC;
|
|||
|
+----------+-----------+------------+
|
|||
|
| nombre | especie | nacimiento |
|
|||
|
+----------+-----------+------------+
|
|||
|
| Claws | gato | 1994-03-17 |
|
|||
|
| Bluffy | gato | 1993-02-04 |
|
|||
|
| Puffball | hamster | 1999-03-30 |
|
|||
|
| Chirpy | p<><70>jaro | 1998-09-11 |
|
|||
|
| Whistler | p<><70>jaro | 1997-12-09 |
|
|||
|
| Fang | perro | 1990-08-27 |
|
|||
|
| Bowser | perro | 1989-08-31 |
|
|||
|
| Buffy | perro | 1989-05-13 |
|
|||
|
| Slim | serpiente | 1996-04-29 |
|
|||
|
+----------+-----------+------------+
|
|||
|
|
|||
|
Observa que la palabra reservada DESC se aplica s<><73>lo al nombre de columna que
|
|||
|
preceda a la palabra reservada (nacimiento); los valores especie siguen siendo
|
|||
|
ordenados en forma ascendente.
|
|||
|
|
|||
|
8.3.4.5 C<><43>lculos de fecha
|
|||
|
============================
|
|||
|
|
|||
|
MySQL ofrece muchas funciones que puedes usar para realizar c<><63>lculos con fechas,
|
|||
|
por ejemplo, para calcular edades o extraer partes de fechas.
|
|||
|
|
|||
|
Para determinar cuantos a<><61>os tiene cada una de tus mascotas, puedes calcular la
|
|||
|
edad como la diferencia entre la fecha de nacimiento y la fecha actual. Puedes
|
|||
|
hacerlo convirtiendo las dos fechas a dias, coge la diferencia, y div<69><76>dela por
|
|||
|
365 (el n<><6E>mero de dias en un a<><61>o):
|
|||
|
|
|||
|
mysql> select nombre, (TO_DAYS(NOW())-TO_DAYS(nacimiento))/365 FROM mascota;
|
|||
|
+----------+------------------------------------------+
|
|||
|
| nombre | (TO_DAYS(NOW())-TO_DAYS(nacimiento))/365 |
|
|||
|
+----------+------------------------------------------+
|
|||
|
| Bluffy | 6.94 |
|
|||
|
| Claws | 5.83 |
|
|||
|
| Buffy | 10.68 |
|
|||
|
| Fang | 9.39 |
|
|||
|
| Bowser | 10.38 |
|
|||
|
| Chirpy | 1.34 |
|
|||
|
| Whistler | 2.10 |
|
|||
|
| Slim | 3.71 |
|
|||
|
| Puffball | 0.79 |
|
|||
|
+----------+------------------------------------------+
|
|||
|
|
|||
|
Aunque la consulta funcione, existen algunos puntos que podr<64><72>an ser
|
|||
|
mejorados. Primero, el resultado podr<64><72>a ser revisado m<><6D>s f<><66>cilmente si las filas
|
|||
|
se presentaran ordenadas de alguna manera. Segundo, la cabecera de la columna
|
|||
|
edad no es muy significativa.
|
|||
|
|
|||
|
El primer problema puede ser solucionado a<><61>adiendo una cl<63><6C>usula ORDER BY nombre
|
|||
|
para ordenar la salida por nombre. Para arreglar el tema del encabezamiento de
|
|||
|
columna, puedes darle un nombre a dicha columna de tal forma que aparezca una
|
|||
|
etiqueta diferente en la salida (esto es lo que se llama un alias de columna):
|
|||
|
|
|||
|
mysql> select nombre, (TO_DAYS(NOW())-TO_DAYS(nacimiento))/365 AS edad
|
|||
|
-> FROM mascota ORDER BY nombre;
|
|||
|
+----------+-------+
|
|||
|
| nombre | edad |
|
|||
|
+----------+-------+
|
|||
|
| Bluffy | 6.94 |
|
|||
|
| Bowser | 10.38 |
|
|||
|
| Buffy | 10.68 |
|
|||
|
| Chirpy | 1.34 |
|
|||
|
| Claws | 5.83 |
|
|||
|
| Fang | 9.39 |
|
|||
|
| Puffball | 0.79 |
|
|||
|
| Slim | 3.71 |
|
|||
|
| Whistler | 2.10 |
|
|||
|
+----------+-------+
|
|||
|
|
|||
|
Para ordenar la salida por edad en lugar de por nombre, puedes hacerlo usando
|
|||
|
s<EFBFBD><EFBFBD>mplemente una cl<63><6C>usula ORDER BY diferente:
|
|||
|
|
|||
|
mysql> select nombre, (TO_DAYS(NOW())-TO_DAYS(nacimiento))/365 AS edad
|
|||
|
-> FROM mascota ORDER BY edad;
|
|||
|
+----------+-------+
|
|||
|
| nombre | edad |
|
|||
|
+----------+-------+
|
|||
|
| Puffball | 0.79 |
|
|||
|
| Chirpy | 1.34 |
|
|||
|
| Whistler | 2.10 |
|
|||
|
| Slim | 3.71 |
|
|||
|
| Claws | 5.83 |
|
|||
|
| Bluffy | 6.94 |
|
|||
|
| Fang | 9.39 |
|
|||
|
| Bowser | 10.38 |
|
|||
|
| Buffy | 10.68 |
|
|||
|
+----------+-------+
|
|||
|
|
|||
|
Puede usarse una consulta similar para determinar la edad de la muerte para los
|
|||
|
animales que hayan muerto. Puedes determinar qu<71><75> animales son estos comprobando
|
|||
|
si el valor muerte es NULL o no. Despu<70><75>s, para aquellos que no tengan un valor
|
|||
|
NULL, calcular la diferencia entre los valores muerte y nacimiento:
|
|||
|
|
|||
|
mysql> select nombre, nacimiento, muerte,
|
|||
|
-> (TO_DAYS(NOW())-TO_DAYS(nacimiento))/365 AS edad
|
|||
|
-> FROM mascota WHERE muerte IS NOT NULL ORDER BY edad;
|
|||
|
+--------+------------+------------+-------+
|
|||
|
| nombre | nacimiento | muerte | edad |
|
|||
|
+--------+------------+------------+-------+
|
|||
|
| Bowser | 1989-08-31 | 1995-07-29 | 10.38 |
|
|||
|
+--------+------------+------------+-------+
|
|||
|
|
|||
|
La consulta usa muerte IS NOT NULL en lugar de muerte != NULL dado que NULL es
|
|||
|
un valor especial. Esto se explica m<><6D>s adelante. [Puedes consultar la secci<63><69>n
|
|||
|
[Working with NULL] del manual de MySQL.
|
|||
|
|
|||
|
<EFBFBD><EFBFBD>Qu<EFBFBD><EFBFBD> har<61><72>as si quisieras saber qu<71><75> animales cumplen a<><61>os el mes que viene? Para
|
|||
|
este tipo de c<><63>lculos, a<><61>o y d<><64>a son irrelevantes, simplemente querr<72><72>s extraer
|
|||
|
la parte mes de la columna nacimiento. MySQL ofrece muchas funciones de
|
|||
|
extracci<EFBFBD><EFBFBD>n de parte-de-fecha, como YEAR(),MONTH() y DAY(). La funci<63><69>n apropiada
|
|||
|
para nuestro problema es MONTH(). Para ver c<><63>mo funciona, ejecuta una consulta
|
|||
|
r<EFBFBD><EFBFBD>pida que muestre el valor de la fecha de nacimiento y el mes de nacimiento
|
|||
|
(MONTH(nacimiento)):
|
|||
|
|
|||
|
mysql> SELECT nombre, nacimiento, MONTH(nacimiento) FROM mascota;
|
|||
|
+----------+------------+-------------------+
|
|||
|
| nombre | nacimiento | MONTH(nacimiento) |
|
|||
|
+----------+------------+-------------------+
|
|||
|
| Bluffy | 1993-02-04 | 2 |
|
|||
|
| Claws | 1994-03-17 | 3 |
|
|||
|
| Buffy | 1989-05-13 | 5 |
|
|||
|
| Fang | 1990-08-27 | 8 |
|
|||
|
| Bowser | 1989-08-31 | 8 |
|
|||
|
| Chirpy | 1998-09-11 | 9 |
|
|||
|
| Whistler | 1997-12-09 | 12 |
|
|||
|
| Slim | 1996-04-29 | 4 |
|
|||
|
| Puffball | 1999-03-30 | 3 |
|
|||
|
+----------+------------+-------------------+
|
|||
|
|
|||
|
Buscar animales que hayan nacido en el mes pr<70><72>ximo es tambi<62><69>n sencillo de
|
|||
|
realizar. Suponte que Abril es el mes actual. Entonces el valor del mes es 4 y
|
|||
|
lo que buscas son animales nacidos en Mayo (mes 5):
|
|||
|
|
|||
|
mysql> SELECT nombre, nacimiento FROM mascota WHERE MONTH(nacimiento) = 5;
|
|||
|
+--------+------------+
|
|||
|
| nombre | nacimiento |
|
|||
|
+--------+------------+
|
|||
|
| Buffy | 1989-05-13 |
|
|||
|
+--------+------------+
|
|||
|
|
|||
|
Existe una peque<75><65>a complicaci<63><69>n si el mes actual es Diciembre, por supuesto. No
|
|||
|
puedes a<><61>adir simplemente uno al n<><6E>mero de mes (12) y buscar animales nacidos en
|
|||
|
el mes 13, dado que no existe tal mes. En lugar de eso, debes buscar animales
|
|||
|
nacidos en Enero (mes 1).
|
|||
|
|
|||
|
Puedes escribir la consulta de tal forma que funcione independientemente del mes
|
|||
|
en el que estemos. De esa forma no tendr<64><72>s que usar un n<><6E>mero de mes en
|
|||
|
particular en la consulta. DATE_ADD() te permite a<><61>adir un intervalo de tiempo a
|
|||
|
una fecha dada. Si a<><61>ades un mes al valor de NOW(), y despu<70><75>s extraes la parte
|
|||
|
del mes con MONTH(), el resultado produce el mes del cumplea<65><61>os que buscamos:
|
|||
|
|
|||
|
|
|||
|
mysql> select NOW();
|
|||
|
+---------------------+
|
|||
|
| NOW() |
|
|||
|
+---------------------+
|
|||
|
| 2000-01-13 18:13:09 |
|
|||
|
+---------------------+
|
|||
|
|
|||
|
mysql> SELECT nombre, nacimiento FROM mascota
|
|||
|
-> WHERE MONTH(nacimiento) = MONTH(DATE_ADD(NOW(),INTERVAL 1 MONTH));
|
|||
|
+--------+------------+
|
|||
|
| nombre | nacimiento |
|
|||
|
+--------+------------+
|
|||
|
| Bluffy | 1993-02-04 |
|
|||
|
+--------+------------+
|
|||
|
|
|||
|
Una manera difente de conseguir los mismos resultados es a<><61>adir 1 al mes actual
|
|||
|
para conseguir el mes siguiente (tras usar la funci<63><69>n m<><6D>dulo (MOD) para
|
|||
|
convertir el valor de mes actual en 0 si estamos en Diciembre (mes 12)):
|
|||
|
|
|||
|
mysql> SELECT nombre, nacimiento FROM mascota
|
|||
|
-> WHERE MONTH(nacimiento) = MOD(MONTH(NOW()),12) +1;
|
|||
|
+--------+------------+
|
|||
|
| nombre | nacimiento |
|
|||
|
+--------+------------+
|
|||
|
| Bluffy | 1993-02-04 |
|
|||
|
+--------+------------+
|
|||
|
|
|||
|
|
|||
|
8.3.4.6 Trabajando con valores NULL
|
|||
|
=======================================
|
|||
|
|
|||
|
Los valores NULL pueden ser sorprenderte hasta que te acostumbras a
|
|||
|
usarlos. Conceptualmente, NULL significa "valor perdido" o "valor desconocido" y
|
|||
|
se trata de forma diferente que otros valores. Para realizar comparaciones
|
|||
|
respecto a NULL, no puedes utilizar los operadores de comparaci<63><69>n aritm<74><6D>ticos
|
|||
|
como =, < o != . Puedes realizar una demostraci<63><69>n de esto, prueba la siguiente consulta:
|
|||
|
|
|||
|
mysql> SELECT 1 = NULL, 1 != NULL, 1 < NULL, 1 > NULL;
|
|||
|
+----------+-----------+----------+----------+
|
|||
|
| 1 = NULL | 1 != NULL | 1 < NULL | 1 > NULL |
|
|||
|
+----------+-----------+----------+----------+
|
|||
|
| NULL | NULL | NULL | NULL |
|
|||
|
+----------+-----------+----------+----------+
|
|||
|
|
|||
|
Ciertamente, de estas comparaciones no se pueden extraer resultados
|
|||
|
significativos. Para conseguirlo, usa los operadores IS NULL y IS NOT NULL:
|
|||
|
|
|||
|
mysql> SELECT 1 IS NULL, 1 IS NOT NULL;
|
|||
|
+-----------+---------------+
|
|||
|
| 1 IS NULL | 1 IS NOT NULL |
|
|||
|
+-----------+---------------+
|
|||
|
| 0 | 1 |
|
|||
|
+-----------+---------------+
|
|||
|
|
|||
|
En MySQL, 0 significa falso y 1 significa VERDADERO.
|
|||
|
|
|||
|
Este tratamiento especial de NULL fue la causa de que en la secci<63><69>n anterior
|
|||
|
fuera necesario determinar qu<71><75> animales ya no viv<69><76>an usando "muerte IS NOT NULL"
|
|||
|
en lugar de "muerte != NULL".
|
|||
|
|
|||
|
8.3.4.7 Asociaci<63><69>n/Coincidencia de patrones (PATTERN MATCHING)
|
|||
|
================================================================
|
|||
|
|
|||
|
MySQL ofrece las caracter<65><72>sticas de asociaci<63><69>n de patrones est<73><74>ndar as<61><73> como
|
|||
|
una forma de coincidencia de patrones basadas en expresiones regulares
|
|||
|
extendidas similares a las usadas por utilidades UNIX como vi, grep y sed.
|
|||
|
|
|||
|
La asociaci<63><69>n de patrones SQL te permite usar '_' para asociar cualquier
|
|||
|
caracter simple, y '%' para asociar un n<><6E>mero arbitrario de caracteres
|
|||
|
(incluyendo cero caracteres). Los patrones SQL no toman en cuenta las
|
|||
|
diferencias entre may<61><79>sculas y min<69><6E>sculas. Se muestran debajo algunos
|
|||
|
ejemplos. Observa que no se utiliza = o != en el trabajo con patrones SQL;
|
|||
|
utiliza en su lugar los operadores de comparaci<63><69>n LIKE o NOT LIKE.
|
|||
|
|
|||
|
Para buscar nombres que comienzan por "b":
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre LIKE "b%";
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
| Bluffy | Harold | gato | f | 1993-02-04 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
| Bowser | Diane | perro | m | 1989-08-31 | 1995-07-29 |
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
|
|||
|
Para buscar nombres que terminen por "fy":
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre LIKE "%fy";
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| Bluffy | Harold | gato | f | 1993-02-04 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
Para buscar nombres que contengan una "w":
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre LIKE "%w%";
|
|||
|
+----------+-------------+---------+------+------------+------------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+----------+-------------+---------+------+------------+------------+
|
|||
|
| Claws | Gwen | gato | m | 1994-03-17 | NULL |
|
|||
|
| Bowser | Diane | perro | m | 1989-08-31 | 1995-07-29 |
|
|||
|
| Whistler | Gwen | p<><70>jaro | NULL | 1997-12-09 | NULL |
|
|||
|
+----------+-------------+---------+------+------------+------------+
|
|||
|
|
|||
|
Para buscar nombres de longitud cinco caracteres, usa el patr<74><72>n "_" :
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre LIKE "_____";
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| Claws | Gwen | gato | m | 1994-03-17 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
El otro tipo de asociaci<63><69>n de patrones ofrecido por MySQL utiliza expresiones
|
|||
|
regulares extendidas. Cuando se realiza una comprobaci<63><69>n buscando una coincidencia
|
|||
|
para este tipo de patr<74><72>n, se deben usar los operadores REGEXP y NOT REGEXP (o
|
|||
|
RLIKE y NOT RLIKE, dado que son sin<69><6E>nimos).
|
|||
|
|
|||
|
Algunas caracter<65><72>sticas de las expresiones regulares extendidas son:
|
|||
|
|
|||
|
* `.' se asocia con cualquier caracter (pero s<><73>lo uno)
|
|||
|
|
|||
|
* Una clase de caracteres `[...]' se asocia con culquier caracter contenido
|
|||
|
dentro de los corchetes. Por ejemplo, `[abc]' se asocia con 'a', 'b' <20><>
|
|||
|
|
|||
|
* 'c'. Para nombrar un rango de caracteres, usa un gui<75><69>n. `[a-z]' se asocia con
|
|||
|
cualquier letra en min<69><6E>scula, donde '[0-9]' se asocia con cualquier d<><64>gito.
|
|||
|
|
|||
|
* '*' se asocia con 0 o m<><6D>s instancias de lo que preceda al asterisco. Por
|
|||
|
ejemplo,'a*' coincide con cualquier n<><6E>mero de a's,'[0-9]*' se asocia con
|
|||
|
cualquier n<><6E>mero de d<><64>gitos, y '.*' se asocia con cualquier cosa.
|
|||
|
|
|||
|
* Las expresiones regulares son sensibles a las may<61><79>sculas/min<69><6E>sculas, pero
|
|||
|
puedes utilizar una clase caracter para asociar ambos casos si los deseas. Por
|
|||
|
ejemplo, '[aA]' coincide tanto con la letra a min<69><6E>scula como con la letra A
|
|||
|
may<61><79>scula y '[a-zA-Z]' coincide con cualquier letra en cualquier modo
|
|||
|
may<61><79>scula/min<69><6E>scula.
|
|||
|
|
|||
|
* El patr<74><72>n se asocia si ocurre en cualquier lugar dentro del valor a ser
|
|||
|
probado (los patrones SQL coinciden s<><73>lo si se asocian con el valor
|
|||
|
completo).
|
|||
|
|
|||
|
* Para anclar un patr<74><72>n de manera que se busque la coincidencia bien al comienzo
|
|||
|
o bien al final del valor a ser comprobado, usa '^' al comienzo del patr<74><72>n o
|
|||
|
'$' al final del patr<74><72>n, respectivamente.
|
|||
|
|
|||
|
Para demostrar c<><63>mo funcionan las expresiones regulares, las consultas LIKE
|
|||
|
mostradas antes son reescritas debajo para usar REGEXP:
|
|||
|
|
|||
|
Para buscar nombres que comiencen por "b", usa '^' para buscar la coincidencia
|
|||
|
al comienzo del nombre y '[bB]' para buscar la asociaci<63><69>n tanto con la b
|
|||
|
min<EFBFBD><EFBFBD>scula como con la b may<61><79>scula:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre REGEXP "^[bB]";
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
| Bluffy | Harold | gato | f | 1993-02-04 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
| Bowser | Diane | perro | m | 1989-08-31 | 1995-07-29 |
|
|||
|
+--------+-------------+---------+------+------------+------------+
|
|||
|
|
|||
|
Para buscar nombres que terminen por "fy", usa "$" para buscar la coincidencia
|
|||
|
al final del nombre:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre REGEXP "fy$";
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| Bluffy | Harold | gato | f | 1993-02-04 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
Para buscar nombres que contengan una "w", utiliza "[wW]" para buscar la
|
|||
|
asociaci<EFBFBD><EFBFBD>n tanto en may<61><79>sculas como min<69><6E>sculas:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre REGEXP "[wW]";
|
|||
|
+----------+-------------+---------+------+------------+------------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+----------+-------------+---------+------+------------+------------+
|
|||
|
| Claws | Gwen | gato | m | 1994-03-17 | NULL |
|
|||
|
| Bowser | Diane | perro | m | 1989-08-31 | 1995-07-29 |
|
|||
|
| Whistler | Gwen | p<><70>jaro | NULL | 1997-12-09 | NULL |
|
|||
|
+----------+-------------+---------+------+------------+------------+
|
|||
|
|
|||
|
Dado que un patr<74><72>n de una expresi<73><69>n regular se asocia si ocurre en cualquier
|
|||
|
lugar del valor, no es necesario poner un caracter comod<6F><64>n en ning<6E><67>n lado del
|
|||
|
patr<EFBFBD><EFBFBD>n para conseguir que se asocie con el valor completo como har<61><72>as si usaras
|
|||
|
un patr<74><72>n SQL.
|
|||
|
|
|||
|
Para buscar nombres conteniendo exactamente cinco caracteres, usa "^" y "$" para
|
|||
|
asociar el comienzo y el final de un nombre, y cinco instancias de "." entre
|
|||
|
ellos:
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre REGEXP "^.....$";
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| Claws | Gwen | gato | m | 1994-03-17 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
Tambi<EFBFBD><EFBFBD>n podr<64><72>as haber escrito la consulta anterior usando el operador '{n}'
|
|||
|
"repetir n veces":
|
|||
|
|
|||
|
mysql> SELECT * FROM mascota WHERE nombre REGEXP "^.{5}$";
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| nombre | propietario | especie | sexo | nacimiento | muerte |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
| Claws | Gwen | gato | m | 1994-03-17 | NULL |
|
|||
|
| Buffy | Harold | perro | f | 1989-05-13 | NULL |
|
|||
|
+--------+-------------+---------+------+------------+--------+
|
|||
|
|
|||
|
|
|||
|
8.3.4.8 Contando filas
|
|||
|
=======================
|
|||
|
|
|||
|
Las bases de datos son usadas a menudo para responder a la pregunta, "<22><>cuantas
|
|||
|
veces aparece un determinado tipo de datos en una tabla?". Por ejemplo, podr<64><72>as
|
|||
|
querer saber cu<63><75>ntas mascotas tienes, o cu<63><75>ntas mascotas tiene cada propietario,
|
|||
|
o podr<64><72>as querer realizar varios tipos de censos respecto a tus animales.
|
|||
|
|
|||
|
Contar el n<><6E>mero total de animales que tienes es lo mismo que preguntar
|
|||
|
"<22><>cu<63><75>ntas filas hay en la tabla mascota?", dado que hay s<><73>lo una fila por
|
|||
|
mascota. La funci<63><69>n COUNT() cuenta el n<><6E>mero de resultados no-NULL , as<61><73> pues,
|
|||
|
la consulta a realizar para contar el n<><6E>mero de animales tiene la siguiente forma:
|
|||
|
|
|||
|
mysql> SELECT COUNT(*) FROM mascota;
|
|||
|
+----------+
|
|||
|
| COUNT(*) |
|
|||
|
+----------+
|
|||
|
| 9 |
|
|||
|
+----------+
|
|||
|
|
|||
|
Antes, conseguiste los nombres de las personas que poseen una mascota. Puedes
|
|||
|
usar COUNT() para averiguar cu<63><75>ntas mascotas tiene cada propietario:
|
|||
|
|
|||
|
mysql> SELECT propietario, COUNT(*) FROM mascota GROUP BY propietario;
|
|||
|
+-------------+----------+
|
|||
|
| propietario | COUNT(*) |
|
|||
|
+-------------+----------+
|
|||
|
| Benny | 2 |
|
|||
|
| Diane | 2 |
|
|||
|
| Gwen | 3 |
|
|||
|
| Harold | 2 |
|
|||
|
+-------------+----------+
|
|||
|
|
|||
|
Observa el uso de GROUP BY para agrupar todos los registros de cada
|
|||
|
propietario. Si no lo hubi<62><69>ramos puesto, todo lo que conseguirias ser<65><72>a un
|
|||
|
mensaje de error:
|
|||
|
|
|||
|
mysql> SELECT propietario, COUNT(propietario) FROM mascota;
|
|||
|
ERROR 1140: Mixing of GROUP columns (MIN(),MAX(),COUNT()...) with no GROUP
|
|||
|
columns is illegal if there is no GROUP BY clause
|
|||
|
|
|||
|
COUNT() y GROUP BY son <20><>tiles para la caracterizaci<63><69>n de tus datos de varias
|
|||
|
formas. Los siguientes ejemplos muestran difentes maneras para realizar
|
|||
|
operaciones de censo animal.
|
|||
|
|
|||
|
N<EFBFBD><EFBFBD>mero de animales por especies:
|
|||
|
|
|||
|
mysql> SELECT especie, COUNT(*) FROM mascota GROUP BY especie;
|
|||
|
+-----------+----------+
|
|||
|
| especie | COUNT(*) |
|
|||
|
+-----------+----------+
|
|||
|
| gato | 2 |
|
|||
|
| hamster | 1 |
|
|||
|
| p<><70>jaro | 2 |
|
|||
|
| perro | 3 |
|
|||
|
| serpiente | 1 |
|
|||
|
+-----------+----------+
|
|||
|
|
|||
|
N<EFBFBD><EFBFBD>mero de animales por sexo:
|
|||
|
|
|||
|
mysql> SELECT sexo , COUNT(*) FROM mascota GROUP BY sexo;
|
|||
|
+------+----------+
|
|||
|
| sexo | COUNT(*) |
|
|||
|
+------+----------+
|
|||
|
| NULL | 1 |
|
|||
|
| f | 4 |
|
|||
|
| m | 4 |
|
|||
|
+------+----------+
|
|||
|
|
|||
|
(En este resultado, NULL indica "sexo desconocido")
|
|||
|
|
|||
|
El n<><6E>mero de animales por combinaci<63><69>n de especies y sexo:
|
|||
|
|
|||
|
mysql> SELECT especie , sexo, COUNT(*) FROM mascota GROUP BY especie, sexo;
|
|||
|
+-----------+------+----------+
|
|||
|
| especie | sexo | COUNT(*) |
|
|||
|
+-----------+------+----------+
|
|||
|
| gato | f | 1 |
|
|||
|
| gato | m | 1 |
|
|||
|
| hamster | f | 1 |
|
|||
|
| p<><70>jaro | NULL | 1 |
|
|||
|
| p<><70>jaro | f | 1 |
|
|||
|
| perro | f | 1 |
|
|||
|
| perro | m | 2 |
|
|||
|
| serpiente | m | 1 |
|
|||
|
+-----------+------+----------+
|
|||
|
|
|||
|
No necesitas recuperar una tabla completa cuando uses COUNT(). Por ejemplo, la
|
|||
|
consulta anterior, cuando se realiza s<><73>lo sobre perros y gatos, se escribe as<61><73>:
|
|||
|
|
|||
|
mysql> SELECT especie , sexo, COUNT(*) FROM mascota
|
|||
|
-> WHERE especie = "perro" OR especie = "gato"
|
|||
|
-> GROUP BY especie, sexo;
|
|||
|
+---------+------+----------+
|
|||
|
| especie | sexo | COUNT(*) |
|
|||
|
+---------+------+----------+
|
|||
|
| gato | f | 1 |
|
|||
|
| gato | m | 1 |
|
|||
|
| perro | f | 1 |
|
|||
|
| perro | m | 2 |
|
|||
|
+---------+------+----------+
|
|||
|
|
|||
|
O, si quieres conocer el n<><6E>mero de animales por sexo s<><73>lo para animales de sexo
|
|||
|
conocido:
|
|||
|
|
|||
|
mysql> SELECT especie , sexo, COUNT(*) FROM mascota
|
|||
|
-> WHERE sexo IS NOT NULL
|
|||
|
-> GROUP BY especie, sexo;
|
|||
|
+-----------+------+----------+
|
|||
|
| especie | sexo | COUNT(*) |
|
|||
|
+-----------+------+----------+
|
|||
|
| gato | f | 1 |
|
|||
|
| gato | m | 1 |
|
|||
|
| hamster | f | 1 |
|
|||
|
| p<><70>jaro | f | 1 |
|
|||
|
| perro | f | 1 |
|
|||
|
| perro | m | 2 |
|
|||
|
| serpiente | m | 1 |
|
|||
|
+-----------+------+----------+
|
|||
|
|
|||
|
|
|||
|
8.3.5 Usando m<><6D>s de una tabla
|
|||
|
===============================
|
|||
|
|
|||
|
La tabla mascota guarda datos sobre las mascotas que posees. Si quieres guardar
|
|||
|
otra informaci<63><69>n sobre ellos, como eventos en sus vidas, visitas al veterinario
|
|||
|
o cu<63><75>ndo han tenido hermanos, necesitas otra tabla. <20><>C<EFBFBD><43>mo deber<65><72>a ser esta otra
|
|||
|
tabla?
|
|||
|
|
|||
|
* Deber<65><72> contener el nombre de la mascota de tal forma que pudieras saber a qu<71><75>
|
|||
|
animal corresponde cada evento almacenado en la misma.
|
|||
|
|
|||
|
* Necesitar<61><72> una fecha para conocer cu<63><75>ndo ocurri<72><69> el evento.
|
|||
|
|
|||
|
* Necesitar<61><72> un campo para describir el evento
|
|||
|
|
|||
|
* Si quieres ser capaz de categorizar los eventos, ser<65><72>a <20><>til tener un campo de
|
|||
|
tipo evento.
|
|||
|
|
|||
|
Dadas estas consideraciones, la sentencia CREATE TABLE para la tabla "evento" se
|
|||
|
parecer<EFBFBD><EFBFBD>a a esto:
|
|||
|
|
|||
|
mysql> CREATE TABLE evento (nombre VARCHAR(20), fecha DATE,
|
|||
|
-> tipo VARCHAR(15), anotaci<63><69>n VARCHAR(255));
|
|||
|
|
|||
|
Como ocurr<72><72>a con la tabla mascota, es m<><6D>s f<><66>cil cargar los registros iniciales
|
|||
|
creando un fichero de texto delimitado por tabuladores conteniendo la
|
|||
|
informaci<EFBFBD><EFBFBD>n:
|
|||
|
|
|||
|
Fluffy 1995-05-15 parto 4 cachorros, 3 hembras, 1 macho
|
|||
|
Buffy 1993-06-23 parto 5 cachorros, 2 hembras, 3 machos
|
|||
|
Buffy 1994-06-19 parto 3 cachorros, 3 hembras
|
|||
|
Chirpy 1999-03-21 veterinario necesit<69><74> enderezamiento de pico
|
|||
|
Slim 1997-08-03 veterinario costilla rota
|
|||
|
Bowser 1991-10-12 perrera
|
|||
|
Fang 1991-10-12 perrera
|
|||
|
Fang 1998-08-28 cumplea<65><61>os Se le regala un nuevo juguete de goma
|
|||
|
Claws 1998-03-17 cumplea<65><61>os Se le regala un nuevo collar de pulgas
|
|||
|
Whistler 1998-12-09 cumplea<65><61>os Primer cumplea<65><61>os
|
|||
|
|
|||
|
|
|||
|
Carga los registros as<61><73>:
|
|||
|
|
|||
|
mysql> LOAD DATA LOCAL INFILE "evento.txt" INTO TABLE evento;
|
|||
|
|
|||
|
Bas<EFBFBD><EFBFBD>ndote en lo que has aprendido de las consultas que has ejecutado em la tabla
|
|||
|
mascota, deber<65><72>as ser capaz de realizar recuperaciones de datos en los registros
|
|||
|
de la tabla "evento"; los principios son los mismos. <20><>Pero qu<71><75> hacer cuando la
|
|||
|
tabla evento no sea suficiente por s<><73> sola de responder a cuestiones que
|
|||
|
podr<EFBFBD><EFBFBD>as llegar a realizar?
|
|||
|
|
|||
|
Sup<EFBFBD><EFBFBD>n que quisieras averiguar las edades de cada mascota al tener cachorros. La
|
|||
|
tabla evento indica cu<63><75>ndo ha ocurrido esto, pero para calcular la edad de la
|
|||
|
madre, necesitas su fecha de nacimiento. Dado que eso est<73><74> almacenado en la
|
|||
|
tabla mascota, necesitas ambas tablas para la consulta:
|
|||
|
|
|||
|
mysql> SELECT mascota.nombre , (TO_DAYS(fecha) - TO_DAYS(nacimiento))/365 AS edad, anotaci<63><69>n
|
|||
|
-> FROM mascota, evento
|
|||
|
-> WHERE mascota.nombre = evento.nombre AND tipo = "parto";
|
|||
|
+--------+------+----------------------------------+
|
|||
|
| nombre | edad | anotaci<63><69>n |
|
|||
|
+--------+------+----------------------------------+
|
|||
|
| Fluffy | 2.27 | 4 cachorros, 3 hembras, 1 macho |
|
|||
|
| Buffy | 4.12 | 5 cachorros, 2 hembras, 3 machos |
|
|||
|
| Buffy | 5.10 | 3 cachorros, 3 hembras |
|
|||
|
+--------+------+----------------------------------+
|
|||
|
|
|||
|
Existen varios puntos que anotar sobre esta consulta:
|
|||
|
|
|||
|
* La cl<63><6C>usula FROM lista dos tablas dado que la consulta necesita extraer
|
|||
|
informaci<63><69>n de las dos.
|
|||
|
|
|||
|
* Cuando se combina la informaci<63><69>n de m<><6D>ltiples tablas, necesitas especificar
|
|||
|
c<><63>mo pueden ser asociados los registros de una tabla con los registros de la
|
|||
|
otra. Esto es f<><66>cil dado que ambas tienen una columna nombre (N.T.: nombre es
|
|||
|
una clave extranjera). La consulta usa la cl<63><6C>usula WHERE para combinar los
|
|||
|
registros de las dos tablas bas<61><73>ndose en los valores de nombre.
|
|||
|
|
|||
|
* Dado que la columna nombre aparece en ambas tablas, debes ser espec<65><63>fico sobre
|
|||
|
a qu<71><75> tabla te refieres cuando est<73><74>s hablando de esa columna. Esto se realiza
|
|||
|
poniendo el nombre de la tabla como prefijo de la columna.
|
|||
|
|
|||
|
No necesitas tener dos tablas diferentes para realizar un join. En algunas
|
|||
|
ocasiones es <20><>til realizar un join de una tabla consigo misma, si quieres comparar
|
|||
|
registros de una tabla con otros registros en la misma tabla. Por ejemplo, para buscar
|
|||
|
parejas de sexos entre tus mascotas, puedes enlazar la tabla mascota consigo
|
|||
|
mismo para emaparejar machos y hembras de las mismas especies:
|
|||
|
|
|||
|
mysql> SELECT p1.nombre, p1.sexo, p2.nombre, p2.sexo, p1.especie
|
|||
|
-> FROM mascota AS p1, mascota AS p2
|
|||
|
-> WHERE p1.especie = p2.especie AND p1.sexo = "f" AND p2.sexo = "m";
|
|||
|
+--------+------+--------+------+---------+
|
|||
|
| nombre | sexo | nombre | sexo | especie |
|
|||
|
+--------+------+--------+------+---------+
|
|||
|
| Fluffy | f | Claws | m | gato |
|
|||
|
| Buffy | f | Fang | m | perro |
|
|||
|
| Buffy | f | Bowser | m | perro |
|
|||
|
+--------+------+--------+------+---------+
|
|||
|
|
|||
|
|
|||
|
En esta consulta, especificamos un par de alias para el nombre de las tablas
|
|||
|
y ser capaces as<61><73> de referirnos a las columnas y saber en todo momento a qu<71><75>
|
|||
|
instancia de qu<71><75> tabla se asocia cada referencia de columna.
|
|||
|
|
|||
|
8.4 Obtener informaci<63><69>n sobre bases de datos y tablas
|
|||
|
================================================================
|
|||
|
|
|||
|
<EFBFBD><EFBFBD>Qu<EFBFBD><EFBFBD> ocurre si olvidas el nombre de una base de datos o de una tabla, o cu<63><75>l es
|
|||
|
la estructura de una tabla dada (ejm. : <20><>c<EFBFBD><63>mo se llaman sus columnas?) MySQL
|
|||
|
soluciona este problema a trav<61><76>s de numerosas sentencias que ofrecen informaci<63><69>n
|
|||
|
sobre las bases de datos y las tablas que soporta.
|
|||
|
|
|||
|
Ya hemos visto SHOW DATABASES, que lista las bases de datos gestionadas por el
|
|||
|
servidor. Para averiguar qu<71><75> base de datos est<73><74> actualmente seleccionada, usa la
|
|||
|
funci<EFBFBD><EFBFBD>n DATABASE():
|
|||
|
|
|||
|
mysql> SELECT DATABASE();
|
|||
|
+------------+
|
|||
|
| DATABASE() |
|
|||
|
+------------+
|
|||
|
| zoo |
|
|||
|
+------------+
|
|||
|
|
|||
|
Si a<><61>n no has seleccionado ninguna base de datos, el resultado estar<61><72> en blanco.
|
|||
|
|
|||
|
Para averiguar qu<71><75> tablas contiene la base de datos actual (por ejemplo, cuando
|
|||
|
no est<73><74>s seguro sobre el nombre de una tabla), usa este comando:
|
|||
|
|
|||
|
mysql> SHOW TABLES;
|
|||
|
+---------------+
|
|||
|
| Tables in zoo |
|
|||
|
+---------------+
|
|||
|
| evento |
|
|||
|
| mascota |
|
|||
|
+---------------+
|
|||
|
|
|||
|
Si quieres averiguar la estructura de una tabla, el comando DESCRIBE te ser<65><72>
|
|||
|
<EFBFBD><EFBFBD>til; muestra informaci<63><69>n sobre cada columna de una tabla:
|
|||
|
|
|||
|
mysql> DESCRIBE mascota;
|
|||
|
+-------------+-------------+------+-----+---------+-------+
|
|||
|
| Field | Type | Null | Key | Default | Extra |
|
|||
|
+-------------+-------------+------+-----+---------+-------+
|
|||
|
| nombre | varchar(20) | YES | | NULL | |
|
|||
|
| propietario | varchar(20) | YES | | NULL | |
|
|||
|
| especie | varchar(20) | YES | | NULL | |
|
|||
|
| sexo | char(1) | YES | | NULL | |
|
|||
|
| nacimiento | date | YES | | NULL | |
|
|||
|
| muerte | date | YES | | NULL | |
|
|||
|
+-------------+-------------+------+-----+---------+-------+
|
|||
|
|
|||
|
Field indica el nombre de la columna, Type es el tipo de datos para la columna,
|
|||
|
Null indica si la columna puede contener o no valores NULL, Key indica si la
|
|||
|
columna est<73><74> indexada o no, y Default especifica el valor por defecto para la
|
|||
|
columna.
|
|||
|
|
|||
|
Si tienes <20><>ndices en una tabla, SHOW INDEX FROM nombre_tabla te mostrar<61><72>
|
|||
|
informaci<EFBFBD><EFBFBD>n sobre ellos.
|
|||
|
|
|||
|
8.5 Usando mysql en modo batch
|
|||
|
=================================
|
|||
|
|
|||
|
En las secciones previas, hemos usado mysql interactivamente para introducir
|
|||
|
consultas y observar los resultados. Tambi<62><69>n puedes ejecutar mysql en modo
|
|||
|
batch. Para realizarlo, escribe los comandos que quieras ejecutar en un fichero,
|
|||
|
despu<EFBFBD><EFBFBD>s p<><70>dele a mysql que lea su entrada desde el fichero:
|
|||
|
|
|||
|
shell> mysql < fichero-batch
|
|||
|
|
|||
|
(N.T.: en ocasiones se traduce como fichero por lotes)
|
|||
|
|
|||
|
Si necesitas especificar par<61><72>metros de conexi<78><69>n en la l<><6C>nea de comandos, el
|
|||
|
comando podr<64><72>a parecerse a esto:
|
|||
|
|
|||
|
shell> mysql -h host -u user -p < fichero-batch
|
|||
|
Enter password: ********
|
|||
|
|
|||
|
Cuando usas MySQL de esta manera, est<73><74>s creando un fichero script (de gui<75><69>n), y
|
|||
|
despu<EFBFBD><EFBFBD>s ejecutando el script.
|
|||
|
|
|||
|
<EFBFBD><EFBFBD>Por qu<71><75> usar un script? He aqu<71><75> algunas razones:
|
|||
|
|
|||
|
* Si ejecutas una consulta repetidamente (digamos, cada d<><64>a o cada semana), el
|
|||
|
construir un script con esta consulta te permite evitar volver a teclearla
|
|||
|
cada vez que la ejecutes.
|
|||
|
|
|||
|
* Puedes generar nuevas consultas a partir de consultas ya existentes similares
|
|||
|
simplemente copiando y editando los ficheros script.
|
|||
|
|
|||
|
* El modo batch puede ser tambi<62><69>n muy <20><>til cuando est<73><74>s desarrollando una
|
|||
|
consulta, particularmente para comandos multi-l<><6C>nea o m<><6D>ltiples secuencias de
|
|||
|
comandos de m<><6D>ltiples sentencias. Si cometes un error, no necesitas reescribir
|
|||
|
todo. S<><53>mplemente edita el script para corregir el error, y despu<70><75>s p<><70>dele a
|
|||
|
mysql que lo vuelva a ejecutar.
|
|||
|
|
|||
|
* Si tienes una consulta que produce resultados muy largos, puedes usar un
|
|||
|
paginador para filtrar esta salida en lugar de ver c<><63>mo se desplaza fuera del
|
|||
|
alcance de tu pantalla:
|
|||
|
|
|||
|
|
|||
|
shell> mysql < fichero_batch | more
|
|||
|
|
|||
|
* Puedes redirigir la salida a un fichero para un procesamiento posterior:
|
|||
|
|
|||
|
shell> mysql < fichero_batch > mysql.out
|
|||
|
|
|||
|
* Puedes distribuir tu script a otras personas para que puedan ejecutar tambi<62><69>n
|
|||
|
tus comandos.
|
|||
|
|
|||
|
* Algunas situaciones no permiten un uso interactivo, por ejemplo, cuando
|
|||
|
ejecutas una consulta como una tarea de cron. (N.T.: cron es un comando UNIX
|
|||
|
que sirve para planificar y ejecutar comandos UNIX en el tiempo). En este
|
|||
|
caso, debes usar el procesamiento por lotes.
|
|||
|
|
|||
|
El formato de salida por defecto es diferente (m<><6D>s conciso) cuando ejecutas
|
|||
|
mysql en modo batch que cuando lo usas de manera interactiva. Por ejemplo, la
|
|||
|
salida de SELECT DISTINCT especie FROM mascota es la siguiente cuando se
|
|||
|
ejecuta de manera interactiva:
|
|||
|
|
|||
|
mysql> SELECT DISTINCT especie FROM mascota;
|
|||
|
+-----------+
|
|||
|
| especie |
|
|||
|
+-----------+
|
|||
|
| gato |
|
|||
|
| hamster |
|
|||
|
| p<><70>jaro |
|
|||
|
| perro |
|
|||
|
| serpiente |
|
|||
|
+-----------+
|
|||
|
|
|||
|
Y la siguiente si se ejecuta en modo batch:
|
|||
|
|
|||
|
especie
|
|||
|
gato
|
|||
|
hamster
|
|||
|
p<EFBFBD><EFBFBD>jaro
|
|||
|
perro
|
|||
|
serpiente
|
|||
|
|
|||
|
Si quieres obtener el formato de salida del modo interactivo tambi<62><69>n en modo
|
|||
|
batch, usa mysql -t. Para redirigir a salida est<73><74>ndar los comandos que se est<73><74>n
|
|||
|
ejecutando, usa mysql -vvv.
|
|||
|
|
|||
|
|
|||
|
|
|||
|
8.6 Consultas del proyecto gemelos
|
|||
|
|
|||
|
En Analytikerna y Lentus, hemos estado realizando el trabajo de campo y sistemas para
|
|||
|
un gran proyecto de investigaci<63><69>n. Este proyecto es una colaboraci<63><69>n entre el Instituto de
|
|||
|
Medicina Medioambiental en el Karolinska Institutet Stockholm y la Secci<63><69>n en Investigaci<63><69>n
|
|||
|
Cl<EFBFBD><EFBFBD>nica en Envejecimiento y Psicolog<6F><67>a en la Universidad del Sur de California.
|
|||
|
|
|||
|
El proyecto consisti<74><69> en una parte de selecci<63><69>n donde todos los gemelos en Suecia mayores de
|
|||
|
65 a<><61>os eran entrevistados por tel<65><6C>fono. Los gemelos que reun<75><6E>an ciertos criterios pasaban a la
|
|||
|
siguiente fase. En esta fase posterior, los gemelos que quer<65><72>an participar eran visitados por
|
|||
|
un equipo doctor/enfermera. Algunos de los ex<65><78>menes inclu<6C><75>an ex<65><78>menes f<><66>sicos y neuropsicol<6F><6C>gicos,
|
|||
|
pruebas de laboratorio, neuroim<69><6D>genes, valoraci<63><69>n del estado psicol<6F><6C>gico, y recopilaci<63><69>n de la
|
|||
|
historia familiar. Adem<65><6D>s, se recogieron datos sobre los factores de riesgo m<><6D>dicos y
|
|||
|
medioambientales.
|
|||
|
|
|||
|
Puede encontrarse m<><6D>s informaci<63><69>n sobre los estudios de gemelos en :
|
|||
|
|
|||
|
http://www.imm.ki.se/TWIN/TWINUKW.HTM
|
|||
|
|
|||
|
La <20><>ltima parte del proyecto se administra con un interfaz web escrito usando Perl y MySQL.
|
|||
|
Cada noche, todos los datos de las entrevistas son movidos a una base de datos MySQL.
|
|||
|
|
|||
|
8.6.1 Buscar todos los gemelos no-distribuidos
|
|||
|
|
|||
|
La siguiente consulta se usa para determinar qui<75><69>n pasa a la segunda parte del proyecto:
|
|||
|
|
|||
|
select
|
|||
|
concat(p1.id, p1.tvab) + 0 as tvid,
|
|||
|
concat(p1.christian_name, " ", p1.surname) as Name,
|
|||
|
p1.postal_code as Code,
|
|||
|
p1.city as City,
|
|||
|
pg.abrev as Area,
|
|||
|
if(td.participation = "Aborted", "A", " ") as A,
|
|||
|
p1.dead as dead1,
|
|||
|
l.event as event1,
|
|||
|
td.suspect as tsuspect1,
|
|||
|
id.suspect as isuspect1,
|
|||
|
td.severe as tsevere1,
|
|||
|
id.severe as isevere1,
|
|||
|
p2.dead as dead2,
|
|||
|
l2.event as event2,
|
|||
|
h2.nurse as nurse2,
|
|||
|
h2.doctor as doctor2,
|
|||
|
td2.suspect as tsuspect2,
|
|||
|
id2.suspect as isuspect2,
|
|||
|
td2.severe as tsevere2,
|
|||
|
id2.severe as isevere2,
|
|||
|
l.finish_date
|
|||
|
from
|
|||
|
twin_project as tp
|
|||
|
/* For Twin 1 */
|
|||
|
left join twin_data as td on tp.id = td.id and tp.tvab = td.tvab
|
|||
|
left join informant_data as id on tp.id = id.id and tp.tvab = id.tvab
|
|||
|
left join harmony as h on tp.id = h.id and tp.tvab = h.tvab
|
|||
|
left join lentus as l on tp.id = l.id and tp.tvab = l.tvab
|
|||
|
/* For Twin 2 */
|
|||
|
left join twin_data as td2 on p2.id = td2.id and p2.tvab = td2.tvab left join informant_data as id2 on p2.id = id2.id and p2.tvab = id2.tvab
|
|||
|
left join harmony as h2 on p2.id = h2.id and p2.tvab = h2.tvab
|
|||
|
left join lentus as l2 on p2.id = l2.id and p2.tvab = l2.tvab,
|
|||
|
person_data as p1,
|
|||
|
person_data as p2,
|
|||
|
postal_groups as pg
|
|||
|
where
|
|||
|
/* p1 gets main twin and p2 gets his/her twin. */
|
|||
|
/* ptvab is a field inverted from tvab */
|
|||
|
p1.id = tp.id and p1.tvab = tp.tvab and
|
|||
|
p2.id = p1.id and p2.ptvab = p1.tvab and
|
|||
|
/* Just the sceening survey */
|
|||
|
tp.survey_no = 5 and
|
|||
|
/* Skip if partner died before 65 but allow emigration (dead=9) */
|
|||
|
(p2.dead = 0 or p2.dead = 9 or
|
|||
|
(p2.dead = 1 and
|
|||
|
(p2.death_date = 0 or
|
|||
|
(((to_days(p2.death_date) - to_days(p2.birthday)) / 365)
|
|||
|
>= 65))))
|
|||
|
and
|
|||
|
(
|
|||
|
/* Twin is suspect */
|
|||
|
(td.future_contact = 'Yes' and td.suspect = 2) or
|
|||
|
/* Twin is suspect - Informant is Blessed */
|
|||
|
(td.future_contact = 'Yes' and td.suspect = 1 and id.suspect = 1) o
|
|||
|
/* No twin - Informant is Blessed */
|
|||
|
(ISNULL(td.suspect) and id.suspect = 1 and id.future_contact = 'Yes') or
|
|||
|
/* Twin broken off - Informant is Blessed */
|
|||
|
(td.participation = 'Aborted'
|
|||
|
and id.suspect = 1 and id.future_contact = 'Yes') or
|
|||
|
/* Twin broken off - No inform - Have partner */
|
|||
|
(td.participation = 'Aborted' and ISNULL(id.suspect) and p2.dead = 0))
|
|||
|
and
|
|||
|
l.event = 'Finished'
|
|||
|
/* Get at area code */
|
|||
|
and substring(p1.postal_code, 1, 2) = pg.code
|
|||
|
/* Not already distributed */
|
|||
|
and (h.nurse is NULL or h.nurse=00 or h.doctor=00)
|
|||
|
/* Has not refused or been aborted */
|
|||
|
and not (h.status = 'Refused' or h.status = 'Aborted'
|
|||
|
or h.status = 'Died' or h.status = 'Other')
|
|||
|
order by
|
|||
|
tvid;
|
|||
|
|
|||
|
Algunas explicaciones:
|
|||
|
|
|||
|
`concat(p1.id, p1.tvab) + 0 as tvid'
|
|||
|
Queremos ordenar por la concatenaci<63><69>n de `id' y `tvab' en orden num<75><6D>rico.
|
|||
|
A<><41>adiendo `0' al resultado provocamos que *MySQL* trate el resultado como
|
|||
|
un n<><6E>mero.
|
|||
|
|
|||
|
column `id'
|
|||
|
Esto identifica un par de gemelos. Es una clave en todas las tablas.
|
|||
|
|
|||
|
column `tvab'
|
|||
|
Esto identifica un gemelo de una pareja. Tiene un valor de `1' <20><> `2'
|
|||
|
|
|||
|
column `ptvab'
|
|||
|
Esto es la inversa de `tvab'. Cuando `tvab' es `1' esto es `2', y
|
|||
|
vice versa. Esto existe para ahorrarnos teclear y para hacer m<><6D>s f<><66>cil
|
|||
|
la optimizaci<63><69>n de la consulta a MySQL.
|
|||
|
|
|||
|
|
|||
|
Esta consulta demuestra, entre otras cosas, c<><63>mo realizar b<><62>squedas en una tabla
|
|||
|
enlazada con la misma tabla a trav<61><76>s de un join (p1 y p2). En el ejemplo, <20><>sto
|
|||
|
se usa para comprobar cu<63><75>ndo un gemelo de una pareja muri<72><69> antes de cumplir 65.
|
|||
|
En caso afirmativo, la fila no es devuelta.
|
|||
|
|
|||
|
Todo lo anterior existe en todas las tablas con informaci<63><69>n relacionada con los gemelos.
|
|||
|
Tenemos una clave tanto en id, tvab (todas las tablas) como en id,ptvab (person_data) para
|
|||
|
construir consultas m<><6D>s r<><72>pidas.
|
|||
|
|
|||
|
En nuestra m<><6D>quina de producci<63><69>n (una UltraSPARC 200MHz), esta consulta devuelve alrededor
|
|||
|
de 150-200 filas y tarda menos de un segundo.
|
|||
|
|
|||
|
El n<><6E>mero actual de registros en las tablas usadas arriba:
|
|||
|
|
|||
|
Tabla Filas
|
|||
|
|
|||
|
person_data 71074
|
|||
|
lentus 5291
|
|||
|
twin_project 5286
|
|||
|
twin_data 2012
|
|||
|
informant_data 663
|
|||
|
harmony 381
|
|||
|
postal_groups 100
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
8.6.2 Mostrar una tabla con el estado de la pareja de gemelos.
|
|||
|
|
|||
|
Cada entrevista finaliza con un c<><63>digo de estado llamado event. La consulta mostrada
|
|||
|
debajo se usa para imprimir una tabla sobre todas las parejas de gemelos combinadas por evento.
|
|||
|
Esto indica en cu<63><75>ntas parejas ambos gemelos han finalizado, en cu<63><75>ntas parejas
|
|||
|
ha finalizado un gemelo y el otro se rechaz<61><7A>, etc.
|
|||
|
|
|||
|
select
|
|||
|
t1.event,
|
|||
|
t2.event,
|
|||
|
count(*)
|
|||
|
from
|
|||
|
lentus as t1,
|
|||
|
lentus as t2,
|
|||
|
twin_project as tp
|
|||
|
where
|
|||
|
/* We are looking at one pair at a time */
|
|||
|
t1.id = tp.id
|
|||
|
and t1.tvab=tp.tvab
|
|||
|
and t1.id = t2.id
|
|||
|
/* Just the sceening survey */
|
|||
|
and tp.survey_no = 5
|
|||
|
/* This makes each pair only appear once */
|
|||
|
and t1.tvab='1' and t2.tvab='2'
|
|||
|
group by
|
|||
|
t1.event, t2.event;
|
|||
|
|
|||
|
|
|||
|
|