亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Spring中的ImportBeanDefinitionRegistrar接口詳解

 更新時(shí)間:2023年09月20日 08:45:00   作者:信仰_273993243  
這篇文章主要介紹了Spring中的ImportBeanDefinitionRegistrar接口詳解,ImportBeanDefinitionRegistrar接口是也是spring的擴(kuò)展點(diǎn)之一,它可以支持我們自己寫的代碼封裝成BeanDefinition對(duì)象,注冊(cè)到Spring容器中,功能類似于注解@Service @Component,需要的朋友可以參考下

接口功能介紹

描述:ImportBeanDefinitionRegistrar接口是也是spring的擴(kuò)展點(diǎn)之一,它可以支持我們自己寫的代碼封裝成BeanDefinition對(duì)象,注冊(cè)到Spring容器中,功能類似于注解@Service @Component。

很多三方框架集成Spring的時(shí)候,都會(huì)通過該接口,實(shí)現(xiàn)掃描指定的類,然后注冊(cè)到spring容器中,比如Mybatis中的Mapper接口,springCloud中的FeignClient接口,都是通過該接口實(shí)現(xiàn)的自定義注冊(cè)邏輯。

1、ImportBeanDefinitionRegistrar接口實(shí)現(xiàn)類,只能通過@Import注解的方式來注入,通常把@Import修飾在啟動(dòng)類或配置類。

2、使用@Import,如果括號(hào)中的類是ImportBeanDefinitionRegistrar的實(shí)現(xiàn)類,啟動(dòng)時(shí)會(huì)觸發(fā)ImportBeanDefinitionRegistrar接口的方法,將其中要注冊(cè)的類注冊(cè)成bean。

3、實(shí)現(xiàn)該接口的類擁有注冊(cè)bean的能力。

//接口所有抽象方法,合并看就一個(gè)注冊(cè)BeanDefinition的方法
public interface ImportBeanDefinitionRegistrar {
	//把自定義的類封裝成BeanDefinition對(duì)象,注冊(cè)到Spring里面去
	default void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry, BeanNameGenerator importBeanNameGenerator) {
		registerBeanDefinitions(importingClassMetadata, registry);
	}
	//我們平時(shí)重寫這個(gè)就可以了
	default void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
	}
}

使用案例

1、案例1

自定義業(yè)務(wù)類UserServiceTest,通過ImportBeanDefinitionRegistrar將注冊(cè)Spring容器中。在通過spring容器獲取Bean=UserServiceTest

//業(yè)務(wù)類
public class UserServiceTest {
	/**
	 * 獲取用戶名稱
	 * @return 用戶名稱
	 */
	public String getUserName(){
		return "測(cè)試";
	}
}

ImportBeanDefinitionRegistrar實(shí)現(xiàn)類

//注意這里不能加注解,要通過Import導(dǎo)入進(jìn)去。
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
	//業(yè)務(wù)類轉(zhuǎn)成bd,注冊(cè)到spring容器中注入
	@Override
	public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
		//1、通過Bd工具類生成bd對(duì)象,只是這個(gè)Db對(duì)象比較純潔沒有綁定任何類
		BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition();
		GenericBeanDefinition beanDefinition = (GenericBeanDefinition) beanDefinitionBuilder.getBeanDefinition();
		//2、設(shè)置bd綁定的類型
		beanDefinition.setBeanClass(UserServiceTest.class);
		//3、注冊(cè)到spring容器中
		registry.registerBeanDefinition("userServiceTest",beanDefinition);
	}
}
@Configuration
@Import(MyImportBeanDefinitionRegistrar.class)
public class AppConfigClassTest {
	//在配置類導(dǎo)入ImportBeanDefinitionRegistrar實(shí)現(xiàn)類
}
//測(cè)試
public static void main(String[] args) {
	AnnotationConfigApplicationContext AnnotationConfigApplicationContext =
			new AnnotationConfigApplicationContext(AppConfigClassTest.class);
	UserServiceTest userServiceTest = AnnotationConfigApplicationContext.getBean("userServiceTest", UserServiceTest.class);
	String userName = userServiceTest.getUserName();
	System.out.println(userName);
}

如果只是把業(yè)務(wù)類注冊(cè)到Spring容器中我們通過其他注解就可以了,那么ImportBeanDefinitionRegistrar有沒有更高級(jí)的玩法。

2、案例2

public interface UserServiceTestInterface {
	public void list();
}
//因?yàn)槭墙涌?,但是spring的容器里面是不允許注入接口的,只能是接口的實(shí)現(xiàn)類。如果我們寫個(gè)去實(shí)現(xiàn)接口,那就沒有什么意義了,沒必要搞得那么復(fù)雜,這次我們通過代理類來完成對(duì)接口的實(shí)現(xiàn)。
public class MyInvocationHandler implements InvocationHandler {
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("代理類邏輯代碼");
		return null;
	}
}
//注意這里不能加注解,要通過Import導(dǎo)入進(jìn)去。
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
	 @Override
	 public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
        //通過工具類生成一個(gè)bd,只是這個(gè)Db對(duì)象比較純潔沒有綁定任何類
		BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition();
		GenericBeanDefinition beanDefinition = (GenericBeanDefinition) beanDefinitionBuilder.getBeanDefinition();
		//設(shè)置Bean的類型MyInvocationHandler,類型是實(shí)現(xiàn)類的類型,不是接口類型。因?yàn)樵趯?shí)例化的時(shí)候,調(diào)用的是設(shè)置類型所以對(duì)應(yīng)的構(gòu)造方法。
		beanDefinition.setBeanClass(MyInvocationHandler.class);
		//注冊(cè)到Spring容器中進(jìn)去
		registry.registerBeanDefinition("userServiceTest",beanDefinition);
    }
}
//通過配置類,導(dǎo)入ImportBeanDefinitionRegistrar的實(shí)現(xiàn)類
@Configuration
@Import(MyImportBeanDefinitionRegistrar.class)
public class AppConfigClassTest {
}

測(cè)試

public static void main(String[] args) {
	AnnotationConfigApplicationContext AnnotationConfigApplicationContext = new AnnotationConfigApplicationContext(AppConfigClassTest.class);
	//通過name獲取Bean,注意此時(shí)的bean類型不是UserServiceTestInterface類型,而是MyInvocationHandler
	Object userServiceTestInterface = AnnotationConfigApplicationContext.getBean("userServiceTest");
	if(userServiceTestInterface instanceof MyInvocationHandler){
		//代碼實(shí)際會(huì)走到這里,把object轉(zhuǎn)成帶代理類。
		MyInvocationHandler u = (MyInvocationHandler) userServiceTestInterface;
		//生成接口UserServiceTestInterface的代理對(duì)象
		UserServiceTestInterface o = (UserServiceTestInterface) Proxy.newProxyInstance(MainTest.class.getClassLoader(), new Class[]{UserServiceTestInterface.class}, u);
	}
}

3、案例3

定義一個(gè)業(yè)務(wù)接口,通過FactoryBean+InvocationHandler來生成該接口的代理類,無需手動(dòng)寫業(yè)務(wù)接口的實(shí)現(xiàn)類,很多底層框架就是這樣實(shí)現(xiàn)的。

  • InvocationHandler:主要是通過Invoke方法來攔截業(yè)務(wù)接口的方法
  • FactoryBean:主要是用來將生成的代理類。
  • ImportBeanDefinitionRegistrar:在這里的作用就是幫忙我們把自定義的FactoryBean注冊(cè)到Spring中
//業(yè)務(wù)接口
public interface UserServiceTestInterface {
	public void list();
}

自定義FactoryBean這樣我們控制Bean的創(chuàng)建的過程,實(shí)現(xiàn)InvocationHandler用來攔截業(yè)務(wù)接口的方法。

//創(chuàng)建代理類,代理UserServiceTestInterface接口,UserServiceTestInterface接口方法在執(zhí)行前后都會(huì)被invoke方法攔截
//FactoryBean可以生成某一個(gè)類型Bean實(shí)例,它最大的一個(gè)作用是:可以讓我們自定義Bean的創(chuàng)建過程
public class MyFactoryBean implements FactoryBean, InvocationHandler {
	//為了使這個(gè)類更好地?cái)U(kuò)展。創(chuàng)建更多的接口,我們定義一個(gè)參數(shù),讓他們通過參數(shù)傳遞進(jìn)來。
	private Class classs;
	//添加一個(gè)有參的構(gòu)造方法。
	public MyFactoryBean(Class classs){
		this.classs = classs;
	}
	//攔截Class的所有方法
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("diaoyonlejiekou123");
		return null;
	}
	//返回bean的對(duì)象。spring會(huì)自動(dòng)把它add到容器里面去。
	@Override
	public Object getObject() throws Exception {
		Class[] clazzs = new Class[]{classs};//目標(biāo)類集合。
		//通過proxy來得到代理對(duì)象。本來最有一個(gè)參數(shù)需要穿代理類對(duì)象,但因?yàn)楸绢悓?shí)現(xiàn)了InvocationHandler,所以只需傳this
		Object proxy = Proxy.newProxyInstance(this.getClass().getClassLoader(), clazzs, this);
		return proxy;//返回的這個(gè)對(duì)象,會(huì)把加到spring的容器中。
	}
	//返回要添加到容器里bean的類型
	@Override
	public Class<?> getObjectType() {
		return this.classs;
	}
}

自定義ImportBeanDefinitionRegistrar實(shí)現(xiàn)類,把我們自定義的FactoryBean注冊(cè)到Spring中。

//注意這里不能加注解,要通過Import導(dǎo)入進(jìn)去。
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
	@Override
	public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
		//通過工具類生成一個(gè)bd,只是這個(gè)Db對(duì)象比較純潔沒有綁定任何類
		BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition();
		//為什么要轉(zhuǎn)成GenericBeanDefinition這種類型。因?yàn)镚enericBeanDefinition有更多修改bd屬性的方法。后面我會(huì)介紹為什么要修改屬性。
		GenericBeanDefinition beanDefinition = (GenericBeanDefinition) beanDefinitionBuilder.getBeanDefinition();
		//這里很重要。getConstructorArgumentValues是為了獲取該bd的所有構(gòu)造方法,因?yàn)槲覀冎貙懥擞袇?gòu)造方法,所有我們需要帶參數(shù)過去 
		//不然spring沒法幫我們實(shí)例化,addGenericArgumentValue是添加參數(shù),該代碼會(huì)執(zhí)行兩步
		//第一步是匹配對(duì)應(yīng)的構(gòu)造方法,第二步是實(shí)例化。
		beanDefinition.getConstructorArgumentValues().addGenericArgumentValue(UserServiceTestInterface.class.getName());
		//因?yàn)榇韺?duì)象類型的,實(shí)例化的時(shí)候走的是代理類的構(gòu)造方法
		beanDefinition.setBeanClass(MyFactoryBean.class);
		//注冊(cè)bd
		registry.registerBeanDefinition("userServiceTest",beanDefinition);
	}
}

測(cè)試

public static void main(String[] args) {
	AnnotationConfigApplicationContext AnnotationConfigApplicationContext = new AnnotationConfigApplicationContext(AppConfigClassTest.class);
	//通過name獲取Bean
	Object userServiceTestInterface = AnnotationConfigApplicationContext.getBean("userServiceTest");
	//針對(duì)這種場(chǎng)景Bean的類型是,通過FactoryBean的getObjectType方法返回的。
	UserServiceTestInterface u = (UserServiceTestInterface) userServiceTestInterface;
	u.list();
}

?到此這篇關(guān)于Spring中的ImportBeanDefinitionRegistrar接口詳解的文章就介紹到這了,更多相關(guān)ImportBeanDefinitionRegistrar接口內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論