Resolver problemas comunes con JSON en SQL Server

Se aplica a: SQL Server 2016 (13.x) y versiones posteriores de Azure SQL DatabaseAzure SQL Managed InstanceAzure Synapse Analytics (solo grupo de SQL sin servidor)

Encuentre respuestas en este artículo a algunas preguntas habituales sobre la compatibilidad integrada de JSON en SQL Server.

Salida de FOR JSON y JSON

¿FOR JSON PATH o FOR JSON AUTO?

Pregunta. Quiero crear un resultado de texto JSON a partir de una consulta SQL simple en una sola tabla. FOR JSON PATH y FOR JSON AUTO generan el mismo resultado. ¿Cuál de estas dos opciones debo usar?

Respuesta. Use FOR JSON PATH. Aunque no hay ninguna diferencia en el resultado JSON, el modo AUTO aplica alguna lógica adicional que comprueba si se deben anidar columnas. Considere la posibilidad de usar PATH como la opción predeterminada.

Creación de una estructura anidada JSON

Pregunta. Quiero generar texto JSON complejo con varias matrices del mismo nivel. FOR JSON PATH puede crear objetos anidados con rutas de acceso y FOR JSON AUTO crea el nivel de anidamiento adicional para cada tabla. Ninguna de estas dos opciones me permite generar el resultado que quiero. ¿Cómo puedo crear un formato JSON personalizado que no admitan directamente las opciones existentes?

Respuesta. Puede crear cualquier estructura de datos si agrega consultas FOR JSON como expresiones de columna que devuelven texto JSON. También puede crear JSON de forma manual mediante la función JSON_QUERY. En el ejemplo siguiente, se muestran estas técnicas.

SELECT col1, col2, col3,  
     (SELECT col11, col12, col13 FROM t11 WHERE t11.FK = t1.PK FOR JSON PATH) as t11,  
     (SELECT col21, col22, col23 FROM t21 WHERE t21.FK = t1.PK FOR JSON PATH) as t21,  
     (SELECT col31, col32, col33 FROM t31 WHERE t31.FK = t1.PK FOR JSON PATH) as t31,  
     JSON_QUERY('{"'+col4+'":"'+col5+'"}') as t41  
FROM t1  
FOR JSON PATH  

Cada resultado de una consulta FOR JSON o la función JSON_QUERY en las expresiones de columna se formatea como un subobjeto JSON anidado independiente y se incluye en el resultado principal.

Prevención de texto JSON con caracteres de doble escape en el resultado FOR JSON

Pregunta. Tengo texto JSON almacenado en una columna de tabla. Quiero incluirlo en el resultado de FOR JSON. Pero FOR JSON aplica caracteres de escape a todos los caracteres JSON, por lo que obtengo una cadena JSON en lugar de un objeto anidado, como se muestra en el ejemplo siguiente.

SELECT 'Text' AS myText, '{"day":23}' AS myJson  
FOR JSON PATH  

Esta consulta genera el siguiente resultado.

[{"myText":"Text", "myJson":"{\"day\":23}"}]  

¿Cómo puedo evitar este comportamiento? Quiero que se devuelva {"day":23} como un objeto JSON y no como texto con caracteres de escape.

Respuesta. Los textos JSON almacenados en una columna de texto o un literal se tratan como cualquier texto. Es decir, va precedido y seguido de comillas dobles y caracteres de escape. Si quiere que se devuelva un objeto JSON sin caracteres de escape, pase la columna JSON como argumento a la función JSON_QUERY, tal y como se muestra en el ejemplo siguiente.

SELECT col1, col2, col3, JSON_QUERY(jsoncol1) AS jsoncol1  
FROM tab1  
FOR JSON PATH  

JSON_QUERY sin su segundo parámetro opcional devuelve solo el primer argumento como resultado. Puesto que JSON_QUERY siempre devuelve texto JSON válido, FOR JSON reconoce que este resultado no tiene que ir con caracteres de escape.

Texto JSON generado con la cláusula WITHOUT_ARRAY_WRAPPER y con caracteres de escape en el resultado FOR JSON

Pregunta. Estoy tratando de dar formato a una expresión de columna mediante FOR JSON y la opción WITHOUT_ARRAY_WRAPPER.

SELECT 'Text' as myText,  
   (SELECT 12 day, 8 mon FOR JSON PATH, WITHOUT_ARRAY_WRAPPER) as myJson  
FOR JSON PATH   

Parece que al texto que devuelve la consulta FOR JSON se le aplican caracteres de escape como texto sin formato. Esto solo sucede si se especifica WITHOUT_ARRAY_WRAPPER. ¿Por qué no se trata como un objeto JSON y se incluye sin caracteres de escape en el resultado?

Respuesta. Si especifica la opción WITHOUT_ARRAY_WRAPPER en el valor FOR JSON interno, el texto JSON resultante no es necesariamente un JSON válido. Por tanto, la consulta FOR JSON exterior da por hecho que esto es texto sin formato y aplica caracteres de escape a la cadena. Si está seguro de que el resultado JSON es válido, inclúyalo con la función JSON_QUERY para promoverlo a JSON con formato correcto, tal y como se muestra en el ejemplo siguiente.

SELECT 'Text' as myText,  
      JSON_QUERY((SELECT 12 day, 8 mon FOR JSON PATH, WITHOUT_ARRAY_WRAPPER)) as myJson  
FOR JSON PATH    

Entrada de OPENJSON y JSON

Devolución de un subobjeto JSON anidado a partir de texto JSON con OPENJSON

Pregunta. No puedo abrir una matriz de objetos JSON complejos que contiene matrices, objetos y valores escalares empleando OPENJSON con un esquema explícito. Cuando hago referencia a una clave en la cláusula WITH, se devuelven solo los valores escalares. Los objetos y las matrices se devuelven como valores NULL. ¿Cómo puedo extraer objetos o matrices como objetos JSON?

Respuesta. Si quiere devolver un objeto o matriz como una columna, use la opción AS JSON en la definición de columna, tal y como se muestra en el ejemplo siguiente.

SELECT scalar1, scalar2, obj1, obj2, arr1  
FROM OPENJSON(@json)  
    WITH ( scalar1 int,  
        scalar2 datetime2,  
        obj1 NVARCHAR(MAX) AS JSON,  
        obj2 NVARCHAR(MAX) AS JSON,  
        arr1 NVARCHAR(MAX) AS JSON)  

Devolver valores de texto largo con OPENJSON en lugar de JSON_VALUE

Pregunta. Tengo una clave de descripción en JSON que contiene texto largo. JSON_VALUE(@json, '$.description') devuelve NULL en lugar de un valor.

Respuesta. JSON_VALUE se ha diseñado para devolver valores escalares de tamaño reducido. Generalmente, la función devuelve NULL en lugar de un error de desbordamiento. Si quiere que se devuelvan valores de mayor tamaño, utilice OPENJSON, que admite valores NVARCHAR(MAX), tal y como se muestra en el ejemplo siguiente.

SELECT myText FROM OPENJSON(@json) WITH (myText NVARCHAR(MAX) '$.description')  

Controlar claves duplicadas con OPENJSON en lugar de JSON_VALUE

Pregunta. Tengo claves duplicadas en el texto JSON. JSON_VALUE devuelve solo la primera clave que se encuentra en la ruta de acceso. ¿Cómo puedo devolver todas las claves que tengan el mismo nombre?

Respuesta. Las funciones escalares JSON integradas devuelven solo la primera aparición del objeto de referencia. Si necesita más de una clave, utilice la función con valores de tabla OPENJSON, tal y como se muestra en el ejemplo siguiente.

SELECT value FROM OPENJSON(@json, '$.info.settings')  
WHERE [key] = 'color'  

OPENJSON requiere el nivel de compatibilidad 130

Pregunta. Estoy tratando de ejecutar OPENJSON en SQL Server 2016 y obtengo el error siguiente.

Msg 208, Level 16, State 1 'Invalid object name OPENJSON'

Respuesta. La función OPENJSON solo está disponible en el nivel de compatibilidad 130. Si el nivel de compatibilidad de base de datos es inferior a 130, OPENJSON estará oculto. Hay otras funciones JSON que sí están disponibles en todos los niveles de compatibilidad.

Otras preguntas

Claves de referencia que contienen caracteres no alfanuméricos en texto JSON

Pregunta. Tengo caracteres no alfanuméricos en claves de mi texto JSON. ¿Cómo puedo hacer referencia a estas propiedades?

Respuesta. Tiene que incluirlas entre comillas en las rutas de acceso JSON. Por ejemplo, JSON_VALUE(@json, '$."$info"."First Name".value').

Más información sobre JSON en SQL Server y Azure SQL Database

Vídeos de Microsoft

Nota:

Es posible que algunos de los vínculos de vídeo de esta sección no funcionen en este momento. Microsoft está migrando contenido que anteriormente estaba en Channel 9 a una nueva plataforma. Actualizaremos los vínculos a medida que los vídeos se migren a la nueva plataforma.

Para obtener una introducción visual a la compatibilidad integrada de JSON en SQL Server y Azure SQL Database, vea los siguientes vídeos: