The documentation you are viewing is for Dapr v1.13 which is an older version of Dapr. For up-to-date documentation, see the latest version.
Dapr SDK 中的序列化
Dapr 的 SDK 为下面两种情况提供序列化: 首先是对于通过请求和响应的有效载荷传递的 API 对象, 其次,对于要持久化的对象。 对于这两种情况,SDK 都提供了默认的序列化实现。 在Java SDK中,它是DefaultObjectSerializer类,提供JSON序列化。
服务调用
DaprClient client = (new DaprClientBuilder()).build();
client.invokeService("myappid", "saySomething", "My Message", HttpExtension.POST).block();
在上面的示例中,应用程序将收到一个对 saySomething
方法的 POST
请求,请求的有效载荷为 "My Message"
- 这句说明中给它添加了引号是因为序列化工具会把输入的 String 字符串序列化为 JSON。
POST /saySomething HTTP/1.1
Host: localhost
Content-Type: text/plain
Content-Length: 12
"My Message"
State management
DaprClient client = (new DaprClientBuilder()).build();
client.saveState("MyStateStore", "MyKey", "My Message").block();
在这个例子中,My Message
将被保存下来。 在这句说明中,没有给数据加上引号是因为 Dapr 的 API 会在保存 JSON 请求对象之前在内部进行解析。
[
{
"key": "MyKey",
"value": "My Message"
}
]
PubSub
DaprClient client = (new DaprClientBuilder()).build();
client.publishEvent("TopicName", "My Message").block();
事件发布,内容被序列化为 byte[]
类型并发送到 Dapr sidecar。 订阅者将以CloudEvent的形式接收它。 CloudEvent 将data
定义为String。 Dapr SDK 还为 CloudEvent
对象提供了一个内置的反序列化工具。
@PostMapping(path = "/TopicName")
public void handleMessage(@RequestBody(required = false) byte[] body) {
// Dapr's event is compliant to CloudEvent.
CloudEvent event = CloudEvent.deserialize(body);
}
绑定
在这种情况下,对象也会被序列化为 byte[]
类型,而输入绑定会按原样接收原始的 byte[]
,并将其反序列化为预期的对象类型。
- 输出绑定:
DaprClient client = (new DaprClientBuilder()).build();
client.invokeBinding("sample", "My Message").block();
- 输入绑定:
@PostMapping(path = "/sample")
public void handleInputBinding(@RequestBody(required = false) byte[] body) {
String message = (new DefaultObjectSerializer()).deserialize(body, String.class);
System.out.println(message);
}
应当会出现如下输出:
My Message
调用 Actor 方法
Actor 方法调用的对象序列化和反序列化与服务方法调用相同,唯一不同的是应用程序不需要反序列化请求或序列化响应,因为这一切都由 SDK 透明地完成。
对于 Actor 的方法,SDK 只支持无参或一个参数的方法。
- 调用 Actor 的方法:
public static void main() {
ActorProxyBuilder builder = new ActorProxyBuilder("DemoActor");
String result = actor.invokeActorMethod("say", "My Message", String.class).block();
}
- 实现 Actor 的方法:
public String say(String something) {
System.out.println(something);
return "OK";
}
应当会出现如下输出:
My Message
Actor的状态管理
Actor 也可以有状态。 在这种情况下,状态管理器将使用状态序列化工具对对象进行序列化和反序列化,并对应用程序进行透明处理。
public String actorMethod(String message) {
// Reads a state from key and deserializes it to String.
String previousMessage = super.getActorStateManager().get("lastmessage", String.class).block();
// Sets the new state for the key after serializing it.
super.getActorStateManager().set("lastmessage", message).block();
return previousMessage;
}
默认序列化工具
Dapr 的默认序列化工具是 JSON 序列化工具,其期望如下:
- 使用基本的JSON数据类型来实现跨语言和跨平台的兼容性:字符串、数字、数组、布尔值、空值和另一个 JSON 对象。 应用程序的可序列化对象中的每一个复杂的属性类型(例如DateTime),都应该被表示为JSON的基本类型之一。
- 用默认序列化工具持久化的数据也应该被保存为JSON对象,不需要额外的引号或编码。 下面的例子显示了一个字符串和一个JSON对象在Redis存储中的样子。
redis-cli MGET "ActorStateIT_StatefulActorService||StatefulActorTest||1581130928192||message
"This is a message to be saved and retrieved."
redis-cli MGET "ActorStateIT_StatefulActorService||StatefulActorTest||1581130928192||mydata
{"value":"My data value."}
- 自定义序列化工具必须将对象序列化为
byte[]
。 - 自定义序列化工具必须将
byte[]
反序列化为对象。 - 当用户提供一个自定义的序列化工具时,它应该以
byte[]
的形式被传输或持久化。 持久化时,也应当编码为Base64字符串, 大多数JSON库都能够完成这个功能。
redis-cli MGET "ActorStateIT_StatefulActorService||StatefulActorTest||1581130928192||message
"VGhpcyBpcyBhIG1lc3NhZ2UgdG8gYmUgc2F2ZWQgYW5kIHJldHJpZXZlZC4="
redis-cli MGET "ActorStateIT_StatefulActorService||StatefulActorTest||1581130928192||mydata
"eyJ2YWx1ZSI6Ik15IGRhdGEgdmFsdWUuIn0="
*目前而言,Java SDK是唯一实现该规范的Dapr SDK。 在不久的将来,其他SDK也会实现同样的功能。
Feedback
Was this page helpful?
Glad to hear it! Please tell us how we can improve.
Sorry to hear that. Please tell us how we can improve.