python 变量 fields_理解Python数据类:Dataclass fields 的概述(下)

原标题 Understanding Python Dataclasses—Part 2 ,作者为 Shikhar Chauhan 。

8e524c4714f62f0a6aca6093544816ea.png

这是 Python 最新的 Dataclasses 系列的第二部分内容。在第一部分里,我介绍了 dataclasses 的一般用法。这篇主要介绍另一个特征:dataclasses.field。

5cf904513af2ca1e8a1fa9b2b65921a7.png

我们已经知道 Dataclasses 会生成他们自身的__init__方法。它同时把初始化的值赋给这些字段。以下是我们在上一篇博客里定义的内容:

变量名

数据类型

这些内容仅给我们有限的 dataclass 字段使用范围。让我们讨论一下这些局限性,以及它们如何通过 dataclass.field 被解决。

复合初始化

考虑以下情形:你想要初始化一个变量为列表。你如何实现它呢?一种简单的方式是使用__post_init__方法。

ed7f0d3b0e81fd4229de7b1c06c647be.png

数据类 Student 产生了一个名为 marks 的列表。我们不传递 marks 的值,而是使用__post_init__方法初始化。这是我们定义的单一属性。此外,我们必须在__post_init__里调用 get_random_marks 函数。这些工作是额外的。

辛运的是,Python 为我们提供了一个解决方案。我们可以使用 dataclasses.field 来定制化 dataclass 字段的行为以及它们在 dataclass 的影响。

仍然是上述的使用情形,让我们从__post_init__里去除 get_random_marks 的调用。以下是使用 dataclasses.field 的情形:

041e526b2705c64490385693f4ce872c.png

dataclasses.field 接受了一个名为 default_factory 的参数,它的作用是:如果在创建对象时没有赋值,则使用该方法初始化该字段。

default_factory 必须是一个可以调用的无参数方法(通常为一个函数)。

这样我们就可以使用复合形式初始化字段。现在,让我们考虑另一个使用场景。

使用全部字段进行数据比较

通过上篇博文,我们了解到,dataclass 能够自动生成< , =, >, <=和>=这些比较方法。但是这些比较方法的一个缺陷是,它们使用类中的所有字段进行比较,而这种情况往往不常见。更经常地,这种比较方法会给我们使用 dataclasses 造成麻烦。

考虑以下的使用情形:你有一个数据类用于存放用户的信息。现在,它可能存在以下字段:

姓名

年龄

身高

体重

你仅想比较用户对象的年龄、身高和体重。你不想比较姓名。这是后端开发者经常会遇到的使用情景。

5d8de4da4939951c5c73b3852d15513c.png

自动生成的比较方法会比较一下的数组:

3adafb59dde017fd71dd6038986efa32.png

这将会破坏我们的意图。我们不想让姓名(name)用于比较。那么,如何使用 dataclasses.field 来实现我们的想法呢?

下面是具体步骤:

99f4393a4d0f7db7fdc7e56a96689869.png

默认情况下,所用的字段都用于比较,因此我们仅仅需要指定哪些字段用于比较,而实现方法是直接把不需要的字段定义为 filed(compare=False)。

一个更为简单的应用情形也可以被讨论。让我们定义一个数据类,它被用来存储一个数字激起字符串表示。我们想让比较仅仅发生在该数字的值,而不是他的字符串表示。

0edf3abc508f156e3ea2dd55feed6f70.png

现在,我们有更大的自由来控制 dataclasses 的行为。看起来很棒!

d3c1d88fcaf51272359de435219b264e.png

使用全部字段进行数据表示

自动生成的__repr__方法使用所有的字段用于表示。当然,这也不是大多数情形下的理想选择,尤其是当你的数据类有大量的字段时。单个对象的表示会变得异常臃肿,对调试来说也不利。

7b3d95e05696bf5485f26bcc832ccbd3.png

想象一下在你的日志里看到这样的表示吧,然后还要写一个正则表达式来搜索它。太可怕了,对吧?

15b5a6fefe7a30335d636d07aa5a8897.png

当然,我们也能够个性化这种行为。考虑一个类似的使用场景,也许最合适的用于表示的属性是姓名(name)。那么对__repr__,我们仅使用它:

ae7af616fa61f132f38b58f045d73a95.png

这样看起来就很棒了。调试很方便,比较也有意义!

从初始化中省略字段

目前为止我们看到的所有例子,都有一个共同特点——即我们需要为所有被声明的字段传递值,除了有默认值之外。在那种情形下(指有默认值的情况下),我们可以选择传递值,也可以不传递。

0197ef5e08a5e7264d827ea4b399f574.png

但是,还有一种情形:我们可能不想在初始化时设定某个字段的值。这也是一种常见的使用场景。也许你在追踪一个对象的状态,并且希望它在初始化时一直被设为 False。更一般地,这个值在初始化时不能够被传递。

2b5779a01ee7ec7796ad2995268fe3c4.png

那么,我们如何实现上述想法呢?以下是具体内容:

5f15748f0292ca3da4db43ea9560322c.png

瞧啊!我们现在对 dataclasses 的使用有了更大的灵活性。

总结

希望上两篇博文能够帮助你理解 dataclass,希望你能尽快在项目中使用它们!

原文链接:https://medium.com/mindorks/understanding-python-dataclasses-part-2-660ecc11c9b8

  • 8
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Python 中调用 C 语言的和成员函数,通常需要使用 Cython 或 ctypes 库。 使用 Cython 的方法: 1. 安装 Cython:在命令行中输入 `pip install cython` 即可安装。 2. 编写 Cython 代码:在 Python 中创建一个 .pyx 文件,然后使用 Cython 语法编写代码。例如,下面是示例代码: ```python cdef extern from "my_c_library.h": cdef cppclass MyCClass: MyCClass() int my_c_function(int x, int y) cdef class PyMyCClass: cdef MyCClass *thisptr # hold a C++ instance which we're wrapping def __cinit__(self): self.thisptr = new MyCClass() def my_c_function(self, x, y): return self.thisptr.my_c_function(x, y) ``` 3. 编译 Cython 代码:使用 Cython 编译器将 .pyx 文件转换为 C/C++ 代码。例如,在命令行中输入 `cython my_cython_code.pyx` 即可生成名为 `my_cython_code.c` 的 C 代码文件。 4. 编译生成的 C 代码:使用 C 编译器将 C 代码文件编译为可执行文件或动态链接库。 5. 在 Python 中使用生成的可执行文件或动态链接库:使用 Python 的 ctypes 库或其他方法加载可执行文件或动态链接库,然后就可以在 Python 中使用 C 语言的和成员函数了。 使用 ctypes 的方法: 1. 安装 ctypes:ctypes 是 Python 的标准库,因此不需要安装。 2 ### 回答2: 在Python中,可以通过ctypes模块来调用C语言的和成员函数。 首先,需要使用ctypes模块导入C语言的动态链接库文件,使用`ctypes.CDLL()`函数进行加载。 接下来,使用`ctypes.Structure`创建一个与C语言结构相对应的Python,确保Python的成员变量与C语言中的变量一一对应。 然后,使用`ctypes.CFUNCTYPE()`函数创建一个函数型,用于定义C语言的成员函数。 接下来,使用`ctypes.cast()`函数将Python对象转换为指向C对象的指针。 通过`ctypes.POINTER()`函数创建一个指针型,指向C语言的对象。 最后,使用创建的指针型调用C语言的成员函数。 以下是一个示例代码: ```python import ctypes # 加载C语言动态链接库文件 c_lib = ctypes.CDLL("path/to/library.so") # 创建Python,与C语言结构相对应 class MyStruct(ctypes.Structure): _fields_ = [ ("data", ctypes.c_int), ("next", ctypes.POINTER(MyStruct)) ] # 定义C语言成员函数型 MyFuncType = ctypes.CFUNCTYPE(None, ctypes.POINTER(MyStruct)) # 将Python对象转换为指向C对象的指针 obj = MyStruct() c_obj = ctypes.cast(obj, ctypes.POINTER(MyStruct)) # 调用C语言成员函数 c_func = MyFuncType(c_lib.my_function) c_func(c_obj) ``` 需要注意的是,此方法只能调用已被编译为动态链接库的C语言和成员函数。同时,确保Python与C语言的成员变量一一对应。 ### 回答3: Python可以通过ctypes模块调用C的和成员函数。 首先,在Python中导入ctypes模块: import ctypes 然后,使用ctypes.CDLL或ctypes.WinDLL函数加载C库: lib = ctypes.CDLL("C库路径") 或 lib = ctypes.WinDLL("C库路径") 接下来,定义C的结构体型,可以使用ctypes.Structure来定义一个与C对应的结构体型: class CClass(ctypes.Structure): _fields_ = [("成员变量名1", ctypes.c_type1), ("成员变量名2", ctypes.c_type2), ... ("成员变量名n", ctypes.c_typen)] 然后,使用ctypes.POINTER函数定义一个指向C结构体的指针型: CClassPtr = ctypes.POINTER(CClass) 接着,通过ctypes调用C的成员函数: 1. 先定义函数原型,并设置返回值型和参数型: lib.function_name.restype = 返回值型 lib.function_name.argtypes = 参数型列表 2. 然后,通过指针调用函数: result = lib.function_name(参数列表) 注意:参数型列表的顺序应该与C的成员函数的参数顺序保持一致。 此外,还可以通过ctypes调用C的成员变量: 1. 首先,通过ctypes创建一个C对象: obj = CClass() 2. 然后,通过对象的成员变量名访问和修改成员变量的值: obj.成员变量名 = 新的值 以上就是使用ctypes模块在Python中调用C和成员函数的方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值