草庐IT

python - 在不解压其内容的情况下传递集合参数

coder 2023-08-21 原文

问题编写__init__ 直接将集合作为参数而不是解压其内容的优缺点是什么?

上下文:我正在编写一个类来处理来自数据库表中多个字段的数据。我遍历了一些大的(约 1 亿行)查询结果,一次将一行传递给执行处理的类。每一行都作为元组(或可选地作为字典)从数据库中检索。

讨论:假设我只对三个字段感兴趣,但是传递到我的类中的内容取决于查询,而查询是由用户编写的。最基本的方法可能是以下方法之一:

class Direct:
    def __init__(self, names):
        self.names = names

class Simple:
    def __init__(self, names):
        self.name1 = names[0]
        self.name2 = names[1]
        self.name3 = names[2]

class Unpack:
    def __init__(self, names):
        self.name1, self.name2, self.name3 = names

以下是可能传递给新实例的行的一些示例:

good = ('Simon', 'Marie', 'Kent')                 # Exactly what we want
bad1 = ('Simon', 'Marie', 'Kent', '10 Main St')   # Extra field(s) behind
bad2 = ('15', 'Simon', 'Marie', 'Kent')           # Extra field(s) in front
bad3 = ('Simon', 'Marie')                         # Forgot a field

当遇到上述情况时,Direct 始终运行(至少到此为止)但很可能出现错误 (GIGO)。它接受一个参数并完全按照给定的方式分配它,因此这可以是一个元组或任意大小的列表、一个 Null 值、一个函数引用等。这是我能想到的初始化对象,但我觉得当我给它提供显然不是设计用来处理的数据时,类应该立即提示。

Simple 正确处理 bad1,在给定 bad2 时有错误,在给定 bad3 时抛出错误。能够有效地截断来自 bad1 的输入很方便,但不值得来自 bad2 的错误。这让人感觉天真且前后矛盾。

Unpack 似乎是最安全的方法,因为它在所有三种“坏”情况下都会引发错误。我们最不想做的就是悄悄地用不良信息填充我们的数据库,对吧?它直接获取元组,但允许我将其内容标识为不同的属性,而不是强制我继续引用索引,并在元组大小错误时提示。

另一方面,为什么要传递一个集合?因为我知道我总是想要三个字段,所以我可以定义 __init__ 以显式接受三个参数,并在将集合传递给新对象时使用 *-operator 解压集合:

class Explicit:
    def __init__(self, name1, name2, name3):
        self.name1 = name1
        self.name2 = name2
        self.name3 = name3

names = ('Guy', 'Rose', 'Deb')
e = Explicit(*names)

我看到的唯一区别是 __init__ 定义有点冗长,当元组是错误的大小。从哲学上讲,如果我们正在获取一组数据(查询的一行)并检查其部分(三个字段),这似乎是有道理的,我们应该传递一组数据(元组)但存储其部分(三个字段)属性)。所以 Unpack 会更好。

如果我想接受不确定数量的字段,而不是总是三个,我仍然可以选择直接传递元组或使用任意参数列表 (*args, **kwargs) 和 *-运营商拆包。所以我想知道,这是一个完全中性的风格决定吗?

最佳答案

这个问题的最佳答案可能是尝试不同的方法,看看什么对您最有意义并且阅读您的代码的其他人最容易理解

现在我受益于更多的经验,我会问自己,我打算如何访问这些值?

当我访问此集合中的任何一个值时,我是否可能会使用同一子例程或代码部分中的大部分或全部值? 如果是这样,“直接”方法是一个不错的选择;它是最紧凑的,它让我可以将集合视为一个集合,直到我绝对需要注意里面的东西。

另一方面,如果我在这里使用一些值,在那里使用一些值,我不想一直记住要访问哪个索引或在我可以直接引用时以字典键的形式添加冗长的内容使用单独命名的属性的值。在这种情况下,我可能会避免使用“直接”方法,这样我什至只需考虑在类首次初始化时有一个集合这一事实。

其余的每一种方法都涉及将集合拆分为不同的属性,我认为这里明显的赢家是“显式”方法。 “简单”和“解包”方法共享对集合顺序的隐藏依赖性,没有提供任何真正的优势。

关于python - 在不解压其内容的情况下传递集合参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17223607/

有关python - 在不解压其内容的情况下传递集合参数的更多相关文章

  1. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  2. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  3. ruby - 默认情况下使选项为 false - 2

    这是在Ruby中设置默认值的常用方法:classQuietByDefaultdefinitialize(opts={})@verbose=opts[:verbose]endend这是一个容易落入的陷阱:classVerboseNoMatterWhatdefinitialize(opts={})@verbose=opts[:verbose]||trueendend正确的做法是:classVerboseByDefaultdefinitialize(opts={})@verbose=opts.include?(:verbose)?opts[:verbose]:trueendend编写Verb

  4. ruby - RSpec - 使用测试替身作为 block 参数 - 2

    我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere

  5. ruby - 如何在 Ruby 中拆分参数字符串 Bash 样式? - 2

    我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"

  6. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

    我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

  7. ruby - 检查方法参数的类型 - 2

    我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)

  8. ruby-on-rails - 在默认方法参数中使用 .reverse_merge 或 .merge - 2

    两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option

  9. ruby - 定义方法参数的条件 - 2

    我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano

  10. ruby - rails 3 redirect_to 将参数传递给命名路由 - 2

    我没有找到太多关于如何执行此操作的信息,尽管有很多关于如何使用像这样的redirect_to将参数传递给重定向的建议:action=>'something',:controller=>'something'在我的应用程序中,我在路由文件中有以下内容match'profile'=>'User#show'我的表演Action是这样的defshow@user=User.find(params[:user])@title=@user.first_nameend重定向发生在同一个用户Controller中,就像这样defregister@title="Registration"@user=Use

随机推荐