20 SQL/JSON Function JSON_TABLE
SQL/JSON function json_table projects specific JSON data
    to columns of various SQL data types. You use it to map parts of a JSON document into the rows
    and columns of a new, virtual table, which you can also think of as an inline view.
               
You can then insert this virtual table into a pre-existing database table, or you can query it using SQL — in a join expression, for example.
A common
        use of json_table is to create a view of JSON data. You can use such
        a view just as you would use any table or view. This lets applications, tools, and
        programmers operate on JSON data without consideration of the syntax of JSON or JSON path
        expressions.
                  
Defining a view over JSON data in effect maps a kind of schema onto that data. This mapping is after the fact: the underlying JSON data can be defined and created without any regard to a schema or any particular pattern of use. Data first, schema later.
Such a schema (mapping) imposes no restriction on the kind of JSON documents that can be stored in the database (other than being well-formed JSON data). The view exposes only data that conforms to the mapping (schema) that defines the view. To change the schema, just redefine the view — no need to reorganize the underlying JSON data.
You use json_table
        in a SQL FROM clause. It is a row source: it
        generates a row of virtual-table data for each JSON value selected by a row path
          expression (row pattern). The columns of each generated row are defined by the
          column path expressions of the COLUMNS clause.
                  
Typically a json_table invocation is laterally joined, implicitly, with a
        source table in the FROM list, whose rows each contain a JSON document that
        is used as input to the function. json_table generates zero or more new
        rows, as determined by evaluating the row path expression against the input document.
                  
The first argument to json_table is a SQL expression. It can
        be a table or view column value, a PL/SQL variable, or a bind variable with proper casting.
        The result of evaluating the expression is used as the context item for evaluating
        the row path expression.
                  
The second argument to
          json_table is the SQL/JSON row path expression followed by an optional
        error clause for handling the row and the (required) COLUMNS clause, which
        defines the columns of the virtual table to be created. There is no
          RETURNING clause. 
                  
There are two levels of error
        handling for json_table, corresponding to the two levels of path
        expressions: row and column. When present, a column error handler overrides row-level error
        handling. The default error handler for both levels is NULL ON ERROR.
        
                  
In a row path-expression array step, the order of indexes and ranges, multiple occurrences of an array index, and duplication of a specified position due to range overlaps all have the usual effect: the specified positions are matched, in order, against the data, producing one row for each position match.
As an alternative to passing the context-item argument and the row path
        expression, you can use simple dot-notation syntax. (You can still use an error clause, and
        the COLUMNS clause is still required.) Dot notation specifies a table or
        view column together with a simple path to the targeted JSON data. For example, these two
        queries are
        equivalent:
                  
json_table(t.j, '$.ShippingInstructions.Phone[*]' ...)
json_table(t.j.ShippingInstructions.Phone[*] ...)And in cases where the row path expression is only '$', which
        targets the entire document, you can omit the path part. These queries are
        equivalent:
                  
json_table(t.j, '$' ...)
json_table(t.j ...)Example 20-1 illustrates the difference between using the simple dot notation and using the fuller, more explicit notation.
You can also use the dot notation in any PATH clause of a
          COLUMNS clause, as an alternative to using a SQL/JSON path expression.
        For example, you can use just PATH 'ShippingInstructions.name' instead of
          PATH '$.ShippingInstructions.name'.
                  
Example 20-1 Equivalent JSON_TABLE Queries: Simple and Full Syntax
This example uses json_table for two equivalent queries. The
        first query uses the simple, dot-notation syntax for the expressions that target the row and
        column data. The second uses the full syntax.
                  
Except for column
          Special Instructions, whose SQL identifier is quoted, the SQL column
        names are, in effect, uppercase. (Identifier Special Instructions contains
        a space character.)
                  
In the first query the column names are written
        exactly the same as the names of the targeted object fields, including with respect to
        letter case. Regardless of whether they are quoted, they are interpreted case-sensitively
        for purposes of establishing the default path (the path used when there is no explicit
          PATH clause).
                  
The second query has:
- 
                        Separate arguments of a JSON column-expression and a SQL/JSON row path-expression 
- 
                        Explicit column data types of VARCHAR2(4000)
- 
                        Explicit PATHclauses with SQL/JSON column path expressions, to target the object fields that are projected
SELECT jt.*
  FROM j_purchaseorder po,
       json_table(po.po_document
         COLUMNS ("Special Instructions",
                  NESTED LineItems[*]
                    COLUMNS (ItemNumber NUMBER,
                             Description PATH Part.Description))
       ) AS "JT";SELECT jt.*
  FROM j_purchaseorder po,
       json_table(po.po_document, 
         '$'
         COLUMNS (
           "Special Instructions" VARCHAR2(4000)
                                  PATH '$."Special Instructions"',
           NESTED PATH '$.LineItems[*]'
             COLUMNS (
               ItemNumber  NUMBER        PATH '$.ItemNumber',
               Description VARCHAR(4000) PATH '$.Part.Description'))
       ) AS "JT";
See Also:
Oracle Database SQL
                                        Language Reference for information about
        json_table
- SQL NESTED Clause Instead of JSON_TABLE
 In aSELECTclause you can often use aNESTEDclause instead of SQL/JSON functionjson_table. This can mean a simpler query expression. It also has the advantage of including rows with non-NULLrelational columns when the JSON column isNULL.
- COLUMNS Clause of SQL/JSON Function JSON_TABLE
 The mandatoryCOLUMNSclause for SQL/JSON functionjson_tabledefines the columns of the virtual table that the function creates.
- JSON_TABLE Generalizes SQL/JSON Query Functions and Conditions
 SQL/JSON functionjson_tablegeneralizes SQL/JSON conditionjson_existsand SQL/JSON functionsjson_valueandjson_query. Everything that you can do using these functions you can do usingjson_table. For the jobs they accomplish, the syntax of these functions is simpler to use than is the syntax ofjson_table.
- Using JSON_TABLE with JSON Arrays
 A JSON value can be an array or can include one or more arrays, nested to any number of levels inside other JSON arrays or objects. You can use ajson_tableNESTEDpath clause to project specific elements of an array.
- Creating a View Over JSON Data Using JSON_TABLE
 To improve query performance you can create a view over JSON data that you project to columns using SQL/JSON functionjson_table. To further improve query performance you can create a materialized view and place the JSON data in memory.
Related Topics
Parent topic: Query JSON Data
20.1 SQL NESTED Clause Instead of JSON_TABLE
In a SELECT clause you can often use a
            NESTED clause instead of SQL/JSON function json_table.
        This can mean a simpler query expression. It also has the advantage of including rows with
            non-NULL relational columns when the JSON column is
            NULL.
                  
The NESTED clause is a shortcut for using
                json_table with an ANSI left outer join. That is, these two queries
            are equivalent:
                  
SELECT ... 
  FROM mytable NESTED jcol COLUMNS (...);SELECT ...
  FROM mytable t1 LEFT OUTER JOIN
       json_table(t1.jcol COLUMNS (...)
       ON 1=1;Using a left outer join with json_table, or using the
                NESTED clause, allows the selection result to include rows with
            relational columns where there is no corresponding JSON-column data, that is, where the
            JSON column is NULL. The only semantic difference between the two is
            that if you use a NESTED clause then the JSON column itself is not
            included in the result.
                  
The NESTED clause provides the same COLUMNS
            clause as json_table, including the possibility of nested columns.
            These are the advantages of using NESTED:
                  
- 
                        
                        You need not provide a table alias, even if you use the simple dot notation. 
- 
                        
                        You need not provide an is jsoncheck constraint, even if the JSON column is notJSONtype. (The constraint is needed forjson_tablewith the simple dot notation, unless the column isJSONtype.)
- 
                        
                        You need not specify LEFT OUTER JOIN.
 The NESTED clause syntax is simpler, it allows all of the
            flexibility of the COLUMNS clause, and it performs an implicit left
            outer join. This is illustrated in Example 20-2.
                  
Example 20-3 shows the use of a NESTED clause with the simple dot notation.
                  
Example 20-2 Equivalent: SQL NESTED and JSON_TABLE with LEFT OUTER JOIN
These two queries are
                equivalent. One uses SQL/JSON function json_table with an explicit
                    LEFT OUTER JOIN. The other uses a SQL NESTED
                clause.
                     
SELECT id, requestor, type, "number"
  FROM j_purchaseorder LEFT OUTER JOIN
       json_table(po_document
         COLUMNS (Requestor,
                  NESTED ShippingInstructions.Phone[*]
                    COLUMNS (type, "number")))
       ON 1=1);
SELECT id, requestor, type, "number"
  FROM j_purchaseorder NESTED
       po_document
         COLUMNS (Requestor,
                  NESTED ShippingInstructions.Phone[*]
                    COLUMNS (type, "number");The output is the same in both cases:
7C3A54B183056369E0536DE05A0A15E4 Alexis Bull Office 909-555-7307
7C3A54B183056369E0536DE05A0A15E4 Alexis Bull Mobile 415-555-1234
7C3A54B183066369E0536DE05A0A15E4 Sarah BellIf table j_purchaseorder had a row with
                    non-NULL values for columns id and
                    requestor, but a NULL value for column
                    po_document then that row would appear in both cases. But it
                would not appear in the json_table case if LEFT OUTER
                    JOIN were absent.
                     
Example 20-3 Using SQL NESTED To Expand a Nested Array
This example selects columns id and date_loaded
                from table j_purchaseorder, along with the array elements of field
                    Phone, which is nested in the value of field
                    ShippingInstructions of JSON column
                    po_document. It expands the Phone array value
                as columns type and number.
                     
(Column specification "number" requires the double-quote marks
                because number is a reserved term in SQL.)
                     
SELECT *
  FROM j_purchaseorder NESTED
       po_document.ShippingInstructions.Phone[*]
         COLUMNS (type, "number")Parent topic: SQL/JSON Function JSON_TABLE
20.2 COLUMNS Clause of SQL/JSON Function JSON_TABLE
The mandatory COLUMNS clause for SQL/JSON function
            json_table defines the columns of the virtual table that the function
        creates.
                  
 It consists of the keyword COLUMNS followed by the
            following entries, enclosed in parentheses. Other than the optional FOR
                ORDINALITY entry, each entry in the COLUMNS clause is
            either a regular column specification or a nested columns specification. 
                  
- 
                        At most one entry in the COLUMNSclause can be a column name followed by the keywordsFOR ORDINALITY, which specifies a column of generated row numbers (SQL data typeNUMBER). These numbers start with one. For example:COLUMNS (linenum FOR ORDINALITY, ProductID)An array step in a row path expression can lead to any number of rows that match the path expression. In particular, the order of array-step indexes and ranges, multiple occurrences of an array index, and duplication of a specified position due to range overlaps produce one row for each position match. The ordinality row numbers reflect this. 
- 
                        A regular column specification consists of a column name followed by an optional data type for the column, which can be any SQL data type that can be used in the RETURNINGclause ofjson_value, followed by an optional value clause and an optionalPATHclause. The default data type isVARCHAR2(4000).The column data type can thus be any of these: JSON,VARCHAR2,NUMBER,DATE,TIMESTAMP,TIMESTAMP WITH TIME ZONE, orSDO_GEOMETRY.Data type SDO_GEOMETRYis used for Oracle Spatial and Graph data. In particular, this means that you can usejson_tablewith GeoJSON data, which is a format for encoding geographic data in JSON.Oracle extends the SQL/JSON standard in the case when the returning data type for a column is VARCHAR2(N), by allowing optional keywordTRUNCATEimmediately after the data type. WhenTRUNCATEis present and the value to return is wider thanN, the value is truncated — only the firstNcharacters are returned. IfTRUNCATEis absent then this case is treated as an error, handled as usual by an error clause or the default error-handling behavior.
- 
                        A nested columns specification consists of the keyword NESTEDfollowed by an optionalPATHkeyword, a SQL/JSON row path expression, and then aCOLUMNSclause. ThisCOLUMNSclause specifies columns that represent nested data. The row path expression used here provides a refined context for the specified nested columns: each nested column path expression is relative to the row path expression. You can nest columns clauses to project values that are present in arrays at different levels to columns of the same row.A COLUMNSclause at any level (nested or not) has the same characteristics. In other words, theCOLUMNSclause is defined recursively. For each level of nesting (that is, for each use of keywordNESTED), the nestedCOLUMNSclause is said to be the child of theCOLUMNSclause within which it is nested, which is its parent. Two or moreCOLUMNSclauses that have the same parent clause are siblings.The virtual tables defined by parent and child COLUMNSclauses are joined using an outer join, with the parent being the outer table. The virtual columns defined by siblingCOLUMNSclauses are joined using aunionjoin.Example 20-1 and Example 20-9 illustrate the use of a nested columns clause. 
The only thing required in a regular column specification is the column name. Defining the column projection in more detail, by specifying a scalar data type, value handling, or a target path, is optional.
- 
                        The optional value clause specifies how to handle the data projected to the column: whether to handle it as would json_value,json_exists, orjson_query. This value handling includes the return data type, return format (pretty or ASCII), wrapper, and error treatment.If you use keyword EXISTSthen the projected data is handled as if byjson_exists(regardless of the column data type).Otherwise: - 
                              For a column of data type JSON, the projected data is handled as if byjson_query.
- 
                              For a non- JSONtype column (any type that can be used in ajson_valueRETURNINGclause), the projected data is handled by default as if byjson_value. But if you use keywordsFORMAT JSONthen it is handled as if byjson_query. You typically useFORMAT JSONonly when the projected data is a JSON object or array. (An error is raised if you useFORMAT JSONwith aJSONtype column.)
 For example, here the value of column FirstNameis projected directly usingjson_valuesemantics, and the value of columnAddressis projected as a JSON string usingjson_querysemantics:COLUMNS (FirstName, Address FORMAT JSON)json_querysemantics imply that the projected JSON data is well-formed. If the column is a non-JSONtype then this includes ensuring that non-ASCII characters in string values are escaped as needed. For example, a TAB character (CHARACTER TABULATION, U+0009) is escaped as\t. (ForJSONtype data, any such escaping is done when theJSONdata is created, not whenjson_queryis used.)When the column has json_querysemantics:- 
                              If database initialization parameter compatibleis at least20then you can use keywordsDISALLOW SCALARSto affect thejson_querybehavior by excluding scalar JSON values.
- 
                              You can override the default wrapping behavior by adding an explicit wrapper clause. 
 You can override the default error handling for a given handler ( json_exists,json_value, orjson_query) by adding an explicit error clause appropriate for it.
- 
                              
- 
                        The optional PATHclause specifies the portion of the row that is to be used as the column content. The column path expression following keywordPATHis matched against the context item provided by the virtual row. The column path expression must represent a relative path; it is relative to the path specified by the row path expression.If the PATHclause is not present then the behavior is the same as if it were present with a path of'$.<column-name>', where<column-name>is the column name. That is, the name of the object field that is targeted is taken implicitly as the column name.For purposes of specifying the targeted field only, the SQL identifier used for <column-name>is interpreted case-sensitively, even if it is not quoted. The SQL name of the column itself follows the usual rule: if it is enclosed in double quotation marks (") then the letter case used is significant; otherwise, it is not (it is treated as if uppercase).For example, these two COLUMNSclauses are equivalent. For SQL, case is significant only for columnComments(because it is quoted). The other two columns have case-insensitive names (that is, their names are treated case-insensitively), regardless of whether aPATHclause is used. In the firstCOLUMNSclause the first two columns are written with mixed case that matches the field names they target implicitly.COLUMNS(ProductId, Quantity NUMBER, "Comments") COLUMNS(productid VARCHAR2(4000) PATH '$.ProductId', quantity NUMBER PATH '$.Quantity', "Comments" VARCHAR2(4000) PATH '$.Comments')Example 20-1 presents equivalent queries that illustrate this. You can also use the dot notation in a PATHclause, as an alternative to a SQL/JSON path expression. Example 20-2 and Example 20-9 illustrate this.
In a column path-expression array step, the order of indexes and ranges, multiple occurrences of an array index, and duplication of a specified position due to range overlaps have the effect they would have for the particular semantics use for the column: json_exists, json_query, or json_value:
                  
- 
                        json_exists— All that counts is the set of specified positions, not how they are specified, including the order or number of times they are specified. All that is checked is the existence of a match for at least one specified position.
- 
                        json_query— Each occurrence of a specified position is matched against the data, in order.
- 
                        json_value— If only one position is specified then it is matched against the data. Otherwise, there is no match — by default (NULL ON ERROR) a SQLNULLvalue is returned.
Related Topics
See Also:
- 
                           Oracle Spatial Developer's Guide for information about using Oracle Spatial and Graph data 
Parent topic: SQL/JSON Function JSON_TABLE
20.3 JSON_TABLE Generalizes SQL/JSON Query Functions and Conditions
SQL/JSON function json_table generalizes SQL/JSON
    condition json_exists and SQL/JSON functions json_value and
      json_query. Everything that you can do using these functions you can do using
      json_table. For the jobs they accomplish, the syntax of these functions is
    simpler to use than is the syntax of json_table.
                  
If you use any of json_exists, json_value, or
        json_query more than once, or in combination, to access the same data then
      a single invocation of json_table presents the advantage that the data is
      parsed only once.
                  
Because of this, the optimizer often automatically rewrites multiple invocations
      of json_exists, json_value and json_query
      (any combination) to fewer invocations of json_table instead, so the data is
      parsed only once.
                  
Example 20-4 and Example 20-5 illustrate this. They each select the requestor and the set
      of phones used by each object in column j_purchaseorder.po_document. But
        Example 20-5 parses that column only once, not four times.
                  
Note the following in connection with Example 20-5:
- 
                        A JSON value of nullis a value as far as SQL is concerned; it is notNULL, which in SQL represents the absence of a value (missing, unknown, or inapplicable data). In Example 20-5, if the JSON value of object attributezipCodeisnullthen the SQL string'true'is returned.
- 
                        json_existsis a SQL condition; you can use it in a SQLWHEREclause, aCASEstatement, or a check constraint. In Example 20-4 it is used in aWHEREclause. Functionjson_tableemploys the semantics ofjson_existsimplicitly when you specify keywordEXISTS. It must return a SQL value in the virtual column. Since Oracle SQL has no Boolean data type, a SQL string'true'or'false'is used to represent the Boolean value. This is the case in Example 20-5: theVARCHAR2value is stored in columnjt.has_zip, and it is then tested explicitly for equality against the literal SQL string'true'.
- 
                        JSON field AllowPartialShipmenthas a JSON Boolean value. Whenjson_valueis applied to that value it is returned as a string. In Example 20-5, data typeVARCHAR2is used as the column data type. Functionjson_tableimplicitly usesjson_valuefor this column, returning the value as aVARCHAR2value, which is then tested for equality against the literal SQL string'true'.
Example 20-4 Accessing JSON Data Multiple Times to Extract Data
SELECT json_value(po_document, '$.Requestor' RETURNING VARCHAR2(32)),
       json_query(po_document, '$.ShippingInstructions.Phone'
                  RETURNING VARCHAR2(100))
  FROM j_purchaseorder
  WHERE json_exists(po_document, '$.ShippingInstructions.Address.zipCode')
    AND json_value(po_document,  '$.AllowPartialShipment'
                   RETURNING VARCHAR2(5 CHAR))
        = 'true';Example 20-5 Using JSON_TABLE to Extract Data Without Multiple Parses
(If the JSON data is of JSON data type then do not use keywords FORMAT JSON; otherwise, an error is raised.) 
                     
SELECT jt.requestor, jt.phones
  FROM j_purchaseorder,
       json_table(po_document, '$'
         COLUMNS (
           requestor VARCHAR2(32 CHAR) PATH '$.Requestor',
           phones    VARCHAR2(100 CHAR) FORMAT JSON
                     PATH '$.ShippingInstructions.Phone',
           partial   VARCHAR2(5 CHAR) PATH '$.AllowPartialShipment',
           has_zip   VARCHAR2(5 CHAR) EXISTS
                     PATH '$.ShippingInstructions.Address.zipCode')) jt
  WHERE jt.partial = 'true' AND jt.has_zip = 'true';
Related Topics
Parent topic: SQL/JSON Function JSON_TABLE
20.4 Using JSON_TABLE with JSON Arrays
A JSON value can be an array or can include one or more arrays, nested to
    any number of levels inside other JSON arrays or objects. You can use a
      json_table
                     NESTED path clause to project specific elements of an array.
                  
Example 20-6 projects the requestor and associated phone numbers from
        the JSON data in column po_document. The entire JSON array
          Phone is projected as a column of JSON data, ph_arr. To
        format this JSON data as a VARCHAR2 column, the keywords FORMAT
          JSON are needed if the JSON data is not of JSON data type (and
        those keywords raise an error if the type is JSON data).
                     
What if you wanted to project the individual elements of JSON array
          Phone and not the array as a whole? Example 20-7 shows one way to do this, which you can use if the array
        elements are the only data you need to project.
                     
If you want to project
        both the requestor and the corresponding phone data then the row path expression of Example 20-7 ($.Phone[*]) is not appropriate: it
        targets only the (phone object) elements of array Phone. 
                     
Example 20-8 shows one way to target both: use a row path
          expression that targets both the name and the entire phones array, and use column
          path expressions that target fields type and number
        of individual phone objects. 
                     
In Example 20-8 as in Example 20-6, keywords FORMAT JSON are needed if the
        JSON data is not of JSON data type, because the resulting
          VARCHAR2 columns contain JSON data, namely arrays of phone types or phone
        numbers, with one array element for each phone. In addition, unlike the case for Example 20-6, a wrapper clause is needed for column
          phone_type and column phone_num, because array
          Phone contains multiple objects with fields type and
          number.
                     
Sometimes you might not want the effect of Example 20-8. For example, you might want a column that contains a single phone number (one row per number), rather than one that contains a JSON array of phone numbers (one row for all numbers for a given purchase order).
To
        obtain that result, you need to tell json_table to project the array
        elements, by using a json_table
                        NESTED path clause for the array. A NESTED path
        clause acts, in effect, as an additional row source (row pattern). Example 20-9 illustrates this.
                     
You can use any number
        of NESTED keywords in a given json_table invocation.
                     
In Example 20-9 the outer COLUMNS clause is the parent of
        the nested (inner) COLUMNS clause. The virtual tables defined are joined
        using an outer join, with the table defined by the parent clause being the outer table in
        the join.
                     
(If there were a second columns clause nested directly under
        the same parent, the two nested clauses would be sibling COLUMNS
        clauses.)
                     
Example 20-6 Projecting an Entire JSON Array as JSON Data
SELECT jt.*
  FROM j_purchaseorder,
       json_table(po_document, '$'
         COLUMNS (requestor VARCHAR2(32 CHAR) PATH '$.Requestor',
                  ph_arr    VARCHAR2(100 CHAR) FORMAT JSON
                            PATH '$.ShippingInstructions.Phone')
                 ) AS "JT";
Example 20-7 Projecting Elements of a JSON Array
SELECT jt.*
  FROM j_purchaseorder,
       json_table(po_document, '$.ShippingInstructions.Phone[*]'
         COLUMNS (phone_type VARCHAR2(10) PATH '$.type',
                  phone_num  VARCHAR2(20) PATH '$.number')) AS "JT";
PHONE_TYPE     PHONE_NUM
----------     ---------
Office         909-555-7307
Mobile         415-555-1234
Example 20-8 Projecting Elements of a JSON Array Plus Other Data
SELECT jt.*
  FROM j_purchaseorder,
       json_table(po_document, '$'
         COLUMNS (
           requestor  VARCHAR2(32 CHAR) PATH '$.Requestor',
           phone_type VARCHAR2(50 CHAR) FORMAT JSON WITH WRAPPER
                      PATH '$.ShippingInstructions.Phone[*].type',
           phone_num  VARCHAR2(50 CHAR) FORMAT JSON WITH WRAPPER
                      PATH '$.ShippingInstructions.Phone[*].number')) AS "JT";
REQUESTOR    PHONE_TYPE            PHONE_NUM
---------    ----------            ---------
Alexis Bull  ["Office", "Mobile"]  ["909-555-7307", "415-555-1234"]
Example 20-9 JSON_TABLE: Projecting Array Elements Using NESTED
This example shows two equivalent queries that project array elements. The first query uses the simple, dot-notation syntax for the expressions that target the row and column data. The second uses the full syntax.
Except for column number, whose SQL
        identifier is quoted ("number"), the SQL column names are, in effect,
        uppercase. (Column number is lowercase.)
                     
In the first query the column names are written exactly the same as the field names that are targeted, including with respect to letter case. Regardless of whether they are quoted, they are interpreted case-sensitively for purposes of establishing the proper path.
The second query has:
- 
                           Separate arguments of a JSON column-expression and a SQL/JSON row path-expression 
- 
                           Explicit column data types of VARCHAR2(4000)
- 
                           Explicit PATHclauses with SQL/JSON column path expressions, to target the object fields that are projected
SELECT jt.*
  FROM j_purchaseorder po,
       json_table(po.po_document
         COLUMNS (Requestor,
                  NESTED ShippingInstructions.Phone[*]
                    COLUMNS (type, "number"))) AS "JT";SELECT jt.*
  FROM j_purchaseorder po,
       json_table(po.po_document, '$'
         COLUMNS (Requestor VARCHAR2(4000) PATH '$.Requestor',
                  NESTED
                    PATH '$.ShippingInstructions.Phone[*]'
                    COLUMNS (type     VARCHAR2(4000) PATH '$.type',
                             "number" VARCHAR2(4000) PATH '$.number'))
       ) AS "JT";
Parent topic: SQL/JSON Function JSON_TABLE
20.5 Creating a View Over JSON Data Using JSON_TABLE
To improve query performance you can create a view over JSON data that you
    project to columns using SQL/JSON function json_table. To further improve query
    performance you can create a materialized view and place the JSON data in
    memory.
                  
Example 20-10 defines a view over JSON data. It uses a
          NESTED path clause to project the elements of array
          LineItems.
                     
Example 20-11 defines a materialized view that has the same data and structure as Example 20-10.
In general, you cannot update a view directly (whether materialized or not).
        But if a materialized view is created using keywords REFRESH and ON
          STATEMENT, as in Example 20-11, then the view is updated automatically whenever you update
        the base table.
                     
You can use json_table to project any fields as view columns, and the view
        creation (materialized or not) can involve joining any tables and any number of invocations
          of json_table.
                     
The only differences between Example 20-10 and Example 20-11 are these:
- 
                           The use of keyword MATERIALIZED.
- 
                           The use of BUILD IMMEDIATE.
- 
                           The use of REFRESH FAST ON STATEMENT WITH PRIMARY KEY.
The use of REFRESH FAST means that the materialized view
        will be refreshed incrementally. For this to occur, you must use either WITH PRIMARY
          KEY or WITH ROWID (if there is no primary key). Oracle
        recommends that you specify a primary key for a table that has a JSON column and that you
        use WITH PRIMARY KEY when creating a materialized view based on it.
                     
You could use ON COMMIT in place of ON
          STATEMENT for the view creation. The former synchronizes the view with the base
        table only when your table-updating transaction is committed. Until then the table changes
        are not reflected in the view. If you use ON STATEMENT then the view is
        immediately synchronized after each DML statement. This also means that a view created using
          ON STATEMENT reflects any rollbacks that you might perform. (A subsequent
          COMMIT statement ends the transaction, preventing a
      rollback.)
                     
See Also:
Refreshing Materialized Views in Oracle Database Data Warehousing Guide
Example 20-10 Creating a View Over JSON Data
CREATE VIEW j_purchaseorder_detail_view
  AS SELECT jt.*
       FROM j_purchaseorder po,
            json_table(po.po_document, '$'
              COLUMNS (
                po_number        NUMBER(10)         PATH '$.PONumber',
                reference        VARCHAR2(30 CHAR)  PATH '$.Reference',
                requestor        VARCHAR2(128 CHAR) PATH '$.Requestor',
                userid           VARCHAR2(10 CHAR)  PATH '$.User',
                costcenter       VARCHAR2(16)       PATH '$.CostCenter',
                ship_to_name     VARCHAR2(20 CHAR)
                                 PATH '$.ShippingInstructions.name',
                ship_to_street   VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.street',
                ship_to_city     VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.city',
                ship_to_county   VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.county',
                ship_to_postcode VARCHAR2(10 CHAR)
                                 PATH '$.ShippingInstructions.Address.postcode',
                ship_to_state    VARCHAR2(2 CHAR)
                                 PATH '$.ShippingInstructions.Address.state',
                ship_to_zip      VARCHAR2(8 CHAR)
                                 PATH '$.ShippingInstructions.Address.zipCode',
                ship_to_country  VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.country',
                ship_to_phone    VARCHAR2(24 CHAR)
                                 PATH '$.ShippingInstructions.Phone[0].number',
                NESTED PATH '$.LineItems[*]'
                  COLUMNS (
                    itemno      NUMBER(38)         PATH '$.ItemNumber', 
                    description VARCHAR2(256 CHAR) PATH '$.Part.Description', 
                    upc_code    NUMBER             PATH '$.Part.UPCCode', 
                    quantity    NUMBER(12,4)       PATH '$.Quantity', 
                    unitprice   NUMBER(14,2)       PATH '$.Part.UnitPrice'))) jt;
Example 20-11 Creating a Materialized View Over JSON Data
CREATE MATERIALIZED VIEW j_purchaseorder_materialized_view
  BUILD IMMEDIATE
  REFRESH FAST ON STATEMENT WITH PRIMARY KEY
  AS SELECT jt.*
       FROM j_purchaseorder po,
            json_table(po.po_document, '$'
              COLUMNS (
                po_number        NUMBER(10)         PATH '$.PONumber',
                reference        VARCHAR2(30 CHAR)  PATH '$.Reference',
                requestor        VARCHAR2(128 CHAR) PATH '$.Requestor',
                userid           VARCHAR2(10 CHAR)  PATH '$.User',
                costcenter       VARCHAR2(16)       PATH '$.CostCenter',
                ship_to_name     VARCHAR2(20 CHAR)
                                 PATH '$.ShippingInstructions.name',
                ship_to_street   VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.street',
                ship_to_city     VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.city',
                ship_to_county   VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.county',
                ship_to_postcode VARCHAR2(10 CHAR)
                                 PATH '$.ShippingInstructions.Address.postcode',
                ship_to_state    VARCHAR2(2 CHAR)
                                 PATH '$.ShippingInstructions.Address.state',
                ship_to_zip      VARCHAR2(8 CHAR)
                                 PATH '$.ShippingInstructions.Address.zipCode',
                ship_to_country  VARCHAR2(32 CHAR)
                                 PATH '$.ShippingInstructions.Address.country',
                ship_to_phone    VARCHAR2(24 CHAR)
                                 PATH '$.ShippingInstructions.Phone[0].number',
                NESTED PATH '$.LineItems[*]'
                  COLUMNS (
                    itemno      NUMBER(38)         PATH '$.ItemNumber', 
                    description VARCHAR2(256 CHAR) PATH '$.Part.Description', 
                    upc_code    NUMBER             PATH '$.Part.UPCCode', 
                    quantity    NUMBER(12,4)       PATH '$.Quantity', 
                    unitprice   NUMBER(14,2)       PATH '$.Part.UnitPrice'))) jt;