2010-05-19 7 views
92

Est-il possible d'analyser JSON dans TSQL? Je ne veux pas créer une chaîne JSON, je veux dire analyser une chaîne JSON passée en paramètre.Parse JSON dans TSQL

Répondre

46

Mise à jour: A partir de SQL Server 2016 parsing JSON in TSQL is now possible.

Nativement, il n'y a pas de support. Vous devrez utiliser CLR. C'est aussi simple que cela, sauf si vous avez une énorme tendance masochiste et que vous voulez écrire un analyseur JSON dans SQL

Normalement, les gens demandent une sortie JSON de la base de données et il existe des exemples sur Internet. Mais dans une base de données?

+0

Parse JSON sur le côté appelant de la base de données peut être une bonne idée :) –

+55

JSON est un protocole assez simple, donc il ne nécessite pas vraiment une énorme quantité de masochisme. Une fois que vous l'avez, vous pouvez utiliser la routine pour tous vos JSON. Quoi qu'il en soit, je l'ai fait pour vous ici http://www.simple-talk.com/sql/t-sql-programming/consuming-json-strings-in-sql-server/ –

+9

Phil Factor: J'ai été lire vos articles pendant de nombreuses années. Si vous n'aviez pas écrit cet article * aujourd'hui * je l'aurais probablement cité il y a 6 mois quand j'ai répondu ... – gbn

212

Je semble avoir une énorme tendance masochiste en ce sens que j'ai écrit un analyseur JSON. Il convertit un document JSON dans une table de liste SQL Adjacency, facile à utiliser pour mettre à jour vos tables de données. En fait, j'ai fait pire, en ce que j'ai fait du code pour faire le processus inverse, qui est de passer d'une table hiérarchique à une chaîne JSON

L'article et le code est ici: Consuming Json strings in SQL server.

Select * from parseJSON('{ 
    "Person": 
    { 
    "firstName": "John", 
    "lastName": "Smith", 
    "age": 25, 
    "Address": 
    { 
     "streetAddress":"21 2nd Street", 
     "city":"New York", 
     "state":"NY", 
     "postalCode":"10021" 
    }, 
    "PhoneNumbers": 
    { 
     "home":"212 555-1234", 
     "fax":"646 555-4567" 
    } 
    } 
} 
') 

Pour obtenir:

enter image description here

+0

Ceci est une grande fonctionnalité, mais a certaines limitations, par exemple dépouiller le "-" des nombres négatifs. – Gavin

+0

très cool! vous avez une faute de frappe dans le script: IF OBJECT_ID (N'dbo.parseJSON ') EST PAS NULL DROP FUNCTION dbo.JSONEscaped GO - devrait tester pour dbo.JSONEscaped dans le test IF. – isapir

+0

@phil dbo.parseJSON travaille très lentement en cas de données volumineuses. pouvons-nous réduire le temps en utilisant d'autres méthodes? – cracker

-1

J'ai vu un article très soigné à ce sujet ... donc si vous aimez ceci:

CREATE PROC [dbo].[spUpdateMarks] 
    @inputJSON VARCHAR(MAX) -- '[{"ID":"1","C":"60","CPP":"60","CS":"60"}]' 
AS 
BEGIN 
    -- Temp table to hold the parsed data 
    DECLARE @TempTableVariable TABLE(
     element_id INT, 
     sequenceNo INT, 
     parent_ID INT, 
     [Object_ID] INT, 
     [NAME] NVARCHAR(2000), 
     StringValue NVARCHAR(MAX), 
     ValueType NVARCHAR(10) 
    ) 
    -- Parse JSON string into a temp table 
    INSERT INTO @TempTableVariable 
    SELECT * FROM parseJSON(@inputJSON) 
END 

Essayez de regardez ici:

https://www.simple-talk.com/sql/t-sql-programming/consuming-json-strings-in-sql-server/

Il y a un projet ASP.Net sur cette ici: http://www.codeproject.com/Articles/788208/Update-Multiple-Rows-of-GridView-using-JSON-in-ASP

+6

Il existe déjà une réponse par l'auteur original de l'article: http://stackoverflow.com/a/4187412/389424 – janv8000

23

Enfin SQL Server 2016 ajoutera le support natif JSON !!

Ref:

capacités supplémentaires dans SQL Server 2016 comprennent:

  • améliorations de sécurité supplémentaires pour la sécurité de niveau ligne et Dynamic Data Masking pour arrondir nos investissements dans la sécurité avec toujours
    Encrypted.
  • Améliorations de AlwaysOn pour une disponibilité plus robuste et une reprise après sinistre avec plusieurs réplicas synchrones et charges secondaires
    équilibrage.
  • Prise en charge JSON natif pour offrir de meilleures performances et une meilleure prise en charge de vos nombreux types de données.
  • Les outils EIM (Enterprise Information Management) de SQL Server et Analysis Services bénéficient d'une évolutivité en termes de performances, de convivialité et d'évolutivité .
  • Sauvegardes hybrides plus rapides, scénarios de haute disponibilité et de reprise après sinistre pour sauvegarder et restaurer vos bases de données locales sur Azure
    et de placer vos fichiers secondaires SQL Server AlwaysOn dans Azure.

Announcment: http://blogs.technet.com/b/dataplatforminsider/archive/2015/05/04/sql-server-2016-public-preview-coming-this-summer.aspx

Caractéristiques de blog: http://blogs.msdn.com/b/jocapc/archive/2015/05/16/json-support-in-sql-server-2016.aspx

+2

Probablement SQL Server 2016 CTP 3 aura le support de JSON à SQL Server avec la syntaxe OpenJSON: http: // www .kodyaz.com/t-sql/sql-serveur-2016-openjson-error.aspx – Eralper

4

J'ai aussi une énorme séquence masochistes comme je l'ai écrit un autre analyseur de JSON. Celui-ci utilise une approche procédurale. Il utilise une table de liste de hiérarchie SQL similaire pour stocker les données analysées. De plus dans le paquet sont:

  • processus inverse: de la hiérarchie à JSON
  • fonctions Interrogation: chercher des valeurs particulières d'un objet JSON

S'il vous plaît ne hésitez pas à utiliser et amusez-vous avec elle

http://www.codeproject.com/Articles/1000953/JSON-for-Sql-Server-Part

+0

+1 merci fonctionne très bien, mieux que la version de PhilFactor. J'ai dû baisser légèrement pour SQL Server 2008 (pas de fonction 'iif' ou' OFFSET') – Geronimo

4
CREATE FUNCTION dbo.parseJSON(@JSON NVARCHAR(MAX)) 
RETURNS @hierarchy TABLE 
    (
    element_id INT IDENTITY(1, 1) NOT NULL, /* internal surrogate primary key gives the order of parsing and the list order */ 
    sequenceNo [int] NULL, /* the place in the sequence for the element */ 
    parent_ID INT,/* if the element has a parent then it is in this column. The document is the ultimate parent, so you can get the structure from recursing from the document */ 
    Object_ID INT,/* each list or object has an object id. This ties all elements to a parent. Lists are treated as objects here */ 
    NAME NVARCHAR(2000),/* the name of the object */ 
    StringValue NVARCHAR(MAX) NOT NULL,/*the string representation of the value of the element. */ 
    ValueType VARCHAR(10) NOT null /* the declared type of the value represented as a string in StringValue*/ 
) 
AS 
BEGIN 
    DECLARE 
    @FirstObject INT, --the index of the first open bracket found in the JSON string 
    @OpenDelimiter INT,--the index of the next open bracket found in the JSON string 
    @NextOpenDelimiter INT,--the index of subsequent open bracket found in the JSON string 
    @NextCloseDelimiter INT,--the index of subsequent close bracket found in the JSON string 
    @Type NVARCHAR(10),--whether it denotes an object or an array 
    @NextCloseDelimiterChar CHAR(1),--either a '}' or a ']' 
    @Contents NVARCHAR(MAX), --the unparsed contents of the bracketed expression 
    @Start INT, --index of the start of the token that you are parsing 
    @end INT,--index of the end of the token that you are parsing 
    @param INT,--the parameter at the end of the next Object/Array token 
    @EndOfName INT,--the index of the start of the parameter at end of Object/Array token 
    @token NVARCHAR(200),--either a string or object 
    @value NVARCHAR(MAX), -- the value as a string 
    @SequenceNo int, -- the sequence number within a list 
    @name NVARCHAR(200), --the name as a string 
    @parent_ID INT,--the next parent ID to allocate 
    @lenJSON INT,--the current length of the JSON String 
    @characters NCHAR(36),--used to convert hex to decimal 
    @result BIGINT,--the value of the hex symbol being parsed 
    @index SMALLINT,--used for parsing the hex value 
    @Escape INT --the index of the next escape character 


    DECLARE @Strings TABLE /* in this temporary table we keep all strings, even the names of the elements, since they are 'escaped' in a different way, and may contain, unescaped, brackets denoting objects or lists. These are replaced in the JSON string by tokens representing the string */ 
    (
    String_ID INT IDENTITY(1, 1), 
    StringValue NVARCHAR(MAX) 
    ) 
    SELECT--initialise the characters to convert hex to ascii 
    @characters='abcdefghijklmnopqrstuvwxyz', 
    @SequenceNo=0, --set the sequence no. to something sensible. 
    /* firstly we process all strings. This is done because [{} and ] aren't escaped in strings, which complicates an iterative parse. */ 
    @parent_ID=0; 
    WHILE 1=1 --forever until there is nothing more to do 
    BEGIN 
     SELECT 
     @start=PATINDEX('%[^a-zA-Z]["]%', @json collate SQL_Latin1_General_CP850_Bin);--next delimited string 
     IF @start=0 BREAK --no more so drop through the WHILE loop 
     IF SUBSTRING(@json, @start+1, 1)='"' 
     BEGIN --Delimited Name 
      SET @[email protected]+1; 
      SET @end=PATINDEX('%[^\]["]%', RIGHT(@json, LEN(@json+'|')[email protected]) collate SQL_Latin1_General_CP850_Bin); 
     END 
     IF @end=0 --no end delimiter to last string 
     BREAK --no more 
     SELECT @token=SUBSTRING(@json, @start+1, @end-1) 
     --now put in the escaped control characters 
     SELECT @token=REPLACE(@token, FROMString, TOString) 
     FROM 
     (SELECT 
      '\"' AS FromString, '"' AS ToString 
     UNION ALL SELECT '\\', '\' 
     UNION ALL SELECT '\/', '/' 
     UNION ALL SELECT '\b', CHAR(08) 
     UNION ALL SELECT '\f', CHAR(12) 
     UNION ALL SELECT '\n', CHAR(10) 
     UNION ALL SELECT '\r', CHAR(13) 
     UNION ALL SELECT '\t', CHAR(09) 
     ) substitutions 
     SELECT @result=0, @escape=1 
    --Begin to take out any hex escape codes 
     WHILE @escape>0 
     BEGIN 
      SELECT @index=0, 
      --find the next hex escape sequence 
      @escape=PATINDEX('%\x[0-9a-f][0-9a-f][0-9a-f][0-9a-f]%', @token collate SQL_Latin1_General_CP850_Bin) 
      IF @escape>0 --if there is one 
      BEGIN 
       WHILE @index<4 --there are always four digits to a \x sequence 
       BEGIN 
        SELECT --determine its value 
        @[email protected]+POWER(16, @index) 
        *(CHARINDEX(SUBSTRING(@token, @[email protected], 1), 
           @characters)-1), @[email protected]+1 ; 

       END 
       -- and replace the hex sequence by its unicode value 
       SELECT @token=STUFF(@token, @escape, 6, NCHAR(@result)) 
      END 
     END 
     --now store the string away 
     INSERT INTO @Strings (StringValue) SELECT @token 
     -- and replace the string with a token 
     SELECT @JSON=STUFF(@json, @start, @end+1, 
        '@string'+CONVERT(NVARCHAR(5), @@identity)) 
    END 
    -- all strings are now removed. Now we find the first leaf. 
    WHILE 1=1 --forever until there is nothing more to do 
    BEGIN 

    SELECT @[email protected]_ID+1 
    --find the first object or list by looking for the open bracket 
    SELECT @FirstObject=PATINDEX('%[{[[]%', @json collate SQL_Latin1_General_CP850_Bin)--object or array 
    IF @FirstObject = 0 BREAK 
    IF (SUBSTRING(@json, @FirstObject, 1)='{') 
    SELECT @NextCloseDelimiterChar='}', @type='object' 
    ELSE 
    SELECT @NextCloseDelimiterChar=']', @type='array' 
    SELECT @[email protected] 

    WHILE 1=1 --find the innermost object or list... 
    BEGIN 
     SELECT 
     @lenJSON=LEN(@JSON+'|')-1 
    --find the matching close-delimiter proceeding after the open-delimiter 
     SELECT 
     @NextCloseDelimiter=CHARINDEX(@NextCloseDelimiterChar, @json, 
             @OpenDelimiter+1) 
    --is there an intervening open-delimiter of either type 
     SELECT @NextOpenDelimiter=PATINDEX('%[{[[]%', 
      RIGHT(@json, @[email protected])collate SQL_Latin1_General_CP850_Bin)--object 
     IF @NextOpenDelimiter=0 
     BREAK 
     SELECT @[email protected][email protected] 
     IF @NextCloseDelimiter<@NextOpenDelimiter 
     BREAK 
     IF SUBSTRING(@json, @NextOpenDelimiter, 1)='{' 
     SELECT @NextCloseDelimiterChar='}', @type='object' 
     ELSE 
     SELECT @NextCloseDelimiterChar=']', @type='array' 
     SELECT @[email protected] 
    END 
    ---and parse out the list or name/value pairs 
    SELECT 
    @contents=SUBSTRING(@json, @OpenDelimiter+1, 
         @[email protected]) 
    SELECT 
    @JSON=STUFF(@json, @OpenDelimiter, 
       @[email protected]+1, 
       '@'[email protected]+CONVERT(NVARCHAR(5), @parent_ID)) 
    WHILE (PATINDEX('%[[email protected]+.e]%', @contents collate SQL_Latin1_General_CP850_Bin))<>0 
    BEGIN 
     IF @Type='Object' --it will be a 0-n list containing a string followed by a string, number,boolean, or null 
     BEGIN 
      SELECT 
      @SequenceNo=0,@end=CHARINDEX(':', ' '[email protected])--if there is anything, it will be a string-based name. 
      SELECT @start=PATINDEX('%[^[email protected]][@]%', ' '[email protected] collate SQL_Latin1_General_CP850_Bin)--AAAAAAAA 
      SELECT @token=SUBSTRING(' '[email protected], @start+1, @[email protected]), 
      @endofname=PATINDEX('%[0-9]%', @token collate SQL_Latin1_General_CP850_Bin), 
      @param=RIGHT(@token, LEN(@token)[email protected]+1) 
      SELECT 
      @token=LEFT(@token, @endofname-1), 
      @Contents=RIGHT(' '[email protected], LEN(' '[email protected]+'|')[email protected]) 
      SELECT @name=stringvalue FROM @strings 
      WHERE [email protected] --fetch the name 
     END 
     ELSE 
     SELECT @Name=null,@[email protected]+1 
     SELECT 
     @end=CHARINDEX(',', @contents)-- a string-token, object-token, list-token, number,boolean, or null 
     IF @end=0 
     SELECT @end=PATINDEX('%[[email protected]+.e][^[email protected]+.e]%', @Contents+' ' collate SQL_Latin1_General_CP850_Bin) 
      +1 
     SELECT 
     @start=PATINDEX('%[^[email protected]+.e][[email protected]+.e]%', ' '[email protected] collate SQL_Latin1_General_CP850_Bin) 
     --select @start,@end, LEN(@contents+'|'), @contents 
     SELECT 
     @Value=RTRIM(SUBSTRING(@contents, @start, @[email protected])), 
     @Contents=RIGHT(@contents+' ', LEN(@contents+'|')[email protected]) 
     IF SUBSTRING(@value, 1, 7)='@object' 
     INSERT INTO @hierarchy 
      (NAME, SequenceNo, parent_ID, StringValue, Object_ID, ValueType) 
      SELECT @name, @SequenceNo, @parent_ID, SUBSTRING(@value, 8, 5), 
      SUBSTRING(@value, 8, 5), 'object' 
     ELSE 
     IF SUBSTRING(@value, 1, 6)='@array' 
      INSERT INTO @hierarchy 
      (NAME, SequenceNo, parent_ID, StringValue, Object_ID, ValueType) 
      SELECT @name, @SequenceNo, @parent_ID, SUBSTRING(@value, 7, 5), 
       SUBSTRING(@value, 7, 5), 'array' 
     ELSE 
      IF SUBSTRING(@value, 1, 7)='@string' 
      INSERT INTO @hierarchy 
       (NAME, SequenceNo, parent_ID, StringValue, ValueType) 
       SELECT @name, @SequenceNo, @parent_ID, stringvalue, 'string' 
       FROM @strings 
       WHERE string_id=SUBSTRING(@value, 8, 5) 
      ELSE 
      IF @value IN ('true', 'false') 
       INSERT INTO @hierarchy 
       (NAME, SequenceNo, parent_ID, StringValue, ValueType) 
       SELECT @name, @SequenceNo, @parent_ID, @value, 'boolean' 
      ELSE 
       IF @value='null' 
       INSERT INTO @hierarchy 
        (NAME, SequenceNo, parent_ID, StringValue, ValueType) 
        SELECT @name, @SequenceNo, @parent_ID, @value, 'null' 
       ELSE 
       IF PATINDEX('%[^0-9]%', @value collate SQL_Latin1_General_CP850_Bin)>0 
        INSERT INTO @hierarchy 
        (NAME, SequenceNo, parent_ID, StringValue, ValueType) 
        SELECT @name, @SequenceNo, @parent_ID, @value, 'real' 
       ELSE 
        INSERT INTO @hierarchy 
        (NAME, SequenceNo, parent_ID, StringValue, ValueType) 
        SELECT @name, @SequenceNo, @parent_ID, @value, 'int' 
     if @Contents=' ' Select @SequenceNo=0 
    END 
    END 
INSERT INTO @hierarchy (NAME, SequenceNo, parent_ID, StringValue, Object_ID, ValueType) 
    SELECT '-',1, NULL, '', @parent_id-1, @type 
-- 
    RETURN 
END 
GO 

--- Pase JSON

Declare @pars varchar(MAX) = 
' {"shapes":[{"type":"polygon","geofenceName":"","geofenceDescription":"", 
"geofenceCategory":"1","color":"#1E90FF","paths":[{"path":[{ 
"lat":"26.096254906968525","lon":"65.709228515625"} 
,{"lat":"28.38173504322308","lon":"66.741943359375"} 
,{"lat":"26.765230565697482","lon":"68.983154296875"} 
,{"lat":"26.254009699865737","lon":"68.609619140625"} 
,{"lat":"25.997549919572112","lon":"68.104248046875"} 
,{"lat":"26.843677401113002","lon":"67.115478515625"} 
,{"lat":"25.363882272740255","lon":"65.819091796875"}]}]}]}' 
Select * from parseJSON(@pars) AS MyResult 
1

SQL Server 2016 prend en charge json data l'analyse syntaxique en utilisant OPENJSON. Vous pouvez utiliser OPENJSON pour mapper json data aux lignes et aux colonnes.

Votre json Data

[ 
{ "id" : 2,"name": "John"}, 
{ "id" : 5,"name": "John"} 
] 

Voici comment vous pouvez gérer JSON dans sql

//@pJson is json data passed from code. 

INSERT INTO YourTable (id, Name) 
SELECT id, name 
FROM OPENJSON(@pJson) 
WITH (id int, 
     name nvarchar(max)) 

Here est un article détaillé qui couvre ce sujet.