我试图实现一个接受通用有效负载的类,然后将其转换为JSON:
require "json"
class Response
alias Payload = Hash(Symbol | String, String | Bool | Int32 | Int64 | Float32 | Float64 | Nil)
@payload : Payload
def initialize(@payload, @method : String = "sendMessage")
end
def to_json
@payload.merge({"method" => @method}).to_json
end
end
Response.new({
"chat_id" => update.message.not_nil!.chat.id,
"text" => "Crystal is awesome but complicated",
})
但我得到instance variable '@payload' of Response must be Hash(String | Symbol, Bool | Float32 | Float64 | Int32 | Int64 | String | Nil), not Hash(String, Int64 | String)
编译器错误。我怎么能克服这个? Crystal支持通用哈希吗?即使类型重叠,为什么它也无法转换?
Response
是分片的一部分,因此我不知道哪些哈希值会作为参数传递,但Payload
必须足够。
答案 0 :(得分:3)
您的有效负载散列类型为Hash(String, Int32 | String)
:
typeof({
"chat_id" => update.message.not_nil!.chat.id,
"text" => "Crystal is awesome but complicated",
}) # => Hash(String, Int32 | String)
但构造函数需要Hash(Symbol | String, String | Bool | Int32 | Int64 | Float32 | Float64 | Nil)
。
这些是不同的类型,不能神奇地铸造。您需要确保您的有效负载具有正确的类型。
执行此操作的一种方法是显式声明散列文字的类型:
Payload{
"chat_id" => update.message.not_nil!.chat.id,
"text" => "Crystal is awesome but complicated",
}
这当然不是很好,但取决于你的用例,它可能就足够了。
如果您想要一个允许接收任何类型哈希的通用接口,您需要将其转换为Payload
类型。这意味着将数据复制到该类型的新哈希中:
def self.new(hash, method : String = "sendMessage")
payload = Payload.new
hash.each do |key, value|
payload[key] = value
end
new(payload, method)
end
对于一个真实的例子,我在Crinja中使用这种方法将许多不同的类型变体转换为匹配的变体。