Hace unos días me encontré con un problema en SQL Server al momento de unir datos de dos tablas de bases de datos distintas, tablas con la misma estructura pero con Collation distinto.
El problema con Collation en realidad es muy común si son bases de datos creados en servidores distintos, los cuales pueden que no estén configurados de la misma forma, más común si están en países distintos o idiomas distintos. Como dije es algo común y algo fácil de solventar.
Cuando se hace el SELECT a las columnas de texto se pueden definir el Collation ahí mismo, y ajustar los datos para el Union All.
Para hacer el ejemplo vamos a crear 3 tablas con Collation distinto, le ingresamos un par de registros y hacemos el ‘UNION ALL’ con el error y solucionado.
En una base de datos de prueba ejecutamos el siguiente script.
CREATE TABLE dbo.Table_1 ( codigo int NULL, nombre varchar(50) COLLATE SQL_Latin1_General_CP1_CI_AS NULL ) ON [PRIMARY] GO CREATE TABLE dbo.Table_2 ( codigo int NULL, nombre varchar(50) COLLATE Modern_Spanish_BIN NULL ) ON [PRIMARY] GO CREATE TABLE dbo.Table_3 ( codigo int NULL, nombre varchar(50) COLLATE SQL_Latin1_General_CP1_CS_AS NULL ) ON [PRIMARY] GO INSERT INTO Table_1 values(1,'Juan') INSERT INTO Table_1 values(2,'Marco') INSERT INTO Table_2 values(3,'Luis') INSERT INTO Table_2 values(4,'Maria') INSERT INTO Table_3 values(5,'Marta') INSERT INTO Table_3 values(6,'Jorge')
Como podemos ver, las tres tablas las creamos con la misma estructura pero con Collation distintos.
Problema
Podemos utilizar ‘Union’ o ‘Union All’ nos va a dar errores similares.
SELECT nombre FROM Table_1 UNION SELECT nombre FROM Table_2 UNION SELECT nombre FROM Table_3
Nos va a retornar errores similares a estos:
Msg 468, Level 16, State 9, Line 33
Cannot resolve the collation conflict between “SQL_Latin1_General_CP1_CS_AS” and “SQL_Latin1_General_CP1_CI_AS” in the UNION operation.
Msg 468, Level 16, State 9, Line 36
Cannot resolve the collation conflict between “SQL_Latin1_General_CP1_CS_AS” and “Modern_Spanish_BIN” in the UNION operation.
Los datos del error y Collation varían dependiendo de cómo estén configuradas tus tablas.
Con ‘Union All’ el resultado es similar
SELECT nombre FROM Table_1 UNION ALL SELECT nombre FROM Table_2 UNION ALL SELECT nombre FROM Table_3
Msg 457, Level 16, State 1, Line 34
Implicit conversion of varchar value to varchar cannot be performed because the collation of the value is unresolved due to a collation conflict between “SQL_Latin1_General_CP1_CS_AS” and “SQL_Latin1_General_CP1_CI_AS” in UNION ALL operator.
Msg 457, Level 16, State 1, Line 36
Implicit conversion of varchar value to varchar cannot be performed because the collation of the value is unresolved due to a collation conflict between “SQL_Latin1_General_CP1_CS_AS” and “Modern_Spanish_BIN” in UNION ALL operator.
Solución
Para solucionar el problema podemos cambiar en las columnas del Select el Collation el que nos sirva o bien dejar que use el Collation default del SQL.
Veamos algunos ejemplos:
-- Usamos el COLLATION DEFAULT SELECT nombre COLLATE database_default FROM Table_1 UNION ALL SELECT nombre COLLATE database_default FROM Table_2 UNION ALL SELECT nombre COLLATE database_default FROM Table_3 -- Usamos el COLLATION SQL_Latin1_General_CP1_CI_AS SELECT nombre COLLATE SQL_Latin1_General_CP1_CI_AS FROM Table_1 UNION SELECT nombre COLLATE SQL_Latin1_General_CP1_CI_AS FROM Table_2 UNION SELECT nombre COLLATE SQL_Latin1_General_CP1_CI_AS FROM Table_3
Con esto solventamos el problema y nos va a retornar los datos de forma correcta.
Dato extra: Cuál es la diferencia entre UNION y UNION ALL
UNION
El comando UNION se utiliza para seleccionar la información relacionada a partir de dos tablas, al igual que el comando JOIN. Sin embargo, cuando se utiliza el comando UNION todas las columnas seleccionadas deben ser del mismo tipo de datos. Con UNION, sólo se seleccionan valores distintos.
UNION ALL
El comando UNION ALL es igual al comando UNION, excepto que UNION ALL selecciona todos los valores.
La diferencia entre la UNION y UNION ALL todo es que UNION ALL no eliminará todas las filas duplicadas, sino que simplemente retorna todas las filas de todas las tablas que corresponden a su consulta específicos y las combina en un solo resultado.
Una instrucción UNION hace efectiva un DISTINCT SELECT en el conjunto de resultados. Si usted sabe que todos los registros devueltos son únicos entonces use UNION ALL que da resultados más rápidos.
Palabras claves:
- Sql Server union all collation conflict
- Collation conflict in SQL Union All Query
- Cannot resolve collation conflict in Union select
- How to use COLLATE with UNION
- Problemas con el collation un select union all
- SQL SERVER – Cannot resolve collation conflict
- Cannot resolve collation conflict for equal to operation
- SQL SERVER – Change Collation of Database Column – T-SQL Script
- mssql change collation database
- ALTER DATABASE failed. The default collation of database XX cannot be set to
Hola Roy buen día, estupenda colaboración, me gustaría si te es posible me pudieras ayudar un poco con un tema que refiere a migrar una base de datos access 2010 a una base de datos Sql Server 2008 R2, el detalle esta en que la BD Access tiene campos “Autonumerico” en diferentes tablas y en SQL Server se deben tener igual, de antemano agradezco tu apoyo, saludos
PD. Utilizo VB.NET 2010 para desarrollar la migración
para qué usas vb.net para hacer la migración?, eso lo puedes hacer desde el mismo sql server, importando los datos con el “SQL Server Import and Exports Wizard” y ahí selecionas el tipo de fuente, en este caso access y ahí sigues todos los pasos, es muy sencillo.
Hola Roy, buenos dias, sucede que trate de hacer un UNION ALL con dos Vistas, que tienen los mismos campos pero … la primera vista ACINOX.dbo.GENERAL_Contrato_Detalles los campos NOMBRE_Area, nombre y Representante de la BD ACINOX tienen el Colation en SQL_Latin1_General_CP1_CI_AS y en la otra tabla versax.dbo.Gen_Contratos_Import el collation es SQL_Latin1_General_Pref_CP850_CI_AS, trate de hacer lo que usted me explica aqui, y no me funcionan . Me puedes ayudar. Gracias de antemano
SELECT Id_Area, NOMBRE_Area, Numero, nombre, TipoContrato, Eatado , Representante, fecha, fechaTerm, NumRef, Observaciones
FROM ACINOX.dbo.GENERAL_Contrato_Detalles
UNION ALL
SELECT Id_Area, NOMBRE_Area, Numero, nombre, TipoContrato, Estado, Representante, fecha, fechaTerm, NumRef, Observaciones
FROM versax.dbo.Gen_Contratos_Import
hola, me parece que no estas poniendo el collation en las columnas correspondientes.
Deberias de hacerlo algo similar, no lo he probado con vistas, por que tambien puedes ponerle el collation directamente en la vista.
SELECT
Id_Area,
NOMBRE_Area COLLATE SQL_Latin1_General_CP1_CI_AS,
NumRef,
Observaciones COLLATE SQL_Latin1_General_CP1_CI_AS
FROM ACINOX.dbo.GENERAL_Contrato_Detalles
UNION ALL
SELECT
Id_Area,
NOMBRE_Area COLLATE SQL_Latin1_General_CP1_CI_AS,
NumRef,
Observaciones COLLATE SQL_Latin1_General_CP1_CI_AS
FROM versax.dbo.Gen_Contratos_Import
AHH mi versión es MS SQL Server 2008 R2
Disculpa Roy son dos views
Hola buen día Roy, tengo un problema con COLLATE, estoy aplicando COLLATE database_default a cada unos de los campos de la tabla alumnoParticipante de reciente creación como se muestra:
Select clave, nombres, primer_Apellido, segundo_Apellido, ‘PF’ as status, actividad
from alumno_pifi
where @anho_inicio=anho_inicio and folio=@folio –and clave_periodo=@periodo
union
Select clave, nombres, primer_Apellido, segundo_Apellido, ‘SS’ as status, actividad
from prestante_social
where @anho_inicio=anho_inicio and folio=@folio –and clave_periodo=@periodo
union
Select clave, nombres, primer_Apellido, segundo_Apellido, ‘TS’ as status, actividad
from tesista
where @anho_inicio=anho_inicio and folio=@folio –and clave_periodo=@periodo
union
Select clave, nombres, primer_Apellido, segundo_Apellido, ‘PP’ as status, actividad
from practicante_profesional
where @anho_inicio=anho_inicio and folio=@folio –and clave_periodo=@periodo
union
Select clave COLLATE database_default, nombres COLLATE database_default, primer_Apellido COLLATE database_default, segundo_Apellido COLLATE database_default, ‘AP’ as status, actividad COLLATE database_default
from alumnoParticipante
where @anho_inicio=anho_inicio and folio=@folio –and clave_periodo=@periodo
) /*order by primer_Apellido, segundo_Apellido, nombres*/
y me muestra el siguiente error:
Msg 447, Level 16, State 0, Procedure cgpi_DameAlumnos, Line 31
Expression type int is invalid for COLLATE clause.
Podrías ayudarme con este problema? Agradezco tu atención.
Hola, por lo que veo no estas aplicando el collation en todas las subconsultas, es probable que alguno de los selects en los union tenga un collation distinto. Yo te recomiendo que si no estas segura de cual collation es el default le indiques explícitamente.