中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

在spring中如何使用自定義注解注冊監聽器

發布時間:2021-07-27 15:03:43 來源:億速云 閱讀:134 作者:小新 欄目:編程語言

小編給大家分享一下在spring中如何使用自定義注解注冊監聽器,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!

接口回調

監聽器本質上就是利用回調機制,在某個動作發生前或后,執行我們自己的一些代碼。在Java語言中,可以使用接口來實現。

實現一個監聽器案例

為了方便,直接在spring環境中定義:以工作(work)為例,定義工作開始時(或結束時)的監聽器。

1. 定義回調的接口

package com.yawn.demo.listener;

/**
 * @author Created by yawn on 2018-01-21 13:53
 */
public interface WorkListener {

  void onStart(String name);
}

2. 定義動作

package com.yawn.demo.service;

import com.yawn.demo.listener.WorkListener;

/**
 * @author Created by yawn on 2018-01-21 13:39
 */
@Service
public class MyService {

  @Resource
  private PersonService personService;

  private WorkListener listener;
  public void setWorkListener(WorkListener workListener) {
    this.listener = workListener;
  }

  public void work(String name) {
    listener.onStart(name);
    personService.work();
  }
}

動作work為一個具體的方法,在work()方法的適當時機,調用前面定義的接口。此外,在這個動作定義類中,需要提高設置監聽器的方法。

3. 監聽測試

@RunWith(SpringRunner.class)
@SpringBootTest
public class DemoSpringAnnotationApplicationTests {

  @Resource
  private MyService myService;

  @Test
  public void test1() {
    // 接口設置監聽器
    myService.setWorkListener(new WorkListener() {
      @Override
      public void onStart(String name) {
        System.out.println("Start work for " + name + " !");
      }
    });
//    // lambda 表達式設置監聽器
//    myService.setWorkListener(name -> System.out.println("Start work for " + name + " !"));
    // 工作
    myService.work("boss");
  }

 @Test
  public void test2() {
   // 繼承實現類設置監聽器
   myService.setWorkListener(new myWorkListener());
   // 工作
   myService.work("boss");
  }

  class myWorkListener extends WorkListenerAdaptor {
    @Override
    public void onStart(String name) {
      System.out.println("Start work for " + name + " !");
    }
  }
}

使用以上兩種方法測試,得到了結果為:

Start work for boss !
working hard ...

說明在動作work發生之前,執行了我們在測試類中寫下的監聽代碼,實現類監聽的目的。

使用注解實現監聽器

在以上代碼中,調用 setWorkListener(WorkListener listener)  方法一般稱作設置(注冊)監聽器,就是將自己寫好的監聽代碼,設置為動作的監聽器。然而,在每次注冊監聽器時,一般需要寫一個類,實現定義好的接口或繼承實現接口的類,再重寫接口定義的方法即可。因此,聰明的程序員就想簡化這個過程,所以就想出了使用注解的方法。使用注解,將監聽代碼段寫在一個方法中,使用一個注解標記這個方法即可。

的確,使用變得簡單了,但實現卻不見得。

1. 定義一個注解

package com.yawn.demo.anno;
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface WorkListener {
}

2. 解析注解

package com.yawn.demo.anno;
import com.yawn.demo.service.MyService;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.util.LinkedHashMap;
import java.util.Map;
/**
 * @author Created by yawn on 2018-01-21 14:46
 */
@Component
public class WorkListenerParser implements ApplicationContextAware, InitializingBean {
  @Resource
  private MyService myService;
  private ApplicationContext applicationContext;

  @Override
  public void afterPropertiesSet() throws Exception {
    Map<String, Object> listenerBeans = getExpectListenerBeans(Controller.class, RestController.class, Service.class, Component.class);
    for (Object listener : listenerBeans.values()) {
      for (Method method : listener.getClass().getDeclaredMethods()) {
        if (!method.isAnnotationPresent(WorkListener.class)) {
          continue;
        }
        myService.setWorkListener(name -> {
          try {
            method.invoke(listener, name);
          } catch (Exception e) {
            e.printStackTrace();
          }
        });
      }
    }
  }

  /**
   * 找到有可能使用注解的bean
   * @param annotationTypes 需要進行掃描的類級注解類型
   * @return 掃描到的beans的map
   */
  private Map<String, Object> getExpectListenerBeans(Class<? extends Annotation>... annotationTypes) {
    Map<String, Object> listenerBeans = new LinkedHashMap<>();
    for (Class<? extends Annotation> annotationType : annotationTypes) {
      Map<String, Object> annotatedBeansMap = applicationContext.getBeansWithAnnotation(annotationType);
      listenerBeans.putAll(annotatedBeansMap);
    }
    return listenerBeans;
  }

  @Override
  public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
    this.applicationContext = applicationContext;
  }
}

在注解的解析過程中,設置監聽器。

在解析類中,實現了接口ApplicationContextAware,為了在類中拿到ApplicationContext的引用,用于得到 IOC 容器中的 Bean;而實現接口InitializingBean,則是為了在一個合適的時機執行解析注解、設置監聽器的代碼。 如果不這樣做,可以在CommandLineRunner執行時調用解析、設置的代碼,而ApplicationContext也可以自動注入。

3. 測試

在執行完以上代碼后,監聽器就已經設置好了,可以進行測試了。

package com.yawn.demo.controller;
import com.yawn.demo.anno.WorkListener;
import com.yawn.demo.service.MyService;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource;

/**
 * @author Created by yawn on 2018-01-21 13:28
 */
@RestController
public class TestController {
  @Resource
  private MyService myService;
  @GetMapping("/work")
  public Object work() {
    myService.work("boss");
    return "done";
  }

  @WorkListener
  public void listen(String name) {
    System.out.println("Start work for " + name + " !");
  }
}

寫一個監聽方法,參數類型和個數與接口相同,然后加上自定義的注解即可。當啟動環境后,監聽器就已經設置好了。

然后通過url調用myService的work()方法,可以看到結果:

Start work for boss !
working hard ...

已經調用了監聽方法。在接下來的開發中,就可以使用這個注解注冊監聽器了。

看完了這篇文章,相信你對“在spring中如何使用自定義注解注冊監聽器”有了一定的了解,如果想了解更多相關知識,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

澜沧| 宜宾市| 集安市| 中超| 华亭县| 江油市| 平顶山市| 邵东县| 盐山县| 河池市| 黑山县| 上思县| 晋州市| 开平市| 马公市| 珲春市| 阜城县| 牡丹江市| 禹州市| 中江县| 鹤山市| 万州区| 都江堰市| 海南省| 大冶市| 怀宁县| 合肥市| 山西省| 长海县| 日喀则市| 湘阴县| 金湖县| 县级市| 郯城县| 伊宁市| 芦山县| 辽阳市| 新乡县| 大化| 繁峙县| 宣城市|