詳解Spring中Lookup註解的使用

我們知道在spring容器中單獨的一個抽象類是不能成為一個bean的,那麼有沒有辦法呢?這個時候我們可以使用Lookup註解,我們可以看下spring的掃描bean部分邏輯。我們知道在spring中要想成為一個bean,必須先生成BeanDefinition對象,如果一個抽象類中沒有含有Lookup註解的方法,在spring掃描時就會被排除掉。

	/**
	 * 1、判斷是不是獨立的類,非靜態內部類則無法生成bean,
	 * 2、判斷是不是接口或者抽象類(有一種特殊情況),是則無法生成
	 * 3、判斷如果是抽象類,但是裡面有某個方法上面加油@lookup註解,則也可以生成bean
	 * Determine whether the given bean definition qualifies as candidate.
	 * <p>The default implementation checks whether the class is not an interface
	 * and not dependent on an enclosing class.
	 * <p>Can be overridden in subclasses.
	 * @param beanDefinition the bean definition to check
	 * @return whether the bean definition qualifies as a candidate component
	 */
	protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
		AnnotationMetadata metadata = beanDefinition.getMetadata();
		return (metadata.isIndependent() && (metadata.isConcrete() ||
				(metadata.isAbstract() && metadata.hasAnnotatedMethods(Lookup.class.getName()))));
	}

下面我們就驗證下,不使用Lookup註解的情況

@Component
public abstract class C1 {
 
}

運行結果報錯

 在抽象類中隨便寫個方法,然後方法上面增加Lookup註解

@Component
public abstract class C1 {
 
    @Lookup
    public void a(){}
}

運行結果,正常輸出,通過cglib代理生成新的類

 但是一般很少這樣用,另外一種場景可能會用到。在某個單例bean中使用另外一個bean對象,但是每次又想返回的對象不同。但是spring在容器中註入bean的時候,scope默認的是單例模式,也就是說在整個應用中隻能創建一個實例。當scope為PROTOTYPE類型的時候,在每次註入的時候會自動創建一個新的bean實例。但是當一個單例模式的bean去引用PROTOTYPE類型的bean的時候,PROTOTYPE類型的bean也會變成單例。

@Component
public class D3 {
 
    @Autowired
    private E4 e4;
 
    public void a(){
        System.out.println(this.e4);
    }
}
 
@Component
public class E4 {
}

輸出結果,可以看到每次打印出來的對象是同一個 

使用Lookup註解

@Component
public class D3 {
 
    public void a(){
        System.out.println(this.a1());
    }
 
    @Lookup
    public E4 a1(){
        return null;
    }
}

運行輸出結果,每次輸出的結果已經不相同瞭,已經達到瞭我們的需求

這是什麼原因導致的呢?還有就是我們a1方法返回的是空,但是輸出的結果為啥也有值呢?
因為spring在遇到這種標有Lookup註解的方法時,會重寫該方法,然後返回結果,所以我們自己定義的方法不管有沒有返回值已經沒關系瞭。

到此這篇關於詳解Spring中Lookup註解的使用的文章就介紹到這瞭,更多相關Spring Lookup註解的使用內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: