Wie SpringBoot asynchrone Aufrufe elegant implementiert
Vorwort
Synchronische Programmierung: Bei der synchronen Programmierung werden Aufgaben einzeln ausgeführt, und erst wenn eine Aufgabe abgeschlossen ist, wird die Blockierung der nächsten Aufgabe aufgehoben.
Asynchrone Programmierung: Bei der asynchronen Programmierung können mehrere Aufgaben gleichzeitig ausgeführt werden. Sie können zu einer anderen Aufgabe wechseln, bevor die vorherige Aufgabe abgeschlossen ist.
In Spring Boot
können wir die Annotation @Async
verwenden, um asynchrones Verhalten zu implementieren. Spring Boot
中,我们可以使用@Async
注解来实现异步行为。
实现步骤
1.定义一个异步服务接口AsyncService.java
public interface AsyncService { void asyncMethod() throws InterruptedException; Future<String> futureMethod() throws InterruptedException; }
2.实现定义的接口AsyncServiceImpl.java
@Service @Slf4j public class AsyncServiceImpl implements AsyncService { @Async @Override public void asyncMethod() throws InterruptedException { Thread.sleep(3000); log.info("Thread: [{}], Calling other service..", Thread.currentThread().getName()); } @Async @Override public Future<String> futureMethod() throws InterruptedException { Thread.sleep(5000); log.info("Thread: [{}], Calling other service..", Thread.currentThread().getName()); return new AsyncResult<>("task Done"); } }
AsyncServiceImpl
是一个spring
管理的bean
。您的异步方法必须是公共的,而且是被
@Async
注解修饰。返回类型被限制为
void
或Future
。
3.定义一个控制器AsyncController.java
@EnableAsync @RestController @Slf4j public class AsyncController { @Autowired AsyncService asyncService; @GetMapping("/async") public String asyncCallerMethod() throws InterruptedException { long start = System.currentTimeMillis(); log.info("call async method, thread name: [{}]", Thread.currentThread().getName()); asyncService.asyncMethod(); String response = "task completes in :" + (System.currentTimeMillis() - start) + "milliseconds"; return response; } @GetMapping("/asyncFuture") public String asyncFuture() throws InterruptedException, ExecutionException { long start = System.currentTimeMillis(); log.info("call async method, thread name: [{}]", Thread.currentThread().getName()); Future<String> future = asyncService.futureMethod(); // 阻塞获取结果 String taskResult = future.get(); String response = taskResult + "task completes in :" + (System.currentTimeMillis() - start) + "milliseconds"; return response; } }
关键点,需要添加启用异步的注解
@EnableAsync
,当然这个注解加在其他地方也ok得。当外部调用该接口时,
asyncMethod()
将由默认任务执行程序创建的另一个线程执行,主线程不需要等待完成异步方法执行。
4.运行一下
现在我们运行一下看看,是不是异步返回的。
可以看到调用/async
接口,最终一步调用了方法。
调用/asyncFuture
,发现返回5秒多,难道不是异步的吗?其实也是异步的,看日志可以看出来,只不过我们返回的是Future
,调用Futrue.get()
是阻塞的。
自定义异步任务执行器和异常处理
我们现在看看如果异常方法中报错了会怎么样?修改异步代码如下所示,会抛运行时异常:
再次执行异步接口,如下所示,会使用默认的线程池和异常处理。
我们也可以自定义异步方法的处理异常和异步任务执行器,我们需要配置 AsyncUncaughtExceptionHandler
,如下代码所示:
@Configuration public class AsynConfiguration extends AsyncConfigurerSupport { @Override public Executor getAsyncExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(3); executor.setMaxPoolSize(4); executor.setThreadNamePrefix("asyn-task-thread-"); executor.setWaitForTasksToCompleteOnShutdown(true); executor.initialize(); return executor; } @Override public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() { return new AsyncUncaughtExceptionHandler() { @Override public void handleUncaughtException(Throwable ex, Method method, Object... params) { System.out.println("Exception: " + ex.getMessage()); System.out.println("Method Name: " + method.getName()); ex.printStackTrace(); } }; } }
再次运行,得到的结果如下:
@Async如何工作的
必须通过使用 @EnableAsync
注解注解主应用程序类或任何直接或间接异步方法调用程序类来启用异步支持。主要通过代理模式实现,默认模式是 Proxy
,另一种是 AspectJ
。代理模式只允许通过代理拦截调用。永远不要从定义它的同一个类调用异步方法,它不会起作用。
当使用 @Async
对方法进行注解时,它会根据“proxyTargetClass
”属性为该对象创建一个代理。当 spring
执行这个方法时,默认情况下它会搜索关联的线程池定义。上下文中唯一的 spring
框架 TaskExecutor bean
或名为“taskExecutor
”的 Executor bean
。如果这两者都不可解析,默认会使用spring框架SimpleAsyncTaskExecutor
AsyncService.java
🎜rrreee🎜2. Implementieren Sie die definierte Schnittstelle AsyncServiceImpl.java
🎜🎜🎜 AsyncServiceImpl
ist eine Bean
, die von spring
verwaltet wird. 🎜🎜🎜🎜Ihre asynchrone Methode muss öffentlich sein und mit der Annotation @Async
versehen sein. 🎜🎜🎜🎜Der Rückgabetyp ist auf void
oder Future
beschränkt. 🎜🎜🎜🎜3. Definieren Sie einen Controller AsyncController.java
🎜rrreee🎜🎜🎜Der entscheidende Punkt ist das Hinzufügen der Annotation, um asynchrones @EnableAsync
zu aktivieren wird zu anderen hinzugefügt Der Ort ist ok. 🎜🎜🎜🎜Wenn diese Schnittstelle extern aufgerufen wird, wird asyncMethod()
von einem anderen Thread ausgeführt, der vom Standard-Task-Executor erstellt wurde, und der Hauptthread muss nicht auf den Abschluss der asynchronen Methode warten Ausführung. 🎜🎜🎜🎜4. Führen Sie es aus🎜🎜Jetzt lassen Sie es uns ausführen und sehen, ob es asynchron zurückkehrt. 🎜🎜

/async Schnittstelle, der letzte Schritt ruft die Methode auf. 🎜🎜<img src="/static/imghw/default1.png" data-src="https://img.php.cn/upload/article/000/887/227/168390024926989.png" class="lazy" alt="Wie SpringBoot asynchrone Aufrufe elegant implementiert">🎜🎜<img src="/static/imghw/default1.png" data-src="https://img.php.cn/upload/article/000/887/227/168390024965912.png" class="lazy" alt="Wie SpringBoot asynchrone Aufrufe elegant implementiert">🎜🎜Call <code>/asyncFuture
, und festgestellt, dass die Rückgabe länger als 5 Sekunden dauerte. Ist es nicht asynchron? Tatsächlich ist es auch asynchron, wie Sie den Protokollen entnehmen können, aber was wir zurückgeben, ist Future
und der Aufruf von Futrue.get()
wird blockiert. 🎜🎜Angepasster asynchroner Task-Executor und Ausnahmebehandlung🎜🎜Sehen wir uns nun an, was passiert, wenn in der Ausnahmemethode ein Fehler gemeldet wird? Ändern Sie den asynchronen Code wie folgt. Es wird eine Laufzeitausnahme ausgelöst: 🎜🎜

AsyncUncaughtExceptionHandler
konfigurieren, wie im folgenden Code gezeigt: 🎜rrreee🎜Erneut ausführen, die Ergebnisse sind wie folgt: 🎜🎜
@ Die EnableAsync
kommentieren Sie die Hauptanwendungsklasse oder eine beliebige direkte oder indirekte asynchrone Methodenaufruferklasse, um die asynchrone Unterstützung zu aktivieren. Es wird hauptsächlich über den Proxy-Modus implementiert. Der Standardmodus ist Proxy
und der andere ist AspectJ
. Im Proxy-Modus können Anrufe nur über einen Proxy abgefangen werden. Rufen Sie niemals eine asynchrone Methode aus derselben Klasse auf, in der sie definiert ist, da sie sonst nicht funktioniert. 🎜🎜Wenn Sie eine Methode mit @Async
annotieren, wird ein Proxy für das Objekt basierend auf dem Attribut „proxyTargetClass
“ erstellt. Wenn spring
diese Methode ausführt, sucht es standardmäßig nach der zugehörigen Thread-Pool-Definition. Das einzige Spring
Framework TaskExecutor Bean
oder Executor Bean
mit dem Namen „taskExecutor
“ im Kontext. Wenn keines davon auflösbar ist, wird standardmäßig das Spring-Framework SimpleAsyncTaskExecutor
verwendet, um die Ausführung asynchroner Methoden zu verarbeiten. 🎜Das obige ist der detaillierte Inhalt vonWie SpringBoot asynchrone Aufrufe elegant implementiert. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Heiße KI -Werkzeuge

Undresser.AI Undress
KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover
Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool
Ausziehbilder kostenlos

Clothoff.io
KI-Kleiderentferner

AI Hentai Generator
Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

Heiße Werkzeuge

Notepad++7.3.1
Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version
Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1
Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6
Visuelle Webentwicklungstools

SublimeText3 Mac-Version
Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Heiße Themen



Einführung in Jasypt Jasypt ist eine Java-Bibliothek, die es einem Entwickler ermöglicht, seinem Projekt mit minimalem Aufwand grundlegende Verschlüsselungsfunktionen hinzuzufügen und kein tiefes Verständnis der Funktionsweise der Verschlüsselung erfordert. standardbasierte Verschlüsselungstechnologie. Passwörter, Text, Zahlen, Binärdateien verschlüsseln ... Geeignet für die Integration in Spring-basierte Anwendungen, offene API, zur Verwendung mit jedem JCE-Anbieter ... Fügen Sie die folgende Abhängigkeit hinzu: com.github.ulisesbocchiojasypt-spring-boot-starter2 Die Vorteile von Jasypt schützen unsere Systemsicherheit. Selbst wenn der Code durchgesickert ist, kann die Datenquelle garantiert werden.

Nutzungsszenario 1. Die Bestellung wurde erfolgreich aufgegeben, die Zahlung erfolgte jedoch nicht innerhalb von 30 Minuten. Die Zahlung ist abgelaufen und die Bestellung wurde automatisch storniert. 2. Die Bestellung wurde unterzeichnet und es wurde 7 Tage lang keine Bewertung durchgeführt. Wenn die Bestellung abläuft und nicht ausgewertet wird, wird die Bestellung standardmäßig positiv bewertet. Wenn der Händler die Bestellung innerhalb von 5 Minuten nicht erhält, wird die Bestellung abgebrochen Es wird eine SMS-Erinnerung gesendet ... Für Szenarien mit langen Verzögerungen und geringer Echtzeitleistung können wir die Aufgabenplanung verwenden, um eine regelmäßige Abfrageverarbeitung durchzuführen. Zum Beispiel: xxl-job Heute werden wir auswählen

1. Redis implementiert das Prinzip der verteilten Sperren und warum verteilte Sperren erforderlich sind. Bevor über verteilte Sperren gesprochen wird, muss erläutert werden, warum verteilte Sperren erforderlich sind. Das Gegenteil von verteilten Sperren sind eigenständige Sperren. Wenn wir Multithread-Programme schreiben, vermeiden wir Datenprobleme, die durch den gleichzeitigen Betrieb einer gemeinsam genutzten Variablen verursacht werden. Normalerweise verwenden wir eine Sperre, um die Richtigkeit der gemeinsam genutzten Variablen sicherzustellen Die gemeinsam genutzten Variablen liegen im gleichen Prozess. Wenn es mehrere Prozesse gibt, die gleichzeitig eine gemeinsam genutzte Ressource betreiben müssen, wie können sie sich dann gegenseitig ausschließen? Heutige Geschäftsanwendungen sind in der Regel Microservice-Architekturen, was auch bedeutet, dass eine Anwendung mehrere Prozesse bereitstellen muss. Wenn mehrere Prozesse dieselbe Datensatzzeile in MySQL ändern müssen, ist eine Verteilung erforderlich, um fehlerhafte Daten zu vermeiden wird zu diesem Zeitpunkt eingeführt. Der Stil ist gesperrt. Punkte erreichen wollen

Springboot liest die Datei, kann aber nach dem Packen in ein JAR-Paket nicht auf die neueste Entwicklung zugreifen. Es gibt eine Situation, in der Springboot die Datei nach dem Packen in ein JAR-Paket nicht lesen kann ist ungültig und kann nur über den Stream gelesen werden. Die Datei befindet sich unter resources publicvoidtest(){Listnames=newArrayList();InputStreamReaderread=null;try{ClassPathResourceresource=newClassPathResource("name.txt");Input

SpringBoot und SpringMVC sind beide häufig verwendete Frameworks in der Java-Entwicklung, es gibt jedoch einige offensichtliche Unterschiede zwischen ihnen. In diesem Artikel werden die Funktionen und Verwendungsmöglichkeiten dieser beiden Frameworks untersucht und ihre Unterschiede verglichen. Lassen Sie uns zunächst etwas über SpringBoot lernen. SpringBoot wurde vom Pivotal-Team entwickelt, um die Erstellung und Bereitstellung von Anwendungen auf Basis des Spring-Frameworks zu vereinfachen. Es bietet eine schnelle und einfache Möglichkeit, eigenständige, ausführbare Dateien zu erstellen

Wenn Springboot + Mybatis-plus keine SQL-Anweisungen zum Hinzufügen mehrerer Tabellen verwendet, werden die Probleme, auf die ich gestoßen bin, durch die Simulation des Denkens in der Testumgebung zerlegt: Erstellen Sie ein BrandDTO-Objekt mit Parametern, um die Übergabe von Parametern an den Hintergrund zu simulieren dass es äußerst schwierig ist, Multi-Table-Operationen in Mybatis-plus durchzuführen. Wenn Sie keine Tools wie Mybatis-plus-join verwenden, können Sie nur die entsprechende Mapper.xml-Datei konfigurieren und die stinkende und lange ResultMap konfigurieren Schreiben Sie die entsprechende SQL-Anweisung. Obwohl diese Methode umständlich erscheint, ist sie äußerst flexibel und ermöglicht es uns

1. Passen Sie den RedisTemplate1.1-Standard-Serialisierungsmechanismus an. Die API-basierte Redis-Cache-Implementierung verwendet die RedisTemplate-Vorlage für Daten-Caching-Vorgänge. Öffnen Sie hier die RedisTemplate-Klasse und zeigen Sie die Quellcodeinformationen der Klasse publicclassRedisTemplateextendsRedisAccessorimplementsRedisOperations an. Schlüssel deklarieren, verschiedene Serialisierungsmethoden des Werts, der Anfangswert ist leer @NullableprivateRedisSe

In Projekten werden häufig einige Konfigurationsinformationen benötigt. Diese Informationen können in der Testumgebung und in der Produktionsumgebung unterschiedliche Konfigurationen haben und müssen möglicherweise später basierend auf den tatsächlichen Geschäftsbedingungen geändert werden. Wir können diese Konfigurationen nicht fest im Code codieren. Am besten schreiben Sie sie in die Konfigurationsdatei. Sie können diese Informationen beispielsweise in die Datei application.yml schreiben. Wie erhält oder verwendet man diese Adresse im Code? Es gibt 2 Methoden. Methode 1: Wir können den Wert, der dem Schlüssel in der Konfigurationsdatei (application.yml) entspricht, über den mit @Value versehenen Wert erhalten. Diese Methode eignet sich für Situationen, in denen es relativ wenige Mikrodienste gibt: Tatsächlich Projekte, wenn das Geschäft kompliziert ist, Logik
