认识线程和进程
讲解多线程之前,首先要把两个非常容易混淆的概念讲清楚,那就是进程和线程。关于进程核线程的差别在很多书籍和文章里都有涉及,但是我觉得在《Java核心技术》里面给的定义最为精确:每个进程都有自己一整套的变量,而线程则共享数据。
比如我们在电脑上可以同时打开QQ、浏览器、IDE等等,每次我们打开一个程序,操作系统就会为其创建一个进程,而这些进程之间的变量是相互独立的,通讯需要依赖接口间的相互调用;如果我们使用浏览器下载多个文件,那么每个下载动作就是一个单独的线程,这些线程都在浏览器这个程序内,都可以调用程序内的变量,但是线程之间又保持相互独立,我们可以单独选择暂停或者取消某个下载任务,其他的下载任务都不会受影响。
这就是线程和进程的差别,我们在学习多线程的时候,有一定要首先搞清楚这个。线程之间的变量共享使得线程间的通讯非常简单高效,但也会导致多线程并发执行任务时逻辑变得比较复杂。
在这个系列文章里,我会尝试在自己的知识框架内,把多线程这个相对高级一点的开发技能理清楚。
为什么使用多线程
性能要求
我们都知道现在已经是多核时代,如果我们开发的程序还是单线程程序,那么无疑是一个效率低下的程序,这不仅会造成cpu性能的浪费,也会使客户在使用我们的程序的时候增加很多无意义的等待时间。
使用场景要求
我们在进行实际开发任务时会发现程序有许多私下里进行的小动作,比如说需要后台同步数据、跟服务器通信或者进行一些耗时较长的操作,这时候我们就需要单独开一个线程来执行这些操作,以保证用户的正常使用不被影响。
创建一个线程
Java中提供了两种方式供我们创建一个线程,一个是继承Thread类重写run()方法,一个是实现Runnable接口重写run()方法。下面给出简单的代码示例。
继承Thread类创建一个线程
package multithreadTest;
/**
* Created by Vivi on 2017/10/6.
*/
public class ThreadTest extends Thread {
public void run() {
System.out.println("继承Thread类创建一个新线程");
}
}
实现Runnable接口创建一个线程
package multithreadTest;
/**
* Created by Vivi on 2017/10/6.
*/
public class RunnableTest implements Runnable{
public void run() {
System.out.println("继承runnable接口创建一个新线程");
}
}
测试类
package multithreadTest;
/**
* Created by Vivi on 2017/10/6.
*/
public class MultiTest {
public static void main(String args[]) {
ThreadTest threadTest = new ThreadTest();
threadTest.start();
RunnableTest runnableTest = new RunnableTest();
Thread runnableThread = new Thread(runnableTest);
runnableThread.start();
}
}
对应的控制台会输出这样的信息:
继承Thread类创建一个新线程
继承runnable接口创建一个新线程
注意事项
只有Thread类的start()才能新启一个线程并且执行相应的run()方法,假如直接调用run()方法,只是在当前线程执行了run()中的内容,
另外,现在比较推荐使用runnable接口来新建线程。
中断线程
中断线程是多线程中一个很重要的概念,大多数资料也讲的不是很清楚,这边详细说一下线程中断、中断线程和线程的中断状态。
线程中断
在理解java的中断线程之前,我们要先知道线程中断是什么意思。线程中断一般是指暂停当前的线程,然后调到中断入口,执行相应的中断处理,执行完之后再回到中断的地方继续执行代码。其实就是,停下来干点别的事儿,干完了别的事情再回来把自己的事情做完。这就是线程中断的意思。
线程的中断状态
在Java中,每个线程都有一个boolean属性来表示当前线程的中断状态,这个状态初始的时候是false,表示当然线程没有被中断。java提供了相应的方法来查询和改变这个状态,也提供了相应的机制来处理线程收到中断请求时的操作。
java中的中断线程
但是在java中,中断线程是一种协作机制,当一个线程对一个线程发出一个线程中断请求时(调用了interrupt()方法),被要求中断的线程不是立刻中断的,而是根据自身的情况或者程序员预先设定好的代码做出相应的反应。在java中,stop()方法(已经废弃)可以直接中断一个进程,interrupt()只是一个线程敲了另一个线程的门,问问它愿不愿意停下来而已。
java中的interrupt方法
在Thread类中,有三个有interrupt相关的方法,分别是:
方法 | 作用 |
---|---|
interrupt() | 请求中断一个线程并把线程的中断状态由false改为true,但并不是直接中断线程,执行interrupt()方法的线程可以自行决定是否中断 |
isInterrupted() | 读取当前线程的中断状态 |
interrupted() | 读取当前线程的中断状态并将中断状态重置为false |
线程中断的相关异常
在实际的操作中,我们会发现执行interrupt()方法的代码可能会抛出一个 InterruptedException 。这是一个线程在阻塞状态时调用了中断方法。当线程发现自己是在阻塞状态,并且中断状态是true,就会抛出这个异常。当这个异常被抛出的时候,我们可以获取到两个信息,一是:当前线程处在阻塞状态;二是:线程的中断状态恢复了false
如何处理InterruptedException是一个比较复杂的问题,后面有时间再单独写一篇来分析这个问题。
简单演示一下会抛出InterruptedException的代码
Thread类:
package multithreadTest;
/**
* Created by Vivi on 2017/10/6.
*/
public class ThreadTest extends Thread {
public void run() {
try {
System.out.println("继承Thread类创建一个新线程");
Thread.sleep(10000l);
}catch (InterruptedException e){
System.out.println(Thread.currentThread().isInterrupted());
System.out.println("这里抛出了一个InterruptedException,但是我并没有做任何处理");
e.printStackTrace();
}
}
}
测试入口:
package multithreadTest;
/**
* Created by Vivi on 2017/10/6.
*/
public class MultiTest {
public static void main(String args[]) {
ThreadTest threadTest = new ThreadTest();
threadTest.start();
threadTest.interrupt();
}
}
控制台会输出这样的信息:
继承Thread类创建一个新线程
false
这里抛出了一个InterruptedException,但是我并没有做任何处理
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at multithreadTest.ThreadTest.run(ThreadTest.java:11)
这一篇文章写到这里,下一篇介绍线程的状态。
从零学习JAVA多线程(二):线程的状态和属性
**附上版权声明:
原作者:Vi_error,博客地址:Vi_error.nextval
转载请保持署名和注明原地址**____