詳解Java利用ExecutorService實(shí)現(xiàn)同步執(zhí)行大量線程
自從java1.5以后,官網(wǎng)就推出了Executor這樣一個類,這個類,可以維護(hù)我們的大量線程在操作臨界資源時的穩(wěn)定性。
先上一段代碼吧:
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();
}

這樣,我們就看到了,數(shù)據(jù)肯定是亂了的,當(dāng)然這個時候我們可以加上一個synchronized的關(guān)鍵字,但是這樣也會出現(xiàn)點(diǎn)小問題的

下面我打算采用一種java內(nèi)置的線程管理的機(jī)制,來解決這個問題,解決這個問題的思路大概就是,我們維護(hù)了一個線程池,當(dāng)有請求操作的時候統(tǒng)統(tǒng)進(jìn)入線程池,并且我們只開了一個線程,可以讓請求順序執(zhí)行,順序調(diào)用臨界資源,就很安全了。
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> V addTask(Callable<V> callable) {
Future<V> 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<Integer>() {
@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,這兩種方法實(shí)現(xiàn)原理都是一樣的,其中一個是有回調(diào)的,一個是沒有回調(diào)的,就看項(xiàng)目需求了吧。

然后分別調(diào)用這兩個方法咯,就可以看到結(jié)果是非常有序,且不會混亂的。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java代碼實(shí)現(xiàn)對properties文件有序的讀寫的示例
本篇文章主要介紹了Java代碼實(shí)現(xiàn)對properties文件有序的讀寫的示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-11-11
Springboot詳解RocketMQ實(shí)現(xiàn)廣播消息流程
RocketMQ作為一款純java、分布式、隊(duì)列模型的開源消息中間件,支持事務(wù)消息、順序消息、批量消息、定時消息、消息回溯等,本篇我們了解如何實(shí)現(xiàn)廣播消息2022-06-06
基于Java SSM實(shí)現(xiàn)Excel數(shù)據(jù)批量導(dǎo)入
這篇文章主要為大家詳細(xì)介紹了基于Java SSM如何實(shí)現(xiàn)excel數(shù)據(jù)批量導(dǎo)入,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11
springboot中使用@Transactional注解事物不生效的坑
這篇文章主要介紹了springboot中使用@Transactional注解事物不生效的原因,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01

