为什么多线程是个坏主意

在 Unix编程艺术 中,提到了尽量避免多线程编程模型, 认为这样只会增加复杂度, 提倡使用多进程, 这样本质上就可以避免多线程『共享内存数据』产生的 “corruotped memory” 问题。

其中, 提到了一篇文章 Why Threads Are A Bad Idea, 对于多线程编程和事件编程分析的非常好, 具体的翻译如下:


1 介绍

线程的背景:
  • 在操作系统中出现多线程
  • 逐渐演变成 用户层面的编程工具
  • 被认为是多种问题的一种通用解决方案
  • 每一个程序员都需要成为 一个多线程编程的高手吗?
根本性的问题:

多线程的程序非常难以正确的编写!!!

替代性的方案:

使用事件驱动的编程方法

特别声明:
  • 对于大部分的多线程程序,使用事件驱动是一个更好的选择
  • 只有当使用CPU多核的时候, 才需要使用多线程编程

2 多线程的本质

  • 一般用来管理并发问题
  • 多个独立相互执行的任务
  • 共享的内存
  • 预先的安排机制(Pre-emptive scheduling)
  • 同步机制(synchronization)

3 多线程的用途

  • 操作系统: 对每一个用户进程分配一个内核线程
  • 科学应用程序: 每个CPU分配一个线程(对计算要求性很高的程序)
  • 分布式系统: 进程请求并行(同步记性的I/O操作)
  • GUIs程序
    • 线程对应用户的行为. 在长时间的后台计算过程中仍然可以处理图形展示
    • 多媒体, 动画方面的程序编写

4 多线程有什么问题?

  • 对于一般的程序员而言,难以掌握。
  • 即使对于专家,多线程编程也是痛苦的。

5 为什么多线程编程很难?

  • Synchronization(同步机制):

    • 必须通过锁来共享数据
    • 忘记了加锁?就会导致受污染的数据
  • 死锁
    • 依赖锁,会导致循环依赖
    • 每个处理程序等待其他处理程序: 导致系统挂起

6 为什么多线程编程很难?

  • 难以调试: 因为 数据依赖,时间依赖
  • 线程破坏了抽象: 无法设计出模块化的程序
  • 因为锁导致回调无法完成

7 为什么多线程编程很难?

  • 很难达到非常好的性能

    • 简单的锁导致了低并发
    • 而精密的锁又会导致复杂度提升, 降低了一般情况下的性能
    • OSes限制了性能提升(调度, 环境切换)
  • 线程不受支持
    • 难以支持多线程代码(mac, windows)
    • 一些标志库不是线程安全的
    • 内核调用, windows系统不是多线程
    • 很少有多线程编程的调试工具
  • 通常不需要并发场景

8 时间驱动编程

  • 一个执行流进程: 没有CPU的并发
  • 在时间上注册消息(通过回调)
  • 事件轮询等待消息, 调用处理器模型
  • 时间处理器没有抢断
  • 处理器通常是 短生命周期的

9 事件驱动编程被用来干什么

  • 大多数的GUIs编程:

    • 一个处理器对应一个事件
    • 处理器用来执行行为(撤销,删除文件等)
  • 分布式系统
    • 一个处理器用来对应一个输入源
    • 处理进来的请求,返回结果
    • 事件驱动的I/O 来处理 I/O并发

10 事件驱动编程的问题

  • 长时间运行的时间处理器会导致 程序没有反应, 解决办法:

    • 对于长时间运行的程序Fork off子程序处理, 当处理结束后使用事件
    • 打断处理器执行(比如: 事件驱动的I/O)
    • 定期回调 时间处理器中的 事件循环
  • 通过处理器无法维护本地内存状态(处理器必须返回)
  • 没有CPU的并发(不太合适科学计算程序)
  • 事件驱动的编程并不总是被支持

11 多线程编程 VS 事件驱动编程

  • 事件驱动编发编程尽可能的避免 并发, 而多线程编程则倾向于并发:
    • 使用事件驱动编程更加容易: 不用考虑并发, 不用考虑抢占, 不用考虑同步和死锁
    • 只在特定的情况下,才使用复杂的技术栈
    • 使用多线程编程, 即使最简单的程序也需要面对很高的复杂度(full complexity)
  • 使用事件驱动更加容易调试
    • 事件驱动编程只和时间依赖有关, 不需要考虑内部的调度
    • 问题更加容易跟踪: 较慢的按钮点击反应 和 内存数据污染 时候, 前者问题更加容易定位

12 多线程编程 VS 事件驱动编程

  • 在单个CPU上时间驱动程序比线程更加快速
    • 没有锁的覆盖
    • 没有上下文环境的 切换
  • 事件驱动编程更加面向接口编程
  • 多线程提供了真正的并发性
    • 对于多CPU的机器来说,是可以扩展性能
    • 可以长时间的运行处理程序而不需要冻结

13 你需要放弃多线程吗?

  • 不需要的情况: 对于应该程序性能要求很高的服务(比如: 数据库服务器)
  • 但是, 尽可能的避免多线程编程:
    • 对于 GUIs程序, 分布式系统, 性能要求不高的, 使用事件编程, 不是多线程
    • 只有当真正的多核CPU并发需要使用到的时候,使用多线程编程
    • 当使用多线程编程的时候,将多线程编程模块与其他模块进行隔离, 保持大部分代码都是单线程模型

隔离多线程的模块:

14 总结

  • 并发从根本上是很难的, 尽可能的避免
  • 多线程比事件更加强大,但是这种强大的功能很少真正需要
  • 多线程编程比事件编程更加难以写出正确的代码, 只有真正的专家才能掌握
  • 将事件 编程当做基本的开发工具(对于GUIs 和 分布式系统)
  • 只有当性能要求很高的服务时候,才使用 多线程


 转载自 并发编程网 - ifeve.com

时间: 2024-08-30 00:24:35

为什么多线程是个坏主意的相关文章

C#中的线程(三)多线程

Keywords:C# 线程Source:http://www.albahari.com/threading/Author: Joe AlbahariTranslator: Swanky WuPublished: http://www.cnblogs.com/txw1958/Download:http://www.albahari.info/threading/threading.pdf    第三部分:使用多线程   单元模式和Windows Forms 单元模式线程是一个自动线程安全机制,

java 多线程-线程通信实例讲解_java

线程通信的目标是使线程间能够互相发送信号.另一方面,线程通信使线程能够等待其他线程的信号. 通过共享对象通信 忙等待 wait(),notify()和 notifyAll() 丢失的信号 假唤醒 多线程等待相同信号 不要对常量字符串或全局对象调用 wait() 通过共享对象通信 线程间发送信号的一个简单方式是在共享对象的变量里设置信号值.线程 A 在一个同步块里设置 boolean 型成员变量 hasDataToProcess 为 true,线程 B 也在同步块里读取 hasDataToProc

VB.NET多线程应用

开发者一直要求微软为VB加入更多的多线程功能,对于VB.NET也是这样.VB6已经支持建立多线程的EXE.DLL和OCX.不过使用多线程这个词语,可能也不太确切.因此VB6仅支持运行多个单线程的单元.一个单元实际上是代码执行的空间,而单元的边界限制了代码访问任何单元以外的事物. VB.NET就不同了,它支持建立自由线程(free-threaded)的应用.这意味着多个线程可以访问同样一套的共享数据.本文的以下部分将讨论一下多线程的一些基本点. 问题 虽然VB6支持多个单线程的单元,不过它并不支持

多线程中锁的实现.

*引用本文请注明来自 blog.csdn.net/wtz1985        所谓"锁",就是为了让自己独自占有空间,在自己没用完之前,不让别人来占用自己的资源.现在的操作系统,无论是WINDOWS,还是UNIX等其他操作系统.都采用多线程的环境.这极大提高了任务的执行速度,而且不会影响其他事务的执行.但是它们的执行是靠时间片的轮转的,如果某一个线程没有执行完,可它的时间片用完了,就会被挂起来,直到轮到它的下个时间片.假如继续让它们这么自由的,没有约束的执行命令,将会导致一种不可预见

Linux多线程之同步

引言 条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作:一个线程等待条件变量的条件成立而挂起(此时不再占用cpu):另一个线程使条件成立(给出条件成立信号).为了防止竞争,条件变量的使用总是和一个互斥锁结合在一起. 函数原型 1. 定义条件变量 #include <pthread.h>/* 定义两个条件变量 */pthread_cond_t cond_pro, cond_con; 2. 初始化和销毁条件变量 #include <pthread.h>int pt

Java多线程基础总结八:ReentrantReadWriteLock

说到ReentrantReadWriteLock,首先要做的是与ReentrantLock划清界限.它和后者都是单独的实现,彼此之间没有继承或实现的关系. 然后就是总结这个锁机制的特性了: (a).重入方面其内部的WriteLock可以获取ReadLock,但是反过来ReadLock想要获得WriteLock则永远都不要想. (b).WriteLock可以降级为ReadLock,顺序是:先获得WriteLock再获得ReadLock,然后释放WriteLock,这时候线程将保持Readlock的

上下文切换与多线程实现的代价

多线程中的上下文切换 支持多任务处理是CPU设计史上最大的跨越之一.在计算机中,多任务处理是指同时运行两个或多个程序.从使用者的角度来看,这看起来并不复杂或者难以实现,但是它确实是计算机设计史上一次大的飞跃.在多任务处理系统中,CPU需要处理所有程序的操作,当用户来回切换它们时,需要记录这些程序执行到哪里.上下文切换就是这样一个过程,他允许CPU记录并恢复各种正在运行程序的状态,使它能够完成切换操作. 在上下文切换过程中,CPU会停止处理当前运行的程序,并保存当前程序运行的具体位置以便之后继续运

java多线程总结一:线程的两种创建方式及优劣比较

http://blog.csdn.net/touch_2011/article/details/6891026 1.通过实现Runnable接口线程创建 (1).定义一个类实现Runnable接口,重写接口中的run()方法.在run()方法中加入具体的任务代码或处理逻辑. (2).创建Runnable接口实现类的对象. (3).创建一个Thread类的对象,需要封装前面Runnable接口实现类的对象.(接口可以实现多继承) (4).调用Thread对象的start()方法,启动线程 示例代码

java单线程和多线程的区别

1.单线程和多线程的区别? 你早上上班,正要打卡的时候,手机响了..你如果先接了电话,等接完了,在打卡,就是单线程.如果你一手接电话,一手打卡.就是多线程.这两件事的结果是一样的..你接了电话且打了卡. (1)最常见的一个线程例子: package com.ggx.thread; public class OnlyThread{ public static void main(String[] args){ System.out.println("我就是一个线程"); } } 当程序启