为什么要使用多线程(并发编程)

发布于:2024-06-29 ⋅ 阅读:(17) ⋅ 点赞:(0)

目录

1.上下文的切换

1.1 什么是上下文切换

2. 并发编程的死锁问题

2.1 死锁产生的原因

2.2 避免死锁的方法

3.资源限制的挑战3.1 什么是资源限制


    

并发编程的目的是为了让程序更快,大家都知道并不是开启的线程越多越快,因为开启的线程越多随即面临的挑战也越多,比如:上下文的切换、死锁问题、以及硬件和软件资源的限制问题。

1.上下文的切换

1.1 什么是上下文切换

        学习操作系统的时候,都了解到:单核处理器也支持多线程执行代码,CPU通过给每个线程分配CPU时间片来实现这个机制。时间片是CPU分配给各个线程的时间,因为时间片非常短,所以CPU通过不停地切换线程执行,让我们感觉多个线程是同时执行的,时间片一般是几十毫秒(ms)。

         CPU通过时间片分配算法来循环执行任务,当前任务执行一个时间片后会切换到下一个任务。但是,在切换前会保存上一个任务的状态,以便下次切换回这个任务时,可以再加载这个任务的状态。所以任务从保存到再加载的过程就是一次上下文切换。

1.2   如何减少上下文切换

    减少上下文切换的方法有无锁并发编程、CAS算法、使用最少线程和使用协程。

1. 无锁并发编程。多线程竞争锁时,会引起上下文切换,所以多线程处理数据时,可以用一些办法来避免使用锁,如将数据的ID按照Hash算法取模分段,不同的线程处理不同段的数据。


2.CAS算法。Java的 Atomic包使用CAS算法来更新数据,而不需要加锁。

3.使用最少线程。避免创建不需要的线程,比如任务很少,但是创建了很多线程来处理,这样会造成大量线程都处于等待状态。

4.协程:在单线程里实现多任务的调度,并在单线程里维持多个任务间的切换

2. 并发编程的死锁问题

2.1 死锁产生的原因
  • 资源竞争: 多个线程需要使用相同的资源,并且这些资源只能被一个线程一次性占用。如果一个线程获取了资源A,另一个线程获取了资源B,并且两个线程都在等待对方的资源,就会发生死锁。

  • 资源请求和保持: 一个线程已经持有一个资源,并且在等待获取另一个资源时,不释放已经持有的资源。

  • 资源不可抢占: 资源不能被强制从一个线程中抢占。只有线程自己才可以主动释放已持有的资源。

  • 循环等待: 存在一个线程循环等待资源的情况,例如线程A等待线程B持有的资源,而线程B又等待线程A持有的资源。

2.2 避免死锁的方法
  1. 避免嵌套锁(一个线程获取多个锁): 尽量避免一个线程在持有一个锁的同时再去请求另一个锁。可以通过减少锁的使用来避免死锁。

  2. 使用超时: 在获取锁的时候设置超时,如果超过一定时间没有获取到锁,就放弃请求该锁。这样可以避免线程无限期地等待下去。

    if (lock.tryLock(10, TimeUnit.SECONDS)) {
        try {
            // critical section
        } finally {
            lock.unlock();
        }
    } else {
        // perform alternative action
    }
    

    3.尽量缩短锁的持有时间:尽量缩短持有锁的时间,只在必要的地方加锁,减少在锁内执行的操作

3.资源限制的挑战

3.1 什么是资源限制

        资源限制是指在进行并发编程时,程序的执行速度受限于计算机硬件资源或软件资源例如,服务器的带宽只有 2Mbs,某个资源的下载速度是1Mb/s每秒,系统启动10个线程下载资源,下载速度不会变成10Mb/s,所以在进行并发编程时,要考虑这些资源的限制。硬件资源限制有带宽的上传/下载速度、硬盘读写速度和CPU的处理速度。软件资源限制有数据库的连接数和socket 连接数等。

3.2 资源限制引发的问题

       在并发编程中,将代码执行速度加快的原则是将代码中串行执行的部分变成并发执行但是如果将某段串行的代码并发执行,因为受限于资源,仍然在串行执行,这时候程序不仅不会加快执行,反而会更慢,因为增加了上下文切换和资源调度的时间。例如,之前看到-段程序使用多线程在办公网并发地下载和处理数据时,导致CPU利用率达到100%,几个小时都不能运行完成任务,后来修改成单线程,一个小时就执行完成了。

        强烈建议:如果并发程序写的不严谨,出现问题定位起来比较耗时和棘手java开发工程师尽量还是多使用JDK并发包提供的并发容器和工具类来解决并发问题。(比较jdk大叔的水平可是经历 了历史的检验)