0

我们有一些异步属性(使用@property)和一些常用的。我们必须使用 hy 来检查一个是否是异步的,或者不使用 asyncio.iscoroutine 函数。问题是我们必须使用 getattr 而不是 。运算符,因为 dot 将调用属性内部函数。我们希望在不破坏其他项目中使用的 DSL 的情况下更改我们的 hy 代码,也就是说,我们不想对属性名称使用字符串表示法(双引号),因此我们需要编写一些像 . 但在内部调用 getattr。

=> (defmacro attr [obj attr] `(getattr ~obj '~attr))
=> (attr 5 real)
Traceback (most recent call last):
  File "stdin-5bbedde10a6366314d9be4bd343639582b4d0748", line 1, in <module>
    (attr 5 real)
AttributeError: 'int' object has no attribute 'real'
=> (. 5 real)
5
4

3 回答 3

2

听起来您对 Python 属性的语义感到困惑,这是可以理解的,因为它们相当多毛。在 Python 中(因此在 Hy 中也是等价的结构),foo.x并且getattr(foo, "x")总是等价的,即使x是一个属性。要将属性方法作为对象而不是调用它,您需要检索类 ( type(x)) 而不是实例的属性,并getattr使用字符串文字并.保持可互换。在这个例子中,getattr(type(foo), 'bar').fset也可以写成type(foo).bar.fset

于 2021-07-27T11:49:38.837 回答
1

我试过了

(defmacro attr [obj attr]
  `(getattr ~obj ~(str attr)))

(print (attr 5 real))

网络解释器中,它会打印5.

不过,我无法在网络解释器中重现您的问题。它必须仍在运行旧版本,因为我必须安装您的确切版本才能做到这一点。我怀疑以下针对 1.0a2 宣布的重大更改是造成差异的原因:

Hy 模型对象不再等于普通的 Python 值。例如,(!= 1 '1)hy.as-model 在进行此类检查之前,您可以将值提升到模型。

我的版本attr确实适用于 Hy 1.0a3。我检查了。试试看。

.__dict__Python 中的属性使用字符串键存储在对象中。由于 Hy 的符号不再等于字符串,因此使用符号查找字符串键将始终失败。将符号转换为字符串首先修复它。我最初的评论是正确的。(而且这一步在 Hissp 中是不必要的,因为它缺少 Hy 的单独模型类型。)

于 2021-07-27T07:04:32.890 回答
0

虽然后一个答案有效,但我发现问题在于获取属性背后的方法,正如@kodiologist 所写,getattr 和 dot 是等价的。所以我使用了 fget 和:

(iscoroutinefunction (.fget (. (type obj) ~attr)))
于 2021-07-28T10:10:29.633 回答