自從java1.5以后,官網就推出了Executor這樣一個類,這個類,可以維護我們的大量線程在操作臨界資源時的穩(wěn)定性。
先上一段代碼吧:
創(chuàng)新互聯是一家集網站建設,紅橋企業(yè)網站建設,紅橋品牌網站建設,網站定制,紅橋網站建設報價,網絡營銷,網絡優(yōu)化,紅橋網站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強企業(yè)競爭力??沙浞譂M足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯網需求。同時我們時刻保持專業(yè)、時尚、前沿,時刻以成就客戶成長自我,堅持不斷學習、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實用型網站。
TestRunnable.java
public class TestRunnable implements Runnable { private String name; public TestRunnable(String name) { this.name = name; } @Override public void run() { while (true) { if (Main.Surplus < 0) return; Main.Surplus--; System.out.println(name + " " + Main.Surplus); } } }
main入口
public static void main(String[] args) { TestRunnable runnable = new TestRunnable("runnable1"); TestRunnable runnable2 = new TestRunnable("runnable2"); Thread t1 = new Thread(runnable); Thread t2 = new Thread(runnable2); t1.start(); t2.start(); }
這樣,我們就看到了,數據肯定是亂了的,當然這個時候我們可以加上一個synchronized的關鍵字,但是這樣也會出現點小問題的
下面我打算采用一種java內置的線程管理的機制,來解決這個問題,解決這個問題的思路大概就是,我們維護了一個線程池,當有請求操作的時候統(tǒng)統(tǒng)進入線程池,并且我們只開了一個線程,可以讓請求順序執(zhí)行,順序調用臨界資源,就很安全了。
import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Future; public class Main { public static int Surplus = 10; private ExecutorService executor = Executors.newSingleThreadExecutor(); void addTask(Runnable runnable) { executor.execute(runnable); }V addTask(Callable callable) { Future submit = executor.submit(callable); try { return submit.get(); } catch (InterruptedException e) { System.out.println("InterruptedException" + e.toString()); } catch (ExecutionException e) { System.out.println("ExecutionException" + e.toString()); } return null; } public void testAddTask(String name) { addTask(new Runnable() { @Override public void run() { for (int i = 0; i < 3; i++) { if (Main.Surplus <= 0) return; Main.Surplus--; System.out.println(name + " " + Main.Surplus); } } }); } public void testAddTask2(String name) { int count = addTask(new Callable () { @Override public Integer call() throws Exception { for (int i = 0; i < 3; i++) { if (Main.Surplus <= 0) return 0; Main.Surplus--; System.out.println(name + " " + Main.Surplus); } return Main.Surplus; } }); } public void close() { executor.shutdown(); } public static void main(String[] args) { Main main = new Main(); main.testAddTask("task1"); main.testAddTask2("task2"); main.testAddTask("task3"); main.testAddTask2("task4"); main.close(); } }
在這里,我們定義了兩種方法,分別是addTask,具有泛型的addTask,這兩種方法實現原理都是一樣的,其中一個是有回調的,一個是沒有回調的,就看項目需求了吧。
然后分別調用這兩個方法咯,就可以看到結果是非常有序,且不會混亂的。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持創(chuàng)新互聯。