- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
来自 http://jaynes.colorado.edu/PythonIdioms.html
"Build strings as a list and use''.join at the end. join is a stringmethod called on the separator, notthe list. Calling it from the emptystring concatenates the pieces with noseparator, which is a Python quirk andrather surprising at first. This isimportant: string building with + isquadratic time instead of linear! Ifyou learn one idiom, learn this one.
Wrong: for s in strings: result += s
Right: result = ''.join(strings)"
我不确定为什么这是真的。如果我有一些字符串,我想加入它们,对我来说,将它们放在列表中然后调用 ''.join 对我来说直观上并不是更好。将它们放入列表中不会产生一些开销吗?澄清...
Python 命令行:
>>> str1 = 'Not'
>>> str2 = 'Cool'
>>> str3 = ''.join([str1, ' ', str2]) #The more efficient way **A**
>>> print str3
Not Cool
>>> str3 = str1 + ' ' + str2 #The bad way **B**
>>> print str3
Not Cool
A真的是线性时间而B是二次时间吗?
最佳答案
是的。对于您选择的示例,重要性并不明确,因为您只有两个非常短的字符串,因此追加可能会更快。
但是每次你在 Python 中对字符串执行 a + b
时,它都会导致新的分配,然后将 a 和 b 中的所有字节复制到新字符串中。如果您在包含大量字符串的循环中执行此操作,则必须一次又一次地复制这些字节,并且每次复制的数量都会变长。这给出了二次行为。
另一方面,创建字符串列表不会复制字符串的内容——它只是复制引用。这是非常快的,并且以线性时间运行。然后 join 方法只进行一次内存分配,并将每个字符串复制到正确的位置一次。这也只需要线性时间。
所以是的,如果您可能要处理大量字符串,请务必使用 ''.join
习惯用法。对于两个字符串,这无关紧要。
如果您需要更有说服力,请自己尝试创建一个包含 1000 万个字符的字符串:
>>> chars = ['a'] * 10000000
>>> r = ''
>>> for c in chars: r += c
>>> print len(r)
相比于:
>>> chars = ['a'] * 10000000
>>> r = ''.join(chars)
>>> print len(r)
第一种方法大约需要 10 秒。第二个需要不到 1 秒。
关于Python 字符串连接成语。需要澄清。,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1967723/
我被指出“安全 bool 成语”,在试图破译发生了什么之后(解释 supplied on the site 不足以让我理解它为什么起作用),我决定尝试将以下代码分开并尝试尽可能地简化它。该网站提供的代
访问标量表达式的最简洁和字节码有效的方法是什么 多次从另一个表情的深处? 以下代码中的所有函数(例如 scalar4 )都可以根据需要运行。但只有 字节编码器 发出高效的字节码(尽管它以 ISTORE
我正在处理一个 Chart 类,它有一个 margin 参数,它包含 :top、:bottom、:right 和 :left 值。我的第一个选择是使 margin 成为 setter 并像这样设置值:
想象一下,您正在使用明显的蛮力算法生成斐波那契数列。如果我知道我想提前生成的斐波那契数,我可以使用幂连接 ^: 做这样的事情: (, [: +/ _2&{.)^:20 i.2 当斐波那契数达到某个极限
有很多地方(例如 How to use requestAnimationFrame? )修复了 window.requestAnimationFrame 如下。我不明白为什么赋值的右侧包含在函数调用中
这个问题不太可能帮助任何 future 的访问者;它只与一个小的地理区域、一个特定的时间点或一个非常狭窄的情况有关,这些情况并不普遍适用于互联网的全局受众。为了帮助使这个问题更广泛地适用,visit
我已经使用 idiorm 玩了几天了,并且逐渐设法让它真正开始执行查询。虽然我遇到了一些奇怪的事情,但我无法弄清楚。 find_many() 函数只返回一条记录,它始终是数据库中的最后一条记录。例如,
我读到 Pimpl 有利于二进制兼容性,接口(interface)有利于轻松切换实现。我需要结合这两种技术,让我的应用程序能够通过配置文件切换底层实现。 以下是我当前的设计布局: 类 Foo:提供面向
今天我了解了 C++“memberspace”惯用语,它粗略地滥用了 C++ 的一个属性,该属性使 T::bar 以及 T.bar 工作,当T 在某些范围内既是类型又是对象。 struct A {
这个 jQuery 片段最清楚的常用习惯用法是什么? $('#someTextarea').val( $('#someTextarea').val() + someString ); 将原始代码包装在
我记得很久以前读过有关复杂对象配置情况下的 C++ 构造函数习惯用法。它特别有用,因为它有助于为一些讨厌的概念启用 RAII,这些概念方式太多(通常是相互冲突的)选项。 这是一个简单的例子。假设您要为
boost 库是否提供了安全 bool 习惯用法的实现,以便我可以从中派生我的类? 如果是 - 它在哪里? 如果不是 - 除了我自己实现之外,我还有哪些选择? 我发现了以下类似的问题:“Is ther
我想使用 pimpl idiom 和继承。 这里是基础公共(public)类及其实现类: class A { public: A(){pAImpl = new AImpl;};
通过使用 Copy & Swap我们可以轻松实现具有强大异常安全性的复制分配: T& operator = (T other){ using std::swap; swap(*this
这个问题在这里已经有了答案: Iterating over every two elements in a list [duplicate] (22 个回答) 关闭3年前。 我觉得我花了很多时间用 P
关于 pimpl idiom 有一些关于 SO 的问题,但我更好奇它在实践中的使用频率。 我了解在性能和封装之间需要权衡取舍,另外由于额外的重定向会导致一些调试烦恼。 这样,这是应该在每个类(clas
这个问题在这里已经有了答案: Is the PIMPL idiom really used in practice? (12 个回答) 关闭7年前。 背景: PIMPL Idiom (指向 IPLem
这是网络上最常见的实现方式 private static class LazySomethingHolder { public static Something something = new S
像std::iterator_traits 这样的包罗万象的特征类通过将类型的属性与其定义分开是很有用的,例如,可以在定义完成之前使属性可用。 除了每个客户端类本身之外还定义特征是不方便的,因为特征通
我通常为 pimpl 使用 boost::scoped_ptr(出于一个原因,因为如果我忘记处理复制构造函数,我不会感到惊讶) 然而,对于模板,我不能只将析构函数放在完全定义了 impl 的 cpp
我是一名优秀的程序员,十分优秀!