一、线程与进程的区别
1.线程其实就是一条执行路径,进程是独立应用程序的一个应用程序。
2.线程是一条执行路径,多条执行路径同时执行,在进程中,会有N多个线程,进程是所有线程的集合。
3.使用多线程目的就是为了提高程序效率。
二、创建多线程的方式(以下三种方法不一一举例)
1.第一种是继承Thread类,重写run方法。
2.实现Runnable接口,重写run方法。
3.使用匿名内部类的方式。
三、守护线程
1.java中有两种线程,一种是用户线程,另一种是守护线程。
1.1.用户线程是指用户自定义创建的线程,主线程停止,用户线程不会停止。
1.2.守护线程当进程不存在或主线程停止,守护线程也会被停止。
2.使用setDaemon(true)方法设置为守护线程
3.代码举例
public class DaemonThread {
public static void main(String[] args) {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
while (true) {
try {
Thread.sleep(100);
} catch (Exception e) {
// TODO: handle exception
}
System.out.println("我是子线程...");
}
}
});
thread.setDaemon(true);
thread.start();
for (int i = 0; i < 3; i++) {
try {
Thread.sleep(100);
} catch (Exception e) {
}
System.out.println("我是主线程");
}
System.out.println("主线程执行完毕!");
}
}
4.结果
4.1.设置thread.setDaemon(true),结果如下:
我是子线程...
我是主线程
我是子线程...
我是主线程
我是子线程...
我是主线程
主线程执行完毕!
4.2.不设置thread.setDaemon(true),结果如下(我是子线程...(一直循环)):
我是子线程...
我是主线程
我是子线程...
我是主线程
我是子线程...
我是主线程
主线程执行完毕!
我是子线程...
我是子线程...
我是子线程...
四、多线程运行状态
1.图片解析
线程从创建、运行到结束总是处于下面五个状态之一:新建状态、就绪状态、运行状态、阻塞状态及死亡状态。
2.各个状态解析
2.1.新建状态:当用new操作符创建一个线程时, 例如new Thread(r),线程还没有开始运行,此时线程处在新建状态。 当一个线程处于新生状态时,程序还没有开始运行线程中的代码
2.2.就绪状态:一个新创建的线程并不自动开始运行,要执行线程,必须调用线程的start()方法。当线程对象调用start()方法即启动了线程,start()方法创建线程运行的系统资源,并调度线程运行run()方法。当start()方法返回后,线程就处于就绪状态。处于就绪状态的线程并不一定立即运行run()方法,线程还必须同其他线程竞争CPU时间,只有获得CPU时间才可以运行线程。因为在单CPU的计算机系统中,不可能同时运行多个线程,一个时刻仅有一个线程处于运行状态。因此此时可能有多个线程处于就绪状态。对多个处于就绪状态的线程是由Java运行时系统的线程调度程序(thread scheduler)来调度的。
2.3.运行状态:当线程获得CPU时间后,它才进入运行状态,真正开始执行run()方法.
2.4.阻塞状态:线程运行过程中,可能由于各种原因进入阻塞状态:
1>线程通过调用sleep方法进入睡眠状态;
2>线程调用一个在I/O上被阻塞的操作,即该操作在输入输出操作完成之前不会返回到它的调用者;
3>线程试图得到一个锁,而该锁正被其他线程持有;
4>线程在等待某个触发条件;
2.5.死亡状态:有两个原因会导致线程死亡:
1) run方法正常退出而自然死亡。
2) 一个未捕获的异常终止了run方法而使线程猝死。
为了确定线程在当前是否存活着(就是要么是可运行的,要么是被阻塞了),需要使用isAlive方法。如果是可运行或被阻塞,这个方法返回true; 如果线程仍旧是new状态且不是可运行的, 或者线程死亡了,则返回false.
五、join()方法
1.例如:当在主线程当中执行到t1.join()方法时,就认为主线程应该把执行权让给t1。
2.代码案例:
public class DaemonThread {
public static void main(String[] args) throws Exception {
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);
}
}
}
3.结果
Thread-0i:0
Thread-0i:1
Thread-0i:2
Thread-0i:3
Thread-0i:4
Thread-0i:5
Thread-0i:6
Thread-0i:7
Thread-0i:8
Thread-0i:9
maini:0
maini:1
maini:2
maini:3
maini:4
maini:5
maini:6
maini:7
maini:8
maini:9
4.分析:先把子线程执行完毕,主线程让步,执行完子线程之后再执行主线程。
六、小福利
1.进程与线程的区别?
答:进程是所有线程的集合,每一个线程是进程中的一条执行路径,线程只是一条执行路径。
2.为什么要用多线程?
答:提高程序效率
3.是继承Thread类好还是实现Runnable接口好?
答:Runnable接口好,因为实现了接口还可以继续继承。继承Thread类不能再继承。
七、结束
1.下片文章讲解多线程安全问题以及解决方法。
2.坚持就是胜利!!!哈哈哈哈~~~