草庐IT

python - 字典 dict 键常量的 Python 最佳实践是什么?

coder 2023-08-13 原文

在 Python 中使用字典(dict)键时,似乎有几种通用的方法:

  1. some_dict['key_name'] # 到处都是字符串常量

  2. some_dict[KeyConstants.key_name] # where class KeyConstants: key_name: 'key_name'

  3. some_dict[KEY_NAME] # with from some_module import KEY_NAME # 模块级常量

    1. 'key_name' 的缺点是您在整个代码中重复常量。这不是干的。更糟糕的是,如果你曾经去发布你的 API(在最广泛的意义上),你将让你的 API 的消费者到处重复这些常量,如果你想将 'key_name' 更改为 'better_key_name' 这将是一个破坏性的变化.

    2. 这是类型化语言,DRY 方法,常量集中在一个地方。它唯一的缺点是它很丑陋、可读性差一点并且比较冗长。 Pythonic 原则主要禁止这样做。它使您可以轻松更改表示 key 的常量,因为每个人的编码都是针对变量 KeyConstants.key_name。它还适用于 IDE 进行重构。

    3. 模块级常量在 PEP 8 风格指南中被推荐。 ALL_CAPS_ARE_LOUD 并且更难打字。这具有选项 1 和 2 的一些优点。

字典键常量还有哪些其他最佳实践?首选上述哪种方法,何时使用?

最佳答案

  1. d['键名']
  2. d[Keys.key_name]
  3. d[KEY_NAME]

我真的不认为#3 需要模块级导入;它们只能在模块命名空间中,例如你可以做类似 How to programmatically set a global (module) variable? 的事情

与#1 相比,#2 的优势在于拼写错误和过时的值会引发属性错误“此键不存在!”而不是索引错误“找不到!” -- 这总是更好。 #2>#1。它也不是更冗长,因为如果你输入很多,你只需设置 K=Keys(或其他东西),所以你有 d[K.key_name],只是多两个字()。例如,根据我之前的感觉,我可以做以下任一事情:

import subprocess as proc
proc.Popen(..., stdout=proc.PIPE)

import subprocess as proc
PIPE = proc.PIPE
proc.Popen(..., stdout=PIPE)

from subprocess import *
Popen(..., stdout=PIPE)

关于#3,ALL_CAPS_ARE_LOUD 是有原因的;区分 d[someVariable](可以包含任何关键字)和 d[magicKeyword] 变得很困惑——而 d[MAGIC_KEYWORD] 明确表示它是一个常量,而不是某个可能保持常量的变量,例如对于 magicKeywords 中的一些变量。 #3 基本上等同于 #2,例如re.DOTALL(re 等同于 KeyConstants,无需记住 KeyConstants 容器的名称,因为它模块)。因此,#3 优于#2,除非您处于具有不同类型的键空间的奇怪情况。

DRY/OAOO 非常非常重要,但最终与这些都无关,因为你总是需要重复一个变量名才能引用它;您能做的最好的事情就是创建一个别名。

您还可以考虑#4,即为您的字典赋予属性,例如d.key_name -- 这仅适用于某些可订阅对象。

但引用 Jochen Ritzel 的评论:“使用常量键应该是一个非常罕见的场合”(使用对象的属性,或者他建议的命名元组,尽管我已经总是发现它们笨重)

关于python - 字典 dict 键常量的 Python 最佳实践是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6088581/

有关python - 字典 dict 键常量的 Python 最佳实践是什么?的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

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

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

  4. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  5. ruby-on-rails - 未初始化的常量 Psych::Syck (NameError) - 2

    在我的gem中,我需要yaml并且在我的本地计算机上运行良好。但是在将我的gem推送到ruby​​gems.org之后,当我尝试使用我的gem时,我收到一条错误消息=>"uninitializedconstantPsych::Syck(NameError)"谁能帮我解决这个问题?附言RubyVersion=>ruby1.9.2,GemVersion=>1.6.2,Bundlerversion=>1.0.15 最佳答案 经过几个小时的研究,我发现=>“YAML使用未维护的Syck库,而Psych使用现代的LibYAML”因此,为了解决

  6. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  7. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  8. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

    为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

  9. ruby-on-rails - active_admin 目录中的常量警告重新声明 - 2

    我正在使用active_admin,我在Rails3应用程序的应用程序中有一个目录管理,其中包含模型和页面的声明。时不时地我也有一个类,当那个类有一个常量时,就像这样:classFooBAR="bar"end然后,我在每个必须在我的Rails应用程序中重新加载一些代码的请求中收到此警告:/Users/pupeno/helloworld/app/admin/billing.rb:12:warning:alreadyinitializedconstantBAR知道发生了什么以及如何避免这些警告吗? 最佳答案 在纯Ruby中:classA

  10. ruby - ruby 中的 TOPLEVEL_BINDING 是什么? - 2

    它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput

随机推荐