微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

如何在 Nim 中检查未命名的元组?

如何解决如何在 Nim 中检查未命名的元组?

如果元组就地定义,则下面的宏有效,但如果它是通过类型名称定义的,则无效。

playground

import macros

macro is_unnamed_tuple[T](TT: type[T]): bool =
  # nnkTupleTy - named
  let r = new_lit(TT.getTypeInst[1].kind == nnkTupleConstr)
  quote do:
    `r`

type NamedT = tuple[a: string]
type UnnamedT = (string,)

echo is_unnamed_tuple tuple[a: string] # => false
echo is_unnamed_tuple NamedT           # => false
echo is_unnamed_tuple (string,)        # => true
echo is_unnamed_tuple UnnamedT         # => false <- Error

附言

也许有更简单的方法来检查它,比如 T is named tuple 或类似的东西?

另外,如果对象是variant,是否可以进行类似的检查?

解决方法

Here 都是宏。

import std/macros

macro is_unnamed_tuple(T: typedesc): bool =
  # nnkTupleTy - named
  if T.kind != nnkTupleConstr:
    if T.kind == nnkSym:
      result = newLit( T.getImpl[^1].kind == nnkTupleConstr)
    else:
      result = newLit(false)
  else:
    result = newLit(true)

macro isVariant(T: typedesc): bool =
  result = newLit(false)
  let impl = T.getImpl
  if impl.kind != nnkNilLit:
    let objImpl =
      if impl[^1].kind == nnkRefTy:
        impl[^1][0]
      else:
        impl[^1]
    if objImpl.kind == nnkObjectTy:
      for x in objImpl[^1]:
        if x.kind == nnkRecCase:
          result = newLit(true)
          break
type 
  NamedT = tuple[a: string]
  UnnamedT = (string,)
  VarObj = object
    a: int
    case t: bool
    of true: discard
    of false: discard

assert is_unnamed_tuple(tuple[a: string]) == false
assert is_unnamed_tuple(NamedT) == false
assert is_unnamed_tuple((string,)) == true
assert is_unnamed_tuple(UnnamedT) == true
assert string.isVariant == false
assert VarObj.isVariant == true
assert int.isVariant == false

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。