Conflicto con Collation en consultas utilizando UNION ALL en SQL Server

8
15982
sql server collation

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.

collation

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

8 COMMENTS

  1. 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.

  2. 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

  3. 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.

Responder a Rolando Cancel reply

Please enter your comment!
Please enter your name here