1、什麼是線程?java
線程:程序中一個單一的順序控制流程。進程內一個相對獨立的、可調度的執行單元,是系統獨立調度和分派CPU的基本單位。編程
多線程:單個程序中同時運行多個線程完成不一樣的工做,稱爲多線程。微信
特色:多線程
1)輕量級的進程,程序運行流中可執行的最小單元,線程不擁有系統資源,多個線程共享進程擁有的資源。併發
2)一個線程能夠建立另一個線程,多個線程能夠併發執行。ide
3)多個線程在系統運行中搶佔資源,會出現間斷性,咱們看到的是並行執行,其實在有前後順序的。spa
4)一個進程至少包含一個線程,即主線程。線程
2、線程有哪些狀態?對象
線程具備:新建,就緒,運行,阻塞,終止五種狀態。繼承
①新建:線程被建立,沒有執行任何方法,如,Thread th = new Thread()。
②就緒:當調用線程的start方法時,就會觸發線程狀態變動爲就緒態,等待cpu來調用。處於就緒態的線程纔會被cpu調度,單cpu不是當即執行它。
③運行:當cpu發起對此線程調用時,它就進入了運行態。
④阻塞:當線程因爲某種緣由,再也不擁有cpu使用權,它就會被阻塞。
阻塞有如下幾種狀況:
1)sleep(long mills):參數爲毫秒數,使線程在指定的時間內進入阻塞,時間一過,進入就緒態。
2)suspend() 和 resume():suspend會讓線程掛起,必須執行resume進行線程的恢復。
3)yield():與sleep()相似,可是不能由用戶指定暫停多長時間,只能出讓機會給同優先級的線程,且不進入阻塞。如同排隊,前面的人和後面的人交換位置,可是還處在隊伍中。
4)wait() 和 notify():wait() 使線程進入阻塞狀態,有兩種形式,一種指定毫秒數,另外一種無參。前者可經過notify()喚起或者超過指定時間自動恢復;後者必須經過notify()喚起。
5)同步阻塞:等待同步鎖資源。多線程競爭同一個資源時,只能一個線程得到鎖,其它的線程要等待。
⑤終止:線程執行完畢,或者出現異常,線程結束。
3、如何建立線程?
java線程的實現方式有三種:繼承Thread類、實現Runnable接口,使用Callable和FutureTask(能夠有返回值)
一、經過集成Thread類,覆寫run()方法
class MyThread extends Thread {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
System.out.println(Thread.currentThread().getName() + " " + i);
}
}
}
public class ThreadDemo {
public static void main(String[] args) {
System.out.println(Thread.currentThread().getName());
MyThread th1 = new MyThread();
MyThread th2 = new MyThread();
th1.start();
th2.start();
}
}
輸出:main
Thread-1 0
Thread-0 0
Thread-1 1
Thread-1 2
......
線程要實現的邏輯寫在run方法中,經過執行線程的start()方法,使線程進入就緒狀態,等待CPU分配資源。
能夠看到兩個線程並行執行,且隨機得到CPU。
二、試過實現Runnable接口,實現run()方法
class MyThread implements Runnable {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
System.out.println(Thread.currentThread().getName() + " " + i);
}
}
}
public class ThreadDemo {
public static void main(String[] args) {
System.out.println(Thread.currentThread().getName());
MyThread th = new MyThread();
Thread t1 = new Thread(th);
Thread t2 = new Thread(th);
t1.start();
t2.start();
}
}
輸出:main
Thread-0 0
Thread-0 1
Thread-1 0
Thread-0 2
......
經過將MyThread實例傳入Thread構造方法實例化Thread,調用Thread的start方法,啓動線程。
ps:繼承Thread和實現Runnable接口有什麼區別呢?
1:前者爲單繼承,有侷限性,但接口的方式能夠實現多個。
2:後者能夠實現資源共享。
多線程編程中,強烈建議使用Runnable
三、使用Callable和Future接口建立線程。
具體是建立Callable接口的實現類,並實現clall()方法。
並使用FutureTask類來包裝Callable實現類的對象,且以此FutureTask對象做爲Thread對象的target來建立線程。
class MyCallable implements Callable<Integer> {
@Override
public Integer call() throws Exception {
return 1;
}
}
public class ThreadDemo {
public static void main(String[] args) {
Callable<Integer> myCallable = new MyCallable();//實例化MyCallable
FutureTask<Integer> ft = new FutureTask<>(myCallable);//通FutureTask包裝
Thread thread = new Thread(ft);//將FutureTask傳入Thread構造,實例化線程
thread.start();//線程啓動
Integer result = ft.get();//獲取返回值
System.out.println(result);
}
}
1)實現Callable接口中的call()方法,這個是線程要執行的邏輯。
2)FutureTask的get()方法會一直阻塞,直到call()方法執行完畢取到返回值。
關注老薑談技術,微信號:helojava,或者掃描下面二維碼。
每日一帖,技術雞湯。