Introducción
El lenguaje de consulta estructurado (SQL) es un
lenguaje de base de datos normalizado, utilizado por los diferentes motores de
bases de datos para realizar determinadas operaciones sobre los datos o sobre
la estructura de los mismos. Pero como sucede con cualquier sistema de normalización
hay excepciones para casi todo; de hecho, cada motor de bases de datos tiene
sus peculiaridades y lo hace diferente de otro motor, por lo tanto, el lenguaje
SQL normalizado (ANSI) no nos servirá para resolver todos los problemas, aunque
si se puede asegurar que cualquier sentencia escrita en ANSI será interpretable
por cualquier motor de datos.
Breve Historia
La historia de SQL (que se pronuncia deletreando en
inglés las letras que lo componen, es decir "ese-cu-ele" y no
"siquel" como se oye a menudo) empieza en 1974 con la definición, por
parte de Donald Chamberlin y de otras personas que trabajaban en los
laboratorios de investigación de IBM, de un lenguaje para la especificación de
las características de las bases de datos que adoptaban el modelo relacional.
Este lenguaje se llamaba SEQUEL (Structured English Query Language) y se
implementó en un prototipo llamado SEQUEL-XRM entre 1974 y 1975. Las
experimentaciones con ese prototipo condujeron, entre 1976 y 1977, a una revisión
del lenguaje (SEQUEL/2), que a partir de ese momento cambió de nombre por
motivos legales, convirtiéndose en SQL. El prototipo (System R), basado en este
lenguaje, se adoptó y utilizó internamente en IBM y lo adoptaron algunos de sus
clientes elegidos. Gracias al éxito de este sistema, que no estaba todavía
comercializado, también otras compañías empezaron a desarrollar sus productos
relacionales basados en SQL. A partir de 1981, IBM comenzó a entregar sus
productos relacionales y en 1983 empezó a vender DB2. En el curso de los años
ochenta, numerosas compañías (por ejemplo Oracle y Sybase, sólo por citar
algunos) comercializaron productos basados en SQL, que se convierte en el
estándar industrial de hecho por lo que respecta a las bases de datos relacionales.
En 1986, el ANSI adoptó SQL (sustancialmente adoptó el
dialecto SQL de IBM) como estándar para los lenguajes relacionales y en 1987 se
transfomó en estándar ISO. Esta versión del estándar va con el nombre de
SQL/86. En los años siguientes, éste ha sufrido diversas revisiones que han
conducido primero a la versión SQL/89 y, posteriormente, a la actual SQL/92.
El hecho de tener un estándar definido por un lenguaje
para bases de datos relacionales abre potencialmente el camino a la
intercomunicabilidad entre todos los productos que se basan en él. Desde el
punto de vista práctico, por desgracia las cosas fueron de otro modo.
Efectivamente, en general cada productor adopta e implementa en la propia base
de datos sólo el corazón del lenguaje SQL (el así llamado Entry level o al
máximo el Intermediate level), extendiéndolo de manera individual según la
propia visión que cada cual tenga del mundo de las bases de datos.
Actualmente, está en marcha un proceso de revisión del
lenguaje por parte de los comités ANSI e ISO, que debería terminar en la
definición de lo que en este momento se conoce como SQL3. Las características
principales de esta nueva encarnación de SQL deberían ser su transformación en
un lenguaje stand-alone (mientras ahora se usa como lenguaje hospedado en otros
lenguajes) y la introducción de nuevos tipos de datos más complejos que
permitan, por ejemplo, el tratamiento de datos multimediales.
Componentes del SQL
El lenguaje SQL está compuesto por comandos,
cláusulas, operadores y funciones de agregado. Estos elementos se combinan en
las instrucciones para crear, actualizar y manipular las bases de datos.
Comandos
Existen dos tipos de comandos SQL:
·
Los
DLL que permiten crear y definir nuevas bases de datos, campos e índices.
·
Los
DML que permiten generar consultas para ordenar, filtrar y extraer datos de la
base de datos.
Comandos
DLL:
|
Comando
|
Descripción
|
|
CREATE
|
Utilizado
para crear nuevas tablas, campos e índices
|
|
DROP
|
Empleado
para eliminar tablas e índices
|
|
ALTER
|
Utilizado
para modificar las tablas agregando campos o cambiando la definición de los
campos.
|
Comandos
DML:
|
Comando
|
Descripción
|
|
SELECT
|
Utilizado
para consultar registros de la base de datos que satisfagan un criterio
determinado
|
|
INSERT
|
Utilizado
para cargar lotes de datos en la base de datos en una única operación.
|
|
UPDATE
|
Utilizado
para modificar los valores de los campos y registros especificados
|
|
DELETE
|
Utilizado
para eliminar registros de una tabla de una base de datos
|
Cláusulas
Las cláusulas son condiciones de modificación
utilizadas para definir los datos que desea seleccionar o manipular.
|
Cláusula
|
Descripción
|
|
FROM
|
Utilizada
para especificar la tabla de la cual se van a seleccionar los registros
|
|
WHERE
|
Utilizada
para especificar las condiciones que deben reunir los registros que se van a
seleccionar
|
|
GROUP BY
|
Utilizada
para separar los registros seleccionados en grupos específicos
|
|
HAVING
|
Utilizada
para expresar la condición que debe satisfacer cada grupo
|
|
ORDER BY
|
Utilizada
para ordenar los registros seleccionados de acuerdo con un orden específico
|
Operadores
lógicos
|
Operador
|
Uso
|
|
AND
|
Es el
"y" lógico. Evalua dos condiciones y devuelve un valor de verdad
sólo si ambas son ciertas.
|
|
OR
|
Es el
"o" lógico. Evalúa dos condiciones y devuelve un valor de verdar si
alguna de las dos es cierta.
|
|
NOT
|
Negación
lógica. Devuelve el valor contrario de la expresión.
|
Operadores de
Comparación
|
Operador
|
Uso
|
|
<
|
Menor que
|
|
>
|
Mayor que
|
|
<>
|
Distinto de
|
|
<=
|
Menor ó Igual que
|
|
>=
|
Mayor ó Igual que
|
|
=
|
Igual que
|
|
BETWEEN
|
Utilizado
para especificar un intervalo de valores.
|
|
LIKE
|
Utilizado
en la comparación de un modelo
|
|
In
|
Utilizado
para especificar registros de una base de datos
|
Funciones de Agregado
Las funciones de agregado se usan dentro de una
cláusula SELECT en grupos de registros para devolver un único valor que se
aplica a un grupo de registros.
|
Función
|
Descripción
|
|
AVG
|
Utilizada
para calcular el promedio de los valores de un campo determinado
|
|
COUNT
|
Utilizada
para devolver el número de registros de la selección
|
|
SUM
|
Utilizada
para devolver la suma de todos los valores de un campo determinado
|
|
MAX
|
Utilizada
para devolver el valor más alto de un campo especificado
|
|
MIN
|
Utilizada
para devolver el valor más bajo de un campo especificado
|
Consultas de Selección
Las consultas de selección se utilizan
para indicar al motor de datos que devuelva información de las bases de datos,
esta información es devuelta en forma de conjunto de registros que se pueden
almacenar en un objeto recordset. Este conjunto de registros es modificable.
La sintaxis básica de una consulta de
selección es la siguiente:
SELECT
Campos FROM Tabla;
En donde campos es la lista de campos
que se deseen recuperar y tabla es el origen de los mismos, por ejemplo:
SELECT
Nombre, Telefono FROM Clientes;
Esta consulta devuelve un recordset con
el campo nombre y teléfono de la tabla clientes.
En determinadas ocasiones nos puede
interesar incluir una columna con un texto fijo en una consulta de selección,
por ejemplo, supongamos que tenemos una tabla de empleados y deseamos recuperar
las tarifas semanales de los electricistas, podríamos realizar la siguiente
consulta:
SELECT
Empleados.Nombre, 'Tarifa semanal: ',
Empleados.TarifaHora * 40
FROM
Empleados
WHERE
Empleados.Cargo = 'Electricista'
Adicionalmente se puede especificar el
orden en que se desean recuperar los registros de las tablas mediante la
claúsula ORDER
BY Lista de Campos. En donde Lista de camposrepresenta los campos
a ordenar. Ejemplo:
SELECT
CodigoPostal, Nombre, Telefono FROM Clientes ORDER BY Nombre;
Esta consulta devuelve los campos
CodigoPostal, Nombre, Telefono de la tabla Clientes ordenados por el campo
Nombre.
Se pueden ordenar los registros por mas
de un campo, como por ejemplo:
SELECT
CodigoPostal, Nombre, Telefono FROM Clientes ORDER BY
CodigoPostal,
Nombre;
Incluso se puede especificar el orden de
los registros: ascendente mediante la claúsula ASC (se toma este
valor por defecto) ó descendente (DESC).
SELECT
CodigoPostal, Nombre, Telefono FROM Clientes ORDER BY
CodigoPostal
DESC , Nombre ASC;
Si deseamos que la sentencia SQL utilice
un índice para mostrar los resultados se puede utilizar la palabra
reservada INDEX de la siguiente forma:
SELECT ... FROM Tabla
(INDEX=Indice) ...
Normalmente los motores de las bases de
datos deciden que indice se debe utilizar para la consulta, para ello utilizan
criterios de rendimiento y sobre todo los campos de búsqueda especificados en
la cláusula WHERE. Si se desea forzar a no utilizar ningún índice utilizaremos
la siguiente sintaxis:
SELECT
... FROM Tabla (INDEX=0) ...
El predicado se incluye entre la
claúsula y el primer nombre del campo a recuperar, los posibles predicados son:
|
Predicado
|
Descripción
|
|
ALL
|
Devuelve todos los campos de la tabla
|
|
TOP
|
Devuelve un determinado número de
registros de la tabla
|
|
DISTINCT
|
Omite los registros cuyos campos
seleccionados coincidan totalmente
|
|
DISTINCTROW
|
Omite los registros duplicados
basandose en la totalidad del registro y no sólo en los campos seleccionados.
|
ALL
Si no se incluye ninguno de los
predicados se asume ALL. El Motor de base de datos selecciona todos los registros
que cumplen las condiciones de la instrucción SQL. No se conveniente abusar de
este predicado ya que obligamos al motor de la base de datos a analizar la
estructura de la tabla para averiguar los campos que contiene, es mucho más
rápido indicar el listado de campos deseados.
SELECT ALL FROM
Empleados;
SELECT * FROM
Empleados;
TOP
Devuelve un cierto número de registros
que entran entre al principio o al final de un rango especificado por una
cláusula ORDER
BY.
Supongamos que queremos recuperar los nombres de los 25 primeros estudiantes
del curso 1994:
SELECT
TOP 25 Nombre, Apellido FROM Estudiantes ORDER BY Nota DESC;
Si no se incluye la cláusula ORDER BY, la consulta
devolverá un conjunto arbitrario de 25 registros de la tabla Estudiantes. El
predicado TOP no elige entre valores iguales. En
el ejemplo anterior, si la nota media número 25 y la 26 son iguales, la
consulta devolverá 26 registros. Se puede utilizar la palabra reservada PERCENT para devolver un
cierto porcentaje de registros que caen al principio o al final de un rango
especificado por la cláusula ORDER BY. Supongamos que en
lugar de los 25 primeros estudiantes deseamos el 10 por ciento del curso:
SELECT
TOP 10 PERCENT Nombre, Apellido FROM Estudiantes
ORDER
BY Nota DESC;
El valor que va a continuación de TOP debe ser un
Integer sin signo. TOP no afecta a la posible
actualización de la consulta.
DISTINCT
Omite los registros que contienen datos
duplicados en los campos seleccionados. Para que los valores de cada campo
listado en la instrucción SELECT se incluyan en la consulta deben
ser únicos.
Por ejemplo, varios empleados listados
en la tabla Empleados pueden tener el mismo apellido. Si dos registros
contienen López en el campo Apellido, la siguiente instrucción SQL devuelve un
único registro:
SELECT
DISTINCT Apellido FROM Empleados;
Con otras palabras el predicado DISTINCT devuelve
aquellos registros cuyos campos indicados en la cláusula SELECT posean un
contenido diferente. El resultado de una consulta que utilizaDISTINCT no es actualizable
y no refleja los cambios subsiguientes realizados por otros usuarios.
DISTINCTROW
Este predicado no es compatible con
ANSI. Que yo sepa a día de hoy sólo funciona con ACCESS.
Devuelve los registros diferentes de una
tabla; a diferencia del predicado anterior que sólo se fijaba en el contenido
de los campos seleccionados, éste lo hace en el contenido del registro completo
independientemente de los campo indicados en la cláusula SELECT.
SELECT
DISTINCTROW Apellido FROM Empleados;
Si la tabla empleados contiene dos
registros: Antonio López y Marta López, el ejemplo del predicado DISTINCT devuleve un
único registro con el valor López en el campo Apellido ya que busca no
duplicados en dicho campo. Este último ejemplo devuelve dos registros con el
valor López en el apellido ya que se buscan no duplicados en el registro
completo.
En determinadas circunstancias es
necesario asignar un nombre a alguna columna determinada de un conjunto
devuelto, otras veces por simple capricho o por otras circunstancias. Para
resolver todas ellas tenemos la palabra reservada AS que se encarga
de asignar el nombre que deseamos a la columna deseada. Tomado como referencia
el ejemplo anterior podemos hacer que la columna devuelta por la consulta, en
lugar de llamarse apellido (igual que el campo devuelto) se llame Empleado. En
este caso procederíamos de la siguiente forma:
SELECT
DISTINCTROW Apellido AS Empleado FROM Empleados;
AS no es una
palabra reservada de ANSI, existen diferentes sistemas de asignar los alias en
función del motor de bases de datos. En ORACLE para asignar un alias a un campo
hay que hacerlo de la siguiente forma:
SELECT
Apellido AS "Empleado" FROM Empleados;
También podemos asignar alias a las
tablas dentro de la consulta de selección, en esta caso hay que tener en cuenta
que en todas las referencias que deseemos hacer a dicha tabla se ha de utilizar
el alias en lugar del nombre. Esta técnica será de gran utilidad más adelante
cuando se estudien las vinculaciones entre tablas. Por ejemplo:
SELECT
Apellido AS Empleado FROM Empleados AS Trabajadores;
Para asignar alias a las tablas en
ORACLE y SQL-SERVER los alias se asignan escribiendo el nombre de la tabla,
dejando un espacio en blanco y escribiendo el Alias (se asignan dentro de la
cláusula FROM).
SELECT
Trabajadores.Apellido AS Empleado FROM Empleados Trabajadores;
Esta nomenclatura [Tabla].[Campo] se debe utilizar
cuando se está recuperando un campo cuyo nombre se repite en varias de las
tablas que se utilizan en la sentencia. No obstante cuando en la sentencia se
emplean varias tablas es aconsejable utilizar esta nomenclatura para evitar el
trabajo que supone al motor de datos averiguar en que tabla está cada uno de
los campos indicados en la cláusula SELECT.
Para concluir este capítulo se debe
hacer referencia a la recuperación de registros de bases de datos externa. En
ocasiones es necesario la recuperación de información que se encuentra
contenida en una tabla que no se encuentra en la base de datos que ejecutará la
consulta o que en ese momento no se encuentra abierta, esta situación la
podemos salvar con la palabra reservada IN de la siguiente
forma:
SELECT
DISTINCTROW Apellido AS Empleado FROM Empleados
IN
'c:\databases\gestion.mdb';
En donde c:\databases\gestion.mdb es la base de
datos que contiene la tabla Empleados. Esta técnica es muy sencilla y común en
bases de datos de tipo ACCESS en otros sistemas como SQL-SERVER u ORACLE, la
cosa es más complicada la tener que existir relaciones de confianza entre los
servidores o al ser necesaria la vinculación entre las bases de datos. Este
ejemplo recupera la información de una base de datos de SQL-SERVER ubicada en
otro servidor (se da por supuesto que los servidores están lincados):
SELECT
Apellido FROM Servidor1.BaseDatos1.dbo.Empleados
Consultas de
Selección
Las consultas de selección se utilizan para indicar al
motor de datos que devuelva información de las bases de datos, esta información
es devuelta en forma de conjunto de registros que se pueden almacenar en un
objeto recordset. Este conjunto de registros es modificable.
Consultas Básicas
La sintaxis básica de una consulta de selección es la
siguiente:
SELECT Campos FROM Tabla;
En donde campos es la lista de campos que se deseen
recuperar y tabla es el origen de los mismos, por ejemplo:
SELECT Nombre, Telefono FROM Clientes;
Esta consulta devuelve un recordset con el campo
nombre y teléfono de la tabla clientes.
Devolver Literales
En determinadas ocasiones nos puede interesar incluir
una columna con un texto fijo en una consulta de selección, por ejemplo,
supongamos que tenemos una tabla de empleados y deseamos recuperar las tarifas
semanales de los electricistas, podríamos realizar la siguiente consulta:
SELECT
Empleados.Nombre, 'Tarifa semanal: ', Empleados.TarifaHora * 40
FROM
Empleados
WHERE
Empleados.Cargo = 'Electricista'
Ordenar los Registros
Adicionalmente se puede especificar el orden en que se
desean recuperar los registros de las tablas mediante la claúsula ORDER BY Lista
de Campos. En donde Lista de camposrepresenta los campos a ordenar. Ejemplo:
SELECT CodigoPostal, Nombre, Telefono FROM Clientes ORDER BY Nombre;
Esta consulta devuelve los campos CodigoPostal,
Nombre, Telefono de la tabla Clientes ordenados por el campo Nombre.
Se pueden ordenar los registros por mas de un campo,
como por ejemplo:
SELECT CodigoPostal, Nombre, Telefono FROM Clientes ORDER BY
CodigoPostal, Nombre;
Incluso se puede especificar el orden de los
registros: ascendente mediante la claúsula ASC (se toma este valor por defecto) ó descendente (DESC).
SELECT CodigoPostal, Nombre, Telefono FROM Clientes ORDER BY
CodigoPostal DESC , Nombre ASC;
Uso de Indices de las tablas
Si deseamos que la sentencia SQL utilice un índice
para mostrar los resultados se puede utilizar la palabra reservada INDEX de la siguiente forma:
SELECT ... FROM Tabla (INDEX=Indice) ...
Normalmente los motores de las bases de datos deciden
que indice se debe utilizar para la consulta, para ello utilizan criterios de
rendimiento y sobre todo los campos de búsqueda especificados en la cláusula
WHERE. Si se desea forzar a no utilizar ningún índice utilizaremos la siguiente
sintaxis:
SELECT ... FROM Tabla (INDEX=0) ...
Consultas con Predicado
El predicado se incluye entre la claúsula y el primer
nombre del campo a recuperar, los posibles predicados son:
|
Predicado
|
Descripción
|
|
ALL
|
Devuelve todos los campos de la tabla
|
|
TOP
|
Devuelve un determinado número de
registros de la tabla
|
|
DISTINCT
|
Omite los registros cuyos campos
seleccionados coincidan totalmente
|
|
DISTINCTROW
|
Omite los registros duplicados
basandose en la totalidad del registro y no sólo en los campos seleccionados.
|
ALL
Si no se incluye ninguno de los predicados se asume
ALL. El Motor de base de datos selecciona todos los registros que cumplen las
condiciones de la instrucción SQL. No se conveniente abusar de este predicado
ya que obligamos al motor de la base de datos a analizar la estructura de la
tabla para averiguar los campos que contiene, es mucho más rápido indicar el
listado de campos deseados.
SELECT ALL FROM Empleados;
SELECT * FROM Empleados;
TOP
Devuelve un cierto número de registros que entran
entre al principio o al final de un rango especificado por una cláusula ORDER BY. Supongamos que queremos recuperar los nombres de los
25 primeros estudiantes del curso 1994:
SELECT TOP 25 Nombre, Apellido FROM Estudiantes ORDER BY Nota DESC;
Si no se incluye la cláusula ORDER BY, la consulta devolverá un conjunto arbitrario de 25
registros de la tabla Estudiantes. El predicado TOP no elige entre valores iguales. En el ejemplo
anterior, si la nota media número 25 y la 26 son iguales, la consulta devolverá
26 registros. Se puede utilizar la palabra reservada PERCENT para devolver un cierto porcentaje de registros que
caen al principio o al final de un rango especificado por la cláusula ORDER BY. Supongamos que en lugar de los 25 primeros
estudiantes deseamos el 10 por ciento del curso:
SELECT TOP 10 PERCENT Nombre, Apellido FROM Estudiantes
ORDER BY Nota DESC;
El valor que va a continuación de TOP debe ser un Integer sin signo. TOP no afecta a la posible actualización de la consulta.
DISTINCT
Omite los registros que contienen datos duplicados en
los campos seleccionados. Para que los valores de cada campo listado en la
instrucción SELECT se incluyan en la consulta deben ser únicos.
Por ejemplo, varios empleados listados en la tabla
Empleados pueden tener el mismo apellido. Si dos registros contienen López en
el campo Apellido, la siguiente instrucción SQL devuelve un único registro:
SELECT DISTINCT Apellido FROM Empleados;
Con otras palabras el predicado DISTINCT devuelve aquellos registros cuyos campos indicados en
la cláusula SELECT posean un contenido diferente. El resultado de una
consulta que utilizaDISTINCT no es actualizable y no refleja los cambios
subsiguientes realizados por otros usuarios.
DISTINCTROW
Este predicado no es compatible con ANSI. Que yo sepa
a día de hoy sólo funciona con ACCESS.
Devuelve los registros diferentes de una tabla; a
diferencia del predicado anterior que sólo se fijaba en el contenido de los
campos seleccionados, éste lo hace en el contenido del registro completo
independientemente de los campo indicados en la cláusula SELECT.
SELECT DISTINCTROW Apellido FROM Empleados;
Si la tabla empleados contiene dos registros: Antonio
López y Marta López, el ejemplo del predicado DISTINCT devuleve un único registro con el valor López en el
campo Apellido ya que busca no duplicados en dicho campo. Este último ejemplo
devuelve dos registros con el valor López en el apellido ya que se buscan no
duplicados en el registro completo.
Alias
En determinadas circunstancias es necesario asignar un
nombre a alguna columna determinada de un conjunto devuelto, otras veces por
simple capricho o por otras circunstancias. Para resolver todas ellas tenemos
la palabra reservada AS que se encarga de asignar el nombre que deseamos a la
columna deseada. Tomado como referencia el ejemplo anterior podemos hacer que
la columna devuelta por la consulta, en lugar de llamarse apellido (igual que
el campo devuelto) se llame Empleado. En este caso procederíamos de la
siguiente forma:
SELECT DISTINCTROW Apellido AS Empleado FROM Empleados;
AS no es una palabra reservada de ANSI, existen
diferentes sistemas de asignar los alias en función del motor de bases de
datos. En ORACLE para asignar un alias a un campo hay que hacerlo de la
siguiente forma:
SELECT Apellido AS "Empleado" FROM Empleados;
También podemos asignar alias a las tablas dentro de
la consulta de selección, en esta caso hay que tener en cuenta que en todas las
referencias que deseemos hacer a dicha tabla se ha de utilizar el alias en
lugar del nombre. Esta técnica será de gran utilidad más adelante cuando se
estudien las vinculaciones entre tablas. Por ejemplo:
SELECT Apellido AS Empleado FROM Empleados AS Trabajadores;
Para asignar alias a las tablas en ORACLE y SQL-SERVER
los alias se asignan escribiendo el nombre de la tabla, dejando un espacio en blanco
y escribiendo el Alias (se asignan dentro de la cláusula FROM).
SELECT Trabajadores.Apellido AS Empleado FROM Empleados Trabajadores;
Esta nomenclatura [Tabla].[Campo] se debe utilizar cuando se está recuperando un campo
cuyo nombre se repite en varias de las tablas que se utilizan en la sentencia.
No obstante cuando en la sentencia se emplean varias tablas es aconsejable
utilizar esta nomenclatura para evitar el trabajo que supone al motor de datos
averiguar en que tabla está cada uno de los campos indicados en la cláusula SELECT.
Recuperar Información de una base de Datos Externa
Para concluir este capítulo se debe hacer referencia a
la recuperación de registros de bases de datos externa. En ocasiones es
necesario la recuperación de información que se encuentra contenida en una
tabla que no se encuentra en la base de datos que ejecutará la consulta o que
en ese momento no se encuentra abierta, esta situación la podemos salvar con la
palabra reservada IN de la siguiente forma:
SELECT DISTINCTROW Apellido AS Empleado FROM Empleados
IN 'c:\databases\gestion.mdb';
En donde c:\databases\gestion.mdb es la base de datos que contiene la tabla Empleados.
Esta técnica es muy sencilla y común en bases de datos de tipo ACCESS en otros
sistemas como SQL-SERVER u ORACLE, la cosa es más complicada la tener que
existir relaciones de confianza entre los servidores o al ser necesaria la
vinculación entre las bases de datos. Este ejemplo recupera la información de
una base de datos de SQL-SERVER ubicada en otro servidor (se da por supuesto
que los servidores están lincados):
SELECT Apellido FROM Servidor1.BaseDatos1.dbo.Empleados
No hay comentarios:
Publicar un comentario