El lenguaje SQL nos permite escribir un SELECT en muchos lugares, veamos algunos ejemplos:
Ejemplo 1. Un SELECT simple
SELECT MiColumna1, MiColumna2 FROM MiTabla
En este caso se escribió solamente un SELECT, es útil para consultas sencillas pero cuando las consultas se complican ya no será suficiente.
Ejemplo 2. Un SELECT con una subconsulta
SELECT MiColumna1, (SELECT MiColumna2 FROM MiTabla2 WHERE MiCondición) FROM MiTabla
En este ejemplo hemos reemplazado a MiColumna2 por una subconsulta, o sea por un SELECT dentro de otro SELECT. El SELECT que está entre paréntesis es la subconsulta. Fíjate que además del nombre de la tabla se escribió la cláusula WHERE ¿por qué eso? porque la subconsulta debe devolver una sola columna y una sola fila.
Si la subconsulta devuelve más de una columna entonces verás el error: «Count of column list and variable list do not match». O sea que se esperaba una sola columna pero la subconsulta está devolviendo más de una columna.
Si devuelve una sola columna pero más de una fila entonces el mensaje de error que verás será: «Multiple rows in singleton select». ¿qué significa esa frase? que la subconsulta debería devolver una sola fila pero está devolviendo más de una fila.
La cláusula WHERE sirve justamente para poner una condición que estás seguro que se cumple para una sola fila. Además de la cláusula WHERE también podrías usar, dependiendo del caso, las cláusulas FIRST 1, ó ROWS 1 ó DISTINCT o las funciones agregadas MAX(), MIN(), etc. Lo que debes recordar es que la subconsulta debe devolver una fila y solamente una fila. Puede devolver cero filas, y en ese caso el valor de la columna será NULL (en SQL un valor de NULL significa «valor desconocido») pero jamás más de una fila.
Ejemplo 3. Un SELECT a continuación del FROM
SELECT MiColumna1, MiColumna2 FROM (SELECT MiColumna1, MiColumna2 FROM MiTabla)
Aquí el SELECT interno se escribió después de la cláusula FROM. En estos casos no se le llama subconsulta sino que se le llama tabla derivada.
Para que funcione, la cantidad de columnas de la tabla derivada siempre debe ser igual o mayor a la cantidad de columnas del SELECT principal, nunca puede ser menor (y a la tabla principal si lo deseas puedes agregarle columnas que tengan valores constantes, como 17, 21, ‘Asunción’, etc.). Y los nombres de las columnas deben coincidir, no puedes usar una columna en el SELECT principal que no hayas usado en la tabla derivada.
Fíjate que la subconsulta debe devolver una sola columna pero la tabla derivada puede devolver varias columnas.
Ejemplo 4. Una subconsulta en la cláusula GROUP BY
SELECT MiColumna1, (SELECT MiColumna2 FROM MiTabla2 WHERE MiCondición) FROM MiTabla GROUP BY MiColumna1, (SELECT MiColumna2 FROM MiTabla2 WHERE MiCondición)
En este caso, la subconsulta se usó también luego de la cláusula GROUP BY, eso es necesario porque todas las columnas que se incluyen en el SELECT principal y que no son funciones agregadas también deben encontrarse a continuación de la cláusula GROUP BY.
Ejemplo 5. Una subconsulta en la cláusula ORDER BY
SELECT MiColumna1, MiColumna2 FROM MiTabla ORDER BY (SELECT MiColumna3 FROM MiTabla2 WHERE MiCondición)
Aquí como puedes ver la subconsulta se escribió a continuación de la cláusula ORDER BY y por lo tanto el SELECT principal será ordenado por el contenido de la subconsulta.
Ejemplo 6. Un subconsulta en el JOIN
SELECT MiColumna1, MiColumna2 FROM MiTabla JOIN MiTabla2 ON MiColumna = (SELECT MiOtraColumna FROM MiOtraTabla)
También puede usarse una subconsulta en el JOIN, sea éste del tipo que sea (INNER, LEFT, RIGHT, FULL, CROSS, NATURAL)
Ejemplo 7. Insertando, modificando o borrando filas desde una subconsulta
Si necesitas insertarle datos a una tabla y los datos que necesitas se encuentran en otra u otras tablas la forma más rápida es hacerlo con una subconsulta:
INSERT INTO MiTabla (MiColumna1, MiColumna2, MiColumna3) SELECT MiValor1, MiValor2, MiValor3 FROM MiOtraTabla
También puedes usar una subconsulta para establecer la condición que necesitas para modificar o borrar filas. En general se usan subconsultas cuando las filas que se quiere insertar, modificar o borrar son muchas.
Conclusión:
Como seguramente habrás deducido luego de ver los ejemplos anteriores, las subconsultas pueden usarse en muchos lugares, prácticamente puedes usarlas en cualquier lugar donde necesites el valor de una columna.
Hay muchos ejemplos más que podría escribir pero con los anteriores supongo que ya tienes bien clara la idea.
¿Es conveniente usar subconsultas?
A veces sí, a veces no. En ocasiones es lo mejor que puedes hacer y en ocasiones hay mejores alternativas así que debes tratarlas como una herramienta más, algo que te puede ayudar a conseguir los resultados que estás buscando. Pero debes recordar que en general usar subconsultas es más lento que no usarlas y por lo tanto debes siempre verificar que no exista una mejor opción.
Artículos relacionados:
Francisco
Dic 25, 2013 @ 01:51:19
Interesante esto Walter, definitivamente.
Lo que nunca has anunciado , es tu bebe, se que tu esposa esperaba un bebe, comentame, como esta, orgulloso te has de sentir, no le compres una tablet por lo pronto, saludos!!!
wrov
Dic 25, 2013 @ 03:56:32
Pues sí Francisco ya tenemos una hermosa bebita que como te imaginarás es la alegría de la casa y la mimada de todos quienes la conocen.
Saludos.
Walter.
Juan carlos Ramirez
Feb 11, 2014 @ 12:52:07
Hola buenos dias, agradeceria mucho tu ayuda con esta consulta
SELECT nombreagente,
(SELECT Sum(Ventas.ctotal) FROM Ventas WHERE Agentes.cidagente = Ventas.cidagente) AS ‘Total’ »
FROM mgw10001 me da error de Invalid use of subquery. en FOXPRO
Juan carlos Ramirez
Feb 11, 2014 @ 12:53:53
Hola buenos dias, agradeceria mucho tu ayuda con esta consulta
SELECT nombreagente,
(SELECT Sum(Ventas.ctotal) FROM Ventas WHERE Agentes.cidagente = Ventas.cidagente) AS ‘Total’
FROM Agentes me da error de Invalid use of subquery. en FOXPRO
Alexis Molina
Ene 17, 2018 @ 23:57:12
Gracias master por compartir esta información.
Me quedo claro que la tabla derivada(consulta después del from) puede devolver varias columnas, pero también tiene que devolver una fila como una subconsulta?.
y mi otro duda, que otras opciones uno se puede preguntar antes de usar una subconsulta?(osea que otras alternativas pueden existir).
Saludos y espero su respuesta, de seguro mas de algo aprenderé nuevamente.
wrov
Ene 18, 2018 @ 09:24:40
1. No. Una tabla derivada puede devolver muchas filas y muchas columnas. Se usan tablas derivadas para filtrar tablas. Eso en muchos casos es más rápido que usar la tabla completa (en general esto ocurre cuando tienes buenos índices que te permiten filtrar muy rápido). Por ejemplo, una tabla tiene 10.000 filas, al filtrarla obtienes 40 filas, y en tu SELECT usas la tabla filtrada, o sea las 40 filas.
2. Tienes varias alternativas:
a) Crear un stored procedure seleccionable
b) Usar un JOIN
c) Utilizar UNION
d) Utilizar CTE
Si quieres asegurarte de que tu consulta sea lo más rápida posible, deberías probar todas ellas e ir controlando los tiempos, para quedarte con la mejor. En los programas de administración gráfica (como EMS SQL Manager, IBExpert, SQL Maestro, etc.) existe la opción de ver el rendimiento en forma gráfica, entonces es fácil ver cuales tablas usan índices y cuales no, la cantidad de filas procesadas, el tiempo transcurrido, etc.
Saludos.
Walter.