- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
这个问题与400 threads in 20 processes outperform 400 threads in 4 processes while performing an I/O-bound task非常相似.唯一的区别是链接的问题是关于 I/O 密集型任务,而这个问题是关于 CPU 密集型任务。
下面是实验代码,可以启动指定数量的工作进程,然后在每个进程内启动指定数量的工作线程,执行计算第n个质数的任务。
import math
import multiprocessing
import random
import sys
import time
import threading
def main():
processes = int(sys.argv[1])
threads = int(sys.argv[2])
tasks = int(sys.argv[3])
# Start workers.
in_q = multiprocessing.Queue()
process_workers = []
for _ in range(processes):
w = multiprocessing.Process(target=process_worker, args=(threads, in_q))
w.start()
process_workers.append(w)
start_time = time.time()
# Feed work.
for nth in range(1, tasks + 1):
in_q.put(nth)
# Send sentinel for each thread worker to quit.
for _ in range(processes * threads):
in_q.put(None)
# Wait for workers to terminate.
for w in process_workers:
w.join()
total_time = time.time() - start_time
task_speed = tasks / total_time
print('{:3d} x {:3d} workers => {:6.3f} s, {:5.1f} tasks/s'
.format(processes, threads, total_time, task_speed))
def process_worker(threads, in_q):
thread_workers = []
for _ in range(threads):
w = threading.Thread(target=thread_worker, args=(in_q,))
w.start()
thread_workers.append(w)
for w in thread_workers:
w.join()
def thread_worker(in_q):
while True:
nth = in_q.get()
if nth is None:
break
num = find_nth_prime(nth)
#print(num)
def find_nth_prime(nth):
# Find n-th prime from scratch.
if nth == 0:
return
count = 0
num = 2
while True:
if is_prime(num):
count += 1
if count == nth:
return num
num += 1
def is_prime(num):
for i in range(2, int(math.sqrt(num)) + 1):
if num % i == 0:
return False
return True
if __name__ == '__main__':
main()
下面是我运行这个程序的方式:
python3 foo.py <PROCESSES> <THREADS> <TASKS>
例如,python3 foo.py 20 20 2000
创建 20 个工作进程,每个工作进程有 20 个线程(因此总共有 400 个工作线程)并执行 2000 个任务。最后,该程序打印出执行任务所花费的时间以及平均每秒执行的任务数。
我正在具有 8 GB RAM 和 4 个 CPU 的 Linode 虚拟专用服务器上测试此代码。它正在运行 Debian 9。
$ cat /etc/debian_version
9.9
$ python3
Python 3.5.3 (default, Sep 27 2018, 17:25:39)
[GCC 6.3.0 20170516] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>>
$ free -m
total used free shared buff/cache available
Mem: 7987 67 7834 10 85 7734
Swap: 511 0 511
$ nproc
4
这里有一些试运行,400 个工作线程分布在 20 个工作进程之间(即,20 个工作进程中的每一个有 20 个工作线程)。
结果如下:
$ python3 bar.py 20 20 2000
20 x 20 workers => 12.702 s, 157.5 tasks/s
$ python3 bar.py 20 20 2000
20 x 20 workers => 13.196 s, 151.6 tasks/s
$ python3 bar.py 20 20 2000
20 x 20 workers => 12.224 s, 163.6 tasks/s
$ python3 bar.py 20 20 2000
20 x 20 workers => 11.725 s, 170.6 tasks/s
$ python3 bar.py 20 20 2000
20 x 20 workers => 10.813 s, 185.0 tasks/s
当我使用 top
命令监视 CPU 使用率时,我看到每个 python3
工作进程消耗大约 15% 到 25% 的 CPU。
现在我以为我只有 4 个 CPU。即使我启动 20 个工作进程,在物理时间的任何时间点最多也只能运行 4 个进程。此外,由于全局解释器锁 (GIL),每个进程中只有一个线程(因此总共最多 4 个线程)可以在物理时间的任何时间点运行。
因此,我想如果我把进程数减少到4个,把每个进程的线程数增加到100个,这样总线程数还是400个,性能应该不会变差。
但测试结果表明,4 个进程每个包含 100 个线程的性能始终比 20 个进程每个包含 20 个线程的性能差。
$ python3 bar.py 4 100 2000
4 x 100 workers => 19.840 s, 100.8 tasks/s
$ python3 bar.py 4 100 2000
4 x 100 workers => 22.716 s, 88.0 tasks/s
$ python3 bar.py 4 100 2000
4 x 100 workers => 20.278 s, 98.6 tasks/s
$ python3 bar.py 4 100 2000
4 x 100 workers => 19.896 s, 100.5 tasks/s
$ python3 bar.py 4 100 2000
4 x 100 workers => 19.876 s, 100.6 tasks/s
每个 python3
工作进程的 CPU 使用率在 50% 到 66% 之间。
只是为了比较,我记录的事实是情况 1 和情况 2 都优于我们在单个进程中拥有所有 400 个线程的情况。这显然是由于全局解释器锁 (GIL)。
$ python3 bar.py 1 400 2000
1 x 400 workers => 34.762 s, 57.5 tasks/s
$ python3 bar.py 1 400 2000
1 x 400 workers => 35.276 s, 56.7 tasks/s
$ python3 bar.py 1 400 2000
1 x 400 workers => 32.589 s, 61.4 tasks/s
$ python3 bar.py 1 400 2000
1 x 400 workers => 33.974 s, 58.9 tasks/s
$ python3 bar.py 1 400 2000
1 x 400 workers => 35.429 s, 56.5 tasks/s
单个 python3
工作进程的 CPU 使用率在 110% 到 115% 之间。
同样,只是为了比较,这里是有 400 个进程时的结果,每个进程都有一个线程。
$ python3 bar.py 400 1 2000
400 x 1 workers => 8.814 s, 226.9 tasks/s
$ python3 bar.py 400 1 2000
400 x 1 workers => 8.631 s, 231.7 tasks/s
$ python3 bar.py 400 1 2000
400 x 1 workers => 10.453 s, 191.3 tasks/s
$ python3 bar.py 400 1 2000
400 x 1 workers => 8.234 s, 242.9 tasks/s
$ python3 bar.py 400 1 2000
400 x 1 workers => 8.324 s, 240.3 tasks/s
每个 python3
工作进程的 CPU 使用率在 1% 到 3% 之间。
从每个案例中选择中位数结果,我们得到这个总结:
Case 1: 20 x 20 workers => 12.224 s, 163.6 tasks/s
Case 2: 4 x 100 workers => 19.896 s, 100.5 tasks/s
Case 3: 1 x 400 workers => 34.762 s, 57.5 tasks/s
Case 4: 400 x 1 workers => 8.631 s, 231.7 tasks/s
为什么 20 个进程 x 20 个线程的性能优于 4 个进程 x 100 个线程,即使我只有 4 个 CPU?
事实上,尽管只有 4 个 CPU,但 400 个进程 x 1 个线程的性能最好?为什么?
最佳答案
在 Python 线程可以执行代码之前,它需要获取 Global Interpreter Lock (GIL) .这是一个每个进程 锁。在某些情况下(例如,当等待 I/O 操作完成时),一个线程会定期释放 GIL,以便其他线程可以获取它。如果事件线程在特定时间内没有放弃锁,其他线程可以向事件线程发出信号以释放 GIL,以便它们轮流使用。
考虑到这一点,让我们看看您的代码在我的 4 核笔记本电脑上的执行情况:
在最简单的情况下(1 个进程和 1 个线程)我得到大约 155 个任务/秒。 GIL 并没有妨碍我们。我们使用 100% 的一个核心。
如果我增加线程数(1 个进程有 4 个线程),我将获得大约 70 个任务/秒。起初这可能是违反直觉的,但可以解释为您的代码主要受 CPU 限制,因此所有线程几乎一直都需要 GIL。一次只有其中一个可以运行它的计算,因此我们无法从多线程中获益。结果是我们使用了我的 4 个内核中每个内核的 ~25%。更糟糕的是,获取和释放 GIL 以及上下文切换会增加显着的开销,从而降低整体性能。
添加更多线程(1 个进程有 400 个线程)没有帮助,因为一次只执行其中一个线程。在我的笔记本电脑上,性能与情况 (2) 非常相似,我们再次使用了我的 4 个内核中的大约 25%。
使用 4 个进程,每个进程有 1 个线程,我得到大约 550 个任务/秒。几乎是我在案例 (1) 中得到的 4 倍。实际上,由于进程间通信和锁定共享队列所需的开销,少了一点。请注意,每个进程都使用自己的 GIL。
有 4 个进程,每个进程运行 100 个线程,我得到大约 290 个任务/秒。我们再次看到在 (2) 中看到的减速,这次影响了每个单独的进程。
有 400 个进程,每个进程运行 1 个线程,我得到大约 530 个任务/秒。与 (4) 相比,我们看到由于进程间通信和共享队列上的锁定而产生的额外开销。
请引用David Beazley's talk Understanding the Python GIL以获得对这些影响的更深入解释。
备注:Some Python interpreters like CPython and PyPy have a GIL while others like Jython and IronPython don't .如果您使用另一个 Python 解释器,您可能会看到非常不同的行为。
关于python - 在 4 个 CPU 上执行 CPU 绑定(bind)任务时,20 个进程中的 400 个线程优于 4 个进程中的 400 个线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56274688/
我想做的是让 JTextPane 在 JPanel 中占用尽可能多的空间。对于我使用的 UpdateInfoPanel: public class UpdateInfoPanel extends JP
我在 JPanel 中有一个 JTextArea,我想将其与 JScrollPane 一起使用。我正在使用 GridBagLayout。当我运行它时,框架似乎为 JScrollPane 腾出了空间,但
我想在 xcode 中实现以下功能。 我有一个 View Controller 。在这个 UIViewController 中,我有一个 UITabBar。它们下面是一个 UIView。将 UITab
有谁知道Firebird 2.5有没有类似于SQL中“STUFF”函数的功能? 我有一个包含父用户记录的表,另一个表包含与父相关的子用户记录。我希望能够提取用户拥有的“ROLES”的逗号分隔字符串,而
我想使用 JSON 作为 mirth channel 的输入和输出,例如详细信息保存在数据库中或创建 HL7 消息。 简而言之,输入为 JSON 解析它并输出为任何格式。 最佳答案 var objec
通常我会使用 R 并执行 merge.by,但这个文件似乎太大了,部门中的任何一台计算机都无法处理它! (任何从事遗传学工作的人的附加信息)本质上,插补似乎删除了 snp ID 的 rs 数字,我只剩
我有一个以前可能被问过的问题,但我很难找到正确的描述。我希望有人能帮助我。 在下面的代码中,我设置了varprice,我想添加javascript变量accu_id以通过rails在我的数据库中查找记
我有一个简单的 SVG 文件,在 Firefox 中可以正常查看 - 它的一些包装文本使用 foreignObject 包含一些 HTML - 文本包装在 div 中:
所以我正在为学校编写一个 Ruby 程序,如果某个值是 1 或 3,则将 bool 值更改为 true,如果是 0 或 2,则更改为 false。由于我有 Java 背景,所以我认为这段代码应该有效:
我做了什么: 我在这些账户之间创建了 VPC 对等连接 互联网网关也连接到每个 VPC 还配置了路由表(以允许来自双方的流量) 情况1: 当这两个 VPC 在同一个账户中时,我成功测试了从另一个 La
我有一个名为 contacts 的表: user_id contact_id 10294 10295 10294 10293 10293 10294 102
我正在使用 Magento 中的新模板。为避免重复代码,我想为每个产品预览使用相同的子模板。 特别是我做了这样一个展示: $products = Mage::getModel('catalog/pro
“for”是否总是检查协议(protocol)中定义的每个函数中第一个参数的类型? 编辑(改写): 当协议(protocol)方法只有一个参数时,根据该单个参数的类型(直接或任意)找到实现。当协议(p
我想从我的 PHP 代码中调用 JavaScript 函数。我通过使用以下方法实现了这一点: echo ' drawChart($id); '; 这工作正常,但我想从我的 PHP 代码中获取数据,我使
这个问题已经有答案了: Event binding on dynamically created elements? (23 个回答) 已关闭 5 年前。 我有一个动态表单,我想在其中附加一些其他 h
我正在尝试找到一种解决方案,以在 componentDidMount 中的映射项上使用 setState。 我正在使用 GraphQL连同 Gatsby返回许多 data 项目,但要求在特定的 pat
我在 ScrollView 中有一个 View 。只要用户按住该 View ,我想每 80 毫秒调用一次方法。这是我已经实现的: final Runnable vibrate = new Runnab
我用 jni 开发了一个 android 应用程序。我在 GetStringUTFChars 的 dvmDecodeIndirectRef 中得到了一个 dvmabort。我只中止了一次。 为什么会这
当我到达我的 Activity 时,我调用 FragmentPagerAdapter 来处理我的不同选项卡。在我的一个选项卡中,我想显示一个 RecyclerView,但他从未出现过,有了断点,我看到
当我按下 Activity 中的按钮时,会弹出一个 DialogFragment。在对话框 fragment 中,有一个看起来像普通 ListView 的 RecyclerView。 我想要的行为是当
我是一名优秀的程序员,十分优秀!