是否可以在 Python 中模拟 Scala 的特征?
我想使用可以插入类中的方法创建轻量级接口。下面是 Scala 中的一个简短示例:
class DB {
def find(id: String) = ...
}
trait Transformation extends DB {
def transform(obj: String): String
override def find(id: String) =
transform(super.find(id))
}
trait Cache extends DB {
val cache = Cache()
override def find(id: String) = {
...
if (cache.contains(id))
cache.find(id)
else {
cache.set(id, super.find(id))
cache.get(id)
}
}
}
通过这些类(特征),我们可以使用 Transformation、Cache 或两者来实例化 DB 类。注意Transformation有一个抽象方法transform,它仍然需要在具体的类中实现。
new DB() with Transformation {
def transform(obj: String): obj.toLower()
}
new DB() with Cache
new DB() with Transformation with Cache {
def transform(obj: String): obj.toLower()
}
有什么方法可以在Python中实现这样的事情吗?我知道 Python 有一个 Traits 包,但它的目的似乎不同。
I want to create lightweight interfaces with methods that I can plug into classes. Here is an short example in Scala:
class DB {
def find(id: String) = ...
}
trait Transformation extends DB {
def transform(obj: String): String
override def find(id: String) =
transform(super.find(id))
}
trait Cache extends DB {
val cache = Cache()
override def find(id: String) = {
...
if (cache.contains(id))
cache.find(id)
else {
cache.set(id, super.find(id))
cache.get(id)
}
}
}
With these classes (traits) we can instantiate DB classes with Transformation, with Cache, or both. Note that Transformation has an abstract method transform, which still needs to implemented in concrete classes.
new DB() with Transformation {
def transform(obj: String): obj.toLower()
}
new DB() with Cache
new DB() with Transformation with Cache {
def transform(obj: String): obj.toLower()
}
Is there any way to achieve something like this in Python? I know there exists a Traits package for Python, but its purpose seems to be different.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(4)
这是使用 Class 的答案的替代方案。如果你想在运行时创建 Traits,让我们滥用 type()。
以 http://twitter.github.io/scala_school/basics.html#trait 为例
您也可以将构造函数传递给
BMW
This is an alternative to the answer using Class. If you want to make Traits during runtime, let's abuse type().
Take an example from http://twitter.github.io/scala_school/basics.html#trait
You can pass constructor into
BMW
as well by查看 Enthought 的 Python Traits 项目。但你所描述的似乎与常规的 Python 属性相同。
Take a look at the Python Traits project at Enthought. But what you are describing seems to be the same as regular Python properties.
最简单的解决方案可能是创建另一个子类。
如果你顽固地拒绝实际给类命名,你可以直接调用
type
。替换为
Which 并不比 Scala 示例差太多。
由于 python 类型系统的微妙之处,不从主类 (
DB
) 继承的 mixins 首先出现在基础列表中。不这样做将阻止 mixin 类正确覆盖主基类的方法。同样的微妙之处可以让您将额外的功能作为适当的派生类。钻石继承模式不是问题;基类只出现一次,无论有多少个中间基类继承自它们(毕竟,它们最终都继承自
object
)。The simplest solution is probably to just make another subclass.
If you stubbornly refuse to actually give the class a name, you can call
type
directly. replacewith
Which isn't too much worse than the Scala example.
Due to a subtlety of the python type system, the mixins, which do not inherit from the primary class (
DB
) appear first in the bases list. Not doing so will prevent the mixin classes from properly overriding the methods of the primary base class.That same subtlety can allow you to have the extra features be proper derived classes. The diamond inheritance pattern is a non-issue; base classes only appear once, no matter how many intermediate base classes inherit from them (after all, they all ultimately inherit from
object
).最接近 Scala 特征的解决方案是抽象基类。
它们在 abc 模块中可用:
然后您必须使用抽象方法的正确实现对 Transformation 类进行子类化。
顺便说一句,您还可以通过在您想要抽象的方法上引发
NotImplementedError
来模拟 abc。 ABCMeta 只是不允许您创建抽象类的实例。PS:元类和 super 的 Python 3 语法将略有不同(而且更好!)。
The closest solution to Scala traits are Abstract Base Classes.
They are available in the abc module:
then you must subclass the Transformation class with proper implementation to abstract methods.
BTW, you can also simulate abc's by just raising
NotImplementedError
on methods you want as abstract. ABCMeta just don't let you create instances of abstract classes.PS: Python 3 syntax to both metaclasses and
super
will be a little bit different (and better!).