我正在尝试将一个较长的空心“数据”类转换为命名元组。我的类目前看起来像这样:
class Node(object):
def __init__(self, val, left=None, right=None):
self.val = val
self.left = left
self.right = right
转换为命名后,
它看起来像这样:
from collections import namedtuple
Node = namedtuple('Node', 'val left right')
但是这里有一个问题。我最初的类允许我只传入一个值,并通过为命名/关键字参数使用默认值来处理默认值。类似于:
class BinaryTree(object):
def __init__(self, val):
self.root = Node(val)
但这在我重构的命名元组中不起作用,因为它要求我传递所有字段。我当然可以将出现的Node(val)
替换为Node(val,None,None)
,但这不符合我的喜好。
那么,是否有一个很好的技巧可以让我的重写成功而不会增加很多代码复杂性(元编程),或者我应该吞下药丸并继续“搜索和替换”?:)
将其包装在函数中。
NodeT = namedtuple('Node', 'val left right')
def Node(val, left=None, right=None):
return NodeT(val, left, right)
我将namedtuple子类化并覆盖__new__
方法:
from collections import namedtuple
class Node(namedtuple('Node', ['value', 'left', 'right'])):
__slots__ = ()
def __new__(cls, value, left=None, right=None):
return super(Node, cls).__new__(cls, value, left, right)
这保留了直观的类型层次结构,而伪装成类的工厂函数的创建并没有这样做。
使用默认值参数。
>>> from collections import namedtuple
>>> fields = ('val', 'left', 'right')
>>> Node = namedtuple('Node', fields, defaults=(None,) * len(fields))
>>> Node()
Node(val=None, left=None, right=None)
或者更好的是,使用新的dataclasses库,它比namedtuple好得多。
>>> from dataclasses import dataclass
>>> from typing import Any
>>> @dataclass
... class Node:
... val: Any = None
... left: 'Node' = None
... right: 'Node' = None
>>> Node()
Node(val=None, left=None, right=None)
设置<代码>节点。__新_ _。__defaults__为默认值。
>>> from collections import namedtuple
>>> Node = namedtuple('Node', 'val left right')
>>> Node.__new__.__defaults__ = (None,) * len(Node._fields)
>>> Node()
Node(val=None, left=None, right=None)
设置节点。__new__。func_defaults
设置为默认值。
>>> from collections import namedtuple
>>> Node = namedtuple('Node', 'val left right')
>>> Node.__new__.func_defaults = (None,) * len(Node._fields)
>>> Node()
Node(val=None, left=None, right=None)
在Python的所有版本中,如果设置的默认值少于namedtuple中的值,默认值将应用于最右边的参数。这允许您将一些参数作为必需的参数。
>>> Node.__new__.__defaults__ = (1,2)
>>> Node()
Traceback (most recent call last):
...
TypeError: __new__() missing 1 required positional argument: 'val'
>>> Node(3)
Node(val=3, left=1, right=2)
这里有一个包装器,它甚至允许您(可选)将默认值设置为None
以外的值。这不支持必需的参数。
import collections
def namedtuple_with_defaults(typename, field_names, default_values=()):
T = collections.namedtuple(typename, field_names)
T.__new__.__defaults__ = (None,) * len(T._fields)
if isinstance(default_values, collections.Mapping):
prototype = T(**default_values)
else:
prototype = T(*default_values)
T.__new__.__defaults__ = tuple(prototype)
return T
例子:
>>> Node = namedtuple_with_defaults('Node', 'val left right')
>>> Node()
Node(val=None, left=None, right=None)
>>> Node = namedtuple_with_defaults('Node', 'val left right', [1, 2, 3])
>>> Node()
Node(val=1, left=2, right=3)
>>> Node = namedtuple_with_defaults('Node', 'val left right', {'right':7})
>>> Node()
Node(val=None, left=None, right=7)
>>> Node(4)
Node(val=4, left=None, right=7)
问题内容: 我正在尝试将冗长的空心“数据”类转换为命名元组。我的班级目前看起来像这样: 转换为它后看起来像: 但是这里有一个问题。我的原始类允许我只传入一个值,并通过对named / keyword参数使用默认值来处理默认值。就像是: 但这在重构的元组中不起作用,因为它希望我传递所有字段。我当然可以替换to的出现,但这并不是我喜欢的。 那么,是否存在一个可以使我的重写成功而又不增加很多代码复杂性(
我遇到了一种巧妙的方法,让命名图普尔从这里使用默认参数。 节点(val=无,left=无,right=无) 如果您希望“right”的默认值为空列表,您会怎么做?您可能知道,使用可变的默认参数(例如列表)是不可以的。 有没有一个简单的方法来实现这一点?
问题内容: 当读取a的初始化程序时,我看到一些参数默认为value 。什么是关键词代表什么? 问题答案: 这不是有效的Swift代码,它是即时生成的。 在这里意味着有一些默认值,但发电机不能想像它适合你才能看到它。从技术上来说,默认值是一个内联函数,因此不能轻易将其转换为简单的声明。 您可以看到类似的声明 其中默认为(在Swift 1.x中)和默认为(在Swift 1.x中)。 在的情况下,默认值
问题内容: 我以为我可以在Python 2的函数调用中在变长位置参数之后使用命名参数,但是在导入python类时得到了提示。例如,我正在使用以下“ get”方法编写: 错误看起来像: 我希望能够以几种方式调用该方法: 等等 问题答案: 它确实有效,但仅在Python 3中有效。请参阅PEP 3102。浏览“新功能”文档后,似乎没有2.x向后移植,因此您很不走运。您必须接受任何关键字参数()并手动对
问题内容: 因此,我正在使用的API的Web POST请求中预期的可选参数实际上也是python中的保留字。因此,如何在方法调用中命名参数: 由于“ from”是关键字,因此由于语法错误而失败。如何以不遇到语法错误的方式传递此密码? 问题答案: 将其作为命令传递。
4.2. 使用可选参数和命名参数 Python 允许函数参数有缺省值;如果调用函数时不使用参数,参数将获得它的缺省值。此外,通过使用命名参数还可以以任意顺序指定参数。SQL Server Transact/SQL 中的存储过程也可以做到这些;如果你是脚本高手,你可以略过这部分。 info 函数就是这样一个例子,它有两个可选参数。 def info(object, spacing=10, coll