千锋教育-做有情怀、有良心、有品质的职业教育机构

当前位置:首页  >  IT面试题  >  物联网面试题  >  正文

多线程开发带来的问题与解决方法?

来源:千锋教育
发布时间:2022-09-26 14:09:00
分享

  使用多线程主要会带来以下几个问题:

  一、线程安全问题

  线程安全问题指的是在某一线程从开始访问到结束访问某一数据期间,该数据被其他的线程所修改,那么对于当前线程而言,该线程就发生了线程安全问题,表现形式为数据的缺失,数据不一致等。

  线程安全问题发生的条件:

  1)多线程环境下,即存在包括自己在内存在有多个线程。

  2)多线程环境下存在共享资源,且多线程操作该共享资源。

  3)多个线程必须对该共享资源有非原子性操作。

  线程安全问题的解决思路:

  1)尽量不使用共享变量,将不必要的共享变量变成局部变量来使用。

  2)使用synchronized关键字同步代码块,或者使用jdk包中提供的Lock为操作进行加锁。

  3)使用ThreadLocal为每一个线程建立一个变量的副本,各个线程间独立操作,互不影响。

多线程开发带来的问题与解决方法?

  二、性能问题

  线程的生命周期开销是非常大的,一个线程的创建到销毁都会占用大量的内存。同时如果不合理的创建了多个线程,cup的处理器数量小于了线程数量,那么将会有很多的线程被闲置,闲置的线程将会占用大量的内存,为垃圾回收带来很大压力,同时cup在分配线程时还会消耗其性能。

  解决思路:

  利用线程池,模拟一个池,预先创建有限合理个数的线程放入池中,当需要执行任务时从池中取出空闲的先去执行任务,执行完成后将线程归还到池中,这样就减少了线程的频繁创建和销毁,节省内存开销和减小了垃圾回收的压力。同时因为任务到来时本身线程已经存在,减少了创建线程时间,提高了执行效率,而且合理的创建线程池数量还会使各个线程都处于忙碌状态,提高任务执行效率,线程池还提供了拒绝策略,当任务数量到达某一临界区时,线程池将拒绝任务的进入,保持现有任务的顺利执行,减少池的压力。

  三、活跃性问题

  1)死锁

  假如线程 A 持有资源 2,线程 B 持有资源 1,他们同时都想申请对方的资源,所以这两个线程就会互相等待而进入死锁状态。多个线程环形占用资源也是一样的会产生死锁问题。

  解决方法:

  - 避免一个线程同时获取多个锁

  - 避免一个线程在锁内同时占用多个资源,尽量保证每个锁只占用一个资源。

  - 尝试使用定时锁,使用 lock.tryLock(timeout) 来代替使用内部锁机制。 想要避免死锁,可以使用无锁函数(cas)或者使用重入锁(ReentrantLock),通过重入锁使线程中断或限时等待可以有效的规避死锁问题。

  2)饥饿

  饥饿指的是某一线程或多个线程因为某些原因一直获取不到资源,导致程序一直无法执行。如某一线程优先级太低导致一直分配不到资源,或者是某一线程一直占着某种资源不放,导致该线程无法执行等。

  解决方法:

  与死锁相比,饥饿现象还是有可能在一段时间之后恢复执行的。可以设置合适的线程优先级来尽量避免饥饿的产生。

  3)活锁

  活锁体现了一种谦让的美德,每个线程都想把资源让给对方,但是由于机器“智商”不够,可能会产生一直将资源让来让去,导致资源在两个线程间跳动而无法使某一线程真正的到资源并执行,这就是活锁的问题。

  四、阻塞

  阻塞是用来形容多线程的问题,几个线程之间共享临界区资源,那么当一个线程占用了临界区资源后,所有需要使用该资源的线程都需要进入该临界区等待,等待会导致线程挂起,一直不能工作,这种情况就是阻塞,如果某一线程一直都不释放资源,将会导致其他所有等待在这个临界区的线程都不能工作。

  当我们使用synchronized或重入锁时,我们得到的就是阻塞线程,如论是synchronized或者重入锁,都会在试图执行代码前,得到临界区的锁,如果得不到锁,线程将会被挂起等待,知道其他线程执行完成并释放锁且拿到锁为止。

  解决方法:

  可以通过减少锁持有时间,读写锁分离,减小锁的粒度,锁分离,锁粗化等方式来优化锁的性能。

  临界区: 临界区是用来表示一种公共的资源(共享数据),它可以被多个线程使用,但是在每次只能有一个线程能够使用它,当临界区资源正在被一个线程使用时,其他的线程就只能等待当前线程执行完之后才能使用该临界区资源。

  比如办公室办公室里有一支笔,它一次只能被一个人使用,假如它正在被甲使用时,其他想要使用这支笔的人只能等甲使用完这支笔之后才能允许另一个人去使用。这就是临界区的概念。

声明:本站稿件版权均属千锋教育所有,未经许可不得擅自转载。

相关推荐

  • 物联网中的边缘计算是什么?请解释其优势和应用场景 答案:边缘计算是一种将计算和数据处理能力移动到物联网设备附近的计算模型。在边缘计算中,数据的处理和分析发生在接近数据源的设备或边缘节点上,而不是传输到远程云端进行处理。边缘计算的优势包括:低延迟:通过
  • 什么是物联网(IoT)?请解释其核心原理和应用领域 答案:物联网是指通过互联网连接和交互的物理设备网络。它基于传感器、嵌入式系统、无线通信技术等,将各种物理设备连接起来,实现设备之间的数据交换和智能化控制。物联网的核心原理包括:感知和采集:通过传感器和
  • 谈谈网络的分层结构 1. 物理层:不是指具体的物理设备 指的是物理设备的标准制定 (网线 光纤的接口类型 网卡的电流强弱)比特流 2. 数据链路层:负责完整的帧数据收发 (帧数据可以独立在网络传输的数据) mac地址封装和解封装,交换器就工作在这一层
  • git和GitHub关系 它是一个用于 Linux内核开发的版本管理工具,是可以在你电脑不联网的情况下,只在本地使用的一个版本管理工具,其作用就是可以让你更好的管理你的程序,比如你原来提交过的内容,以后虽然修改了,但是通过git这个工具,可以把你原来提交的内容重现出来,这样对于你后来才意识到的一些错误的更改,可以进行还原。
  • 多线程开发带来的问题与解决方法? 线程安全问题指的是在某一线程从开始访问到结束访问某一数据期间,该数据被其他的线程所修改,那么对于当前线程而言,该线程就发生了线程安全问题,表现形式为数据的缺失,数据不一致等。
  • sleep()方法和wait()方法区别和共同点? - sleep()方法属于Thread类的静态方法,作用于当前线程;而wait()方法是Object类的实例方法,作用于对象本身。 - 执行sleep()方法后,可以通过超时或者调用interrupt()方法唤醒休眠中的线程;执行wait()方法后,通过调用notify()或notifyAll()方法唤醒等待线程。