JAVA多线程到底能不能加快程序运行速度?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JAVA多线程到底能不能加快程序运行速度?相关的知识,希望对你有一定的参考价值。

我写了一个文件转换的程序,大概实现功能是这样的:将doc,docx,xls,xlsx,ppt,pptx...等类型文件拖进某个文件夹A,程序自动将所有文件转换成txt格式文件,并存储到另一个文件夹B,然后删除文件夹A中文件。一次性拖入十几二十个文件的话,转换速度还是挺快的,基本上可以实现一拖进去,文件夹A就被清空了。但是如果是几百个文件的话,就需要一定的时间了。我测试了很多次,479个文件完成整个过程大概需要35秒,这太慢了,太让人难受了。然后我就用多线程,但是经测试,并没有加快速度,基本上有用多线程跟没用没什么两样,这让我怀疑多线程到底能不能加快程序运行速度?当然也有可能是我的多线程写错了。如果各位有多线程的代码,请贴出来参考参考,最好多线程的思想是:有两个运算模块,让多线程分配两个线程,每个线程负责运行一个运算模块,并且是同时运行

磁盘IO的速度在那里了,就算你再多的线程,也绕不过IO瓶颈。不是说多线程不能提高效率,这个要看你项目的性能瓶颈在哪里。 IO密集型,没必要多线程,容易弄巧成拙。建议Cache,某些文件系统在顺序读或写磁盘时速度相当快,如果恰好文件是顺序存储在磁盘上的,建议先尽量读进内存,再一次性写出去。其他什么磁盘内存通道之类的底层技术就不是Java能左右的了。 参考技术A 线程并不能提高速度,而是在执行某个耗时的功能时,在还可以做其它的事。多线程使你的程序在处理文件时不必显得已经卡死。
你可以试试这种方法,使用线程池,每个文件都是使用一个线程进行处理(我觉得你只是有了一个线程仍然进行的线性的处理。),最后使用消息handler 通知主线程工作完毕。

java的多线程:线程基础

1.线程与进程区别

每个正在系统上运行的程序都是一个进程。每个进程包含一到多个线程。线程是一组指令的集合,或者是程序的特殊段,它可以在程序里独立执行。也可以把它理解为代码运行的上下文。所以线程基本上是轻量级的进程,它负责在单个程序里执行多任务。通常由操作系统负责多个线程的调度和执行。

使用线程可以把占据时间长的程序中的任务放到后台去处理,程序的运行速度可能加快,在一些等待的任务实现上如用户输入、文件读写和网络收发数据等,线程就比较有用了。在这种情况下可以释放一些珍贵的资源如内存占用等等。

如果有大量的线程,会影响性能,因为操作系统需要在它们之间切换,更多的线程需要更多的内存空间,线程的中止需要考虑其对程序运行的影响。通常块模型数据是在多个线程间共享的,需要防止线程死锁情况的发生。

总结:进程是所有线程的集合,每一个线程是进程中的一条执行路径。

进程就是一个独立的应用程序比如 qq、wx

线程就是执行代码的最小单元,一条执行路径,比如迅雷同时下载多个电影,多线程下载,数据库连接池、分批发送短信、线程池

技术图片 

多线程也不能开的太多,太多太耗cpu,多线程的目的就是为了提高代码执行效率

二.多线程

1.线程是什么?线程是一条执行路径,每个线程互不影响

2.什么是多线程?多线程是在同一个进程中,有多条不同的执行路径,并行执行,为了提高程序效率

3.在一个进程中一定会有主线程

4.线程分类:

Java中有两种线程,一种是用户线程,另一种是守护线程。

用户线程是用户自定义创建的线程,主线程停止,用户线程不会停止

守护线程进程不存在或主线程停止,守护线程也停止

使用setDaemon(true)方法设置为守护线程三、同步与异步

1.同步概念,代码从上往下执行,http同步请求

2.异步概念 ,并行执行,互不影响,比如ajax

技术图片

四、实现多线程的方式

1.继承Thread,重写run方法

2.实现Runnable,实现run方法

3.匿名内部类,直接new Runnable() run

3.线程池

五、线程的状态

技术图片

 线程从创建、运行到结束总是处于下面五个状态之一:新建状态、就绪状态、运行状态、阻塞状态及死亡状态。

状态

   当用new操作符创建一个线程时, 例如new Thread(r),线程还没有开始运行,此时线程处在新建状态。 当一个线程处于新生状态时,程序还没有开始运行线程中的代码

就绪状态

一个新创建的线程并不自动开始运行,要执行线程,必须调用线程的start()方法。当线程对象调用start()方法即启动了线程,start()方法创建线程运行的系统资源,并调度线程运行run()方法。当start()方法返回后,线程就处于就绪状态。

     处于就绪状态的线程并不一定立即运行run()方法,线程还必须同其他线程竞争CPU时间,只有获得CPU时间才可以运行线程。因为在单CPU的计算机系统中,不可能同时运行多个线程,一个时刻仅有一个线程处于运行状态。因此此时可能有多个线程处于就绪状态。对多个处于就绪状态的线程是由Java运行时系统的线程调度程序(thread scheduler)来调度的。

运行状态

当线程获得CPU时间后,它才进入运行状态,真正开始执行run()方法.

阻塞状态

    线程运行过程中,可能由于各种原因进入阻塞状态:       

  1>线程通过调用sleep方法进入睡眠状态;       

  2>线程调用一个在I/O上被阻塞的操作,即该操作在输入输出操作完成之前不会返回到它的调用者;       

  3>线程试图得到一个锁,而该锁正被其他线程持有;         

  4>线程在等待某个触发条件;

死亡状态

有两个原因会导致线程死亡:   1) run方法正常退出而自然死亡,    2) 一个未捕获的异常终止了run方法而使线程猝死。   为了确定线程在当前是否存活着(就是要么是可运行的,要么是被阻塞了),需要使用isAlive方法。如果是可运行或被阻塞,这个方法返回true; 如果线程仍旧是new状态且不是可运行的, 或者线程死亡了,则返回false.

join()方法作用

如果说t1线程与t2线程都是用户线程,想让t1在t2执行完后再执行:就在t1的run方法内加入t2.join,A想加入到B的前面,就在B的方法里A.join

 

当在主线程当中执行到t1.join()方法时,就认为主线程应该把执行权让给t1

 

技术图片

需求:创建一个线程,子线程执行完毕后,主线程才能执行。

 

Thread t1 = new Thread(new Runnable()

 

@Override

public void run()

for (int i = 0; i < 10; i++)

try 

Thread.sleep(10);

catch (Exception e)

 

System.out.println(Thread.currentThread().getName() + "i:" + i);

);

t1.start();

// 当在主线程当中执行到t1.join()方法时,就认为主线程应该把执行权让给t1

t1.join();

for (int i = 0; i < 10; i++)

try 

Thread.sleep(10);

catch (Exception e)

 

System.out.println("main" + "i:" + i);

 

六.守护线程

 Java中有两种线程,一种是用户线程,另一种是守护线程。

 用户线程是用户自定义创建的线程,主线程停止,用户线程不会停止

守护线程进程不存在或主线程停止,守护线程也停止

 使用setDaemon(true)方法设置为守护线程

非守护线程特征:和主线程互不影响

技术图片

技术图片

技术图片

得出结论:用户创建线程,即使主线程结束,是不会结束的  ,

当 用户线程设置为守护线程 ,使用 :t1.setDaemon(true); 主线程结束,他也结束了

 

以上是关于JAVA多线程到底能不能加快程序运行速度?的主要内容,如果未能解决你的问题,请参考以下文章

Java线程

并发编程之多线程(Java)

Java多线程并发操作数据库能否提高运行速度。

java中啥叫做线程?啥叫多线程?多线程的特点是啥?

Java基础 - 多线程

Python多线程总结