文章目录
第一章 Caché JSON 简介 - 简介
- Caché JSON功能
-
- 在运行时创建和操作动态实体
- 使用字符串JSON构造函数创建动态实体
- 使用Caché动态表达式定义值
- 在动态实体和规范JSON字符串之间进行转换
- 链动态实体法
- 迭代和数据类型发现
- 动态实体方法概述
-
- 创建、读取、更新、删除
- 迭代和数组
- 堆栈的功能
- JSON序列化和反序列化
- 数据类型信息
Caché 和 Caché ObjectScript现在提供了对JSON的集成支持,一组快速、简单、强大的新特性允许像处理对象或表一样轻松地处理JSON数据结构:
- 两个新类%Library.DynamicObject 和 %Library.DynamicArray 。提供一种简单、有效的方法来封装和处理标准JSON数据结构。这些类的实例称为动态实体。
- 使用新的JSON Caché ObjectScript语法,可以使用标准的Caché ObjectScript赋值语句而不是方法调用来在运行时创建和更改动态实体。对象属性和数组元素的值可以指定为JSON字符串文本,也可以指定为Caché ObjectScript动态表达式。
- 动态实体包含用于JSON序列化(在动态实体和规范JSON格式之间进行转换)、迭代、数据类型、创建/读取/更新/删除操作和其他有用功能的方法。
下面是一些新的JSON功能的例子:
在运行时创建和操作动态实体
可以创建动态实体,并定义一个任意的模式为他们在运行时:
/// d ##class(PHA.OP.MOB.Test).TestJson() ClassMethod TestJson() { set dynObject2 = ##class(%DynamicObject).%New() set dynObject2.SomeNumber = 42 set dynObject2.SomeString = "a string" set dynObject1 = ##class(%DynamicObject).%New() set dynObject1.SomeNumber = 42 set dynObject1.SomeString = "a string" w dynObject1.%ToJSON(),! set dynObject1.SomeArray = ##class(%DynamicArray).%New() set dynObject1.SomeArray."0" = "an array element" set dynObject1.SomeArray."1" = 123 set dynObject1.SomeArray."2" =dynObject2 w dynObject1.%ToJSON(),! }
DHC-APP>d ##class(PHA.OP.MOB.Test).TestJson() {"SomeNumber":42,"SomeString":"a string"} {"SomeNumber":42,"SomeString":"a string","SomeArray":["an array element",123,{"SomeNumber":42,"SomeString":"a string"}]} DHC-APP>
使用字符串JSON构造函数创建动态实体
还可以通过分配一个文字JSON字符串来创建一个动态实体。文字JSON构造函数{}和[]可以用来代替%New()构造函数。例如,可以使用set x=[]而不是set x=##class(%DynamicArray)来创建一个动态数组。与%New()不同,文字JSON构造函数也可以接受指定属性或元素的JSON文字字符串。这意味着可以用这些简单的赋值语句创建一个与前一个例子中的dynObject1相同的对象:
set dynObject2 = {"SomeNumber":42,"SomeString":"a string"} set dynObject2.SomeArray = ["an array element",123]
本例为每个构造函数使用了一个语句,但是数组构造函数可以简单地嵌套在对象构造函数中。
为了证明dynObject1和dynObject2是相同的,我们可以将它们显示为由%ToJSON()方法返回的序列化JSON字符串:
write "object 1: "_dynObject1.%ToJSON(),!,"object 2: "_dynObject2.%ToJSON() object 1: {"SomeNumber":42,"SomeString":"a string","SomeArray":["an array element",123]} object 2: {"SomeNumber":42,"SomeString":"a string","SomeArray":["an array element",123]}
使用Caché动态表达式定义值
包含在文字构造函数{}和[]中的文本必须使用有效的JSON语法,但有一个例外。对于元素或属性的值,可以使用圆括号中的Caché表达式,而不是JSON文本。这个ObjectScript动态表达式(相当于set语句的右侧)将在运行时计算并转换为有效的JSON值。本例中的动态表达式包含对Caché $ZDATE函数的调用:
set dynObj = { "Date":($ZD($H,3)) }
当我们检索dynObject.Date时,动态表达式的计算值就会显示出来。
write "Value of dynamic expression is: "_dynObject.Date Value of dynamic expression is: 2016-07-27
/// d ##class(PHA.OP.MOB.Test).TestJson() ClassMethod TestJson() { set dynObject2 = ##class(%DynamicObject).%New() set dynObject2.SomeNumber = 42 set dynObject2.SomeString = "a string" set dynObj = { "Date":($ZD($H,3)) } set dynObject1 = ##class(%DynamicObject).%New() set dynObject1.SomeNumber = 42 set dynObject1.SomeString = "a string" w dynObject1.%ToJSON(),! set dynObject1.SomeArray = ##class(%DynamicArray).%New() set dynObject1.SomeArray."0" = "an array element" set dynObject1.SomeArray."1" = 123 set dynObject1.SomeArray."2" =dynObject2 set dynObject1.SomeArray."2" =dynObj w dynObject1.%ToJSON(),! }
DHC-APP>d ##class(PHA.OP.MOB.Test).TestJson() {"SomeNumber":42,"SomeString":"a string"} {"SomeNumber":42,"SomeString":"a string","SomeArray":["an array element",123,{"Date":"2020-02-09"}]}
在动态实体和规范JSON字符串之间进行转换
动态实体有序列化方法,允许将对象转换为JSON字符串。在下面的示例中,使用文字构造函数创建动态对象,并调用对象的%ToJSON()方法将其序列化为myJSONstring:
set myJSONstring = {"aNumber":(21*2),"aDate":($ZD($H,3)),"anArray":["string",123]}.%ToJSON()
这个序列化的JSON对象可以像任何其他字符串一样存储和检索。类方法%FromJSON()可以从任何源获取有效的JSON字符串并将其转换为动态对象。下面的代码将myJSONstring反序列化为动态对象myObject,并使用%ToJSON()来显示它:
set myObject = ##class(%DynamicAbstractObject).%FromJSON(myJSONstring) write myObject.%ToJSON() {"aNumber":42,"aDate":"2016-08-29","anArray":["string",123]}
链动态实体法
一些动态实体方法可以被链接。本例创建了一个包含两个元素的动态数组,然后将%Push()方法链接起来,以向数组的末尾添加另外三个元素。最后一个链调用%ToJSON()显示序列化的字符串:
set dynArray = ["a","b"] write dynArray.%Push(12).%Push({"a":1,"b":2}).%Push("final").%ToJSON() ["a","b",12,{"a":1,"b":2},"final"]
迭代和数据类型发现
还提供了动态实体方法,用于迭代和数据类型发现等目的。这个例子创建了两个JSON字符串,将其中一个反序列化为dynEntity(任何一个都可以),然后为dynEntity获取一个迭代器:
set arrayStr = [12,"some string",[1,2]].%ToJSON() set objectStr = {"a":12,"b":"some string","c":{"x":1,"y":2},"d":true,"e":1.1}.%ToJSON() set dynEntity = {}.%FromJSON(objectStr) set itr = dynEntity.%GetIterator() while itr.%GetNext(.key,.val) {write !,key_": "_"/"_val_"/, type: "_dynEntity.%GetTypeOf(key)}
对于while循环的每次迭代,%GetNext()将在key中返回属性名或数组索引,在val中返回成员值。
while itr.%GetNext(.key,.val) {write !,key_": "_"/"_val_"/, type: "_dynEntity.%GetTypeOf(key)} a: /12/, type: number b: /some string/, type: string c: /5@%Library.DynamicObject/, type: object d: /1/, type: boolean e: /1.1/, type: number动态实体方法概述
动态实体方法可分为以下几类:
创建、读取、更新、删除
- %Set() 可以更改现有动态实体成员(属性或元素)的值,也可以创建新成员并为其赋值。
- %Remove() 删除现有成员
- %Get() 检索成员的值
迭代和数组
- %GetIterator() 返回一个迭代器,其中包含指向动态实体每个成员的指针。
- %GetNext() 返回迭代器标识的成员的键和值,并将光标移到下一个成员。
- %Size() 返回成员数(包括数组中未分配的元素)。
- %IsDefined() 测试成员是否具有指定的值。
堆栈的功能
- %Push() 将新元素添加到动态数组的末尾。
- %Pop() 删除数组的最后一个元素并返回其值。这些方法不适用于动态对象,因为对象属性不是按可预测的顺序存储的。
JSON序列化和反序列化
- %FromJSON() 将JSON字符串转换为动态实体。
- %ToJSON() 将动态实体序列化为规范JSON字符串。
数据类型信息
- %GetTypeOf() 返回一个字符串,该字符串指示指定成员值的数据类型。%Set()和%Push()提供一个可选的第三个参数来显式地指定值的数据类型。