通用哈希作为参数?

时间:2017-07-31 20:34:54

标签: crystal-lang

我试图实现一个接受通用有效负载的类,然后将其转换为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必须足够。

1 个答案:

答案 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中使用这种方法将许多不同的类型变体转换为匹配的变体。