使用ApplicationRunner,以注解的方式直接注入基类,解决继承父类后,不再复写获取getBaseDao()方法

  • 继承ApplicationRunner接口,可以在ApplicationContext加载完成后被调用
  • 执行顺序是在所有的初始化和自动配置完成之后
  • 可以使用@Component或@Configuration标记
  • 当有多个继承ApplicationRunner的事后,执行的顺序是根据其注入的顺序决定的。如果想要指定顺序执行可以使用@Order进行排序
  • 除了ApplicationRunner外,CommandLineRunner也提供了相似的功能,最大的区别在于一个用ApplicationArguments接收参数,一个用Sting数组进行接收参数

示例,修改传统实现获取基础持久对象方法,主动注入基类
原始调用方法

//传统实现调用方式1
@Slf4j
@Repository
@RequiredArgsConstructor
public class UserRepository extends BaseRepository<User, Long> {
	private final UserDao userDao;
	@Override
	public BaseDao<User, Long> getBaseDao(){
		return userDao;
	}
}

//传统实现调用方式2
@Slf4j
@Repository
@RequiredArgsConstructor
public class UserRepository extends BaseRepository<User, Long> {
	private final UserDao userDao;
	//PostConstruct会在初始化对象技术,并且注入完类后,BeanPostProcessor后置处理器之前执行
	@PostConstruct
	public void init(){
		this.baseDao = userDao;
	}
}

修改后注入方式

//启动注入方式
@Slf4j
@Repository
@RequiredArgsConstructor
@AutomaticInjectionClassProvider(value = UserDao.class, fieldName = "baseDao")
public class UserRepository extends BaseRepository<User, Long> {
}

继承ApplicationRunner,当ApplicationContext加载完成后被调用,并且在所有的初始化和自动配置完成之后执行

@Configuration
@RequiredArgsConstructor
public class AutomaticInjectionPostProcessor implements ApplicationRunner {

	private final ApplicationContext applicationContext;

	@Override
	public void run(ApplicationArguments args) throws Exception {
		Map<String, Object> beans = applicationContext.getBeansWithAnnotation(AutomaticInjectionClassProvider.class);
		Collection<Object> values = beans.values();
		for (Object beanObject : values) {
			Class<?> beanClass = beanObject.getClass();
			AutomaticInjectionClassProvider injectionClassProvider = beanClass.getAnnotation(AutomaticInjectionClassProvider.class);
			Class<?> injectionClassProviderClass = injectionClassProvider.value();
			String fieldName = injectionClassProvider.fieldName();
			Object providerObject = SpringUtil.getBean(injectionClassProviderClass);
			if (Objects.isNull(providerObject)) {
				throw new RuntimeException("启动注入异常,类未实例化: " + beanObject.getClass().getTypeName());
			}
			this.injectObject(beanObject, fieldName, providerObject);
		}
	}

	/**
	 * 注入对象
	 *
	 * @param beanObject     被注入对象
	 * @param fieldName      注入字段名
	 * @param providerObject 注入对象
	 * @throws Exception
	 */
	public void injectObject(Object beanObject, String fieldName, Object providerObject) throws Exception {

		Field baseDaoField = this.getClassByName(beanObject, fieldName);
		if (Objects.isNull(baseDaoField)) {
			throw new RuntimeException("启动注入异常: " + beanObject.getClass().getTypeName());
		}
		baseDaoField.setAccessible(true);

		if (AopUtils.isCglibProxy(beanObject)) {
			Object target = this.getProxyObject(beanObject);
			try {
				baseDaoField.set(target, providerObject);
			} catch (Exception e) {
				e.printStackTrace();
			}

		} else {
			baseDaoField.set(beanObject, providerObject);
		}
	}

	/**
	 * 根据字段名称查找字段
	 *
	 * @param object 被查找对象
	 * @param name   字段名
	 * @return
	 */
	public Field getClassByName(Object object, String name) {
		Assert.notNull(object, "Class must not be null");
		Assert.isTrue(!Object.class.getName().equals(object.getClass().getName()), "没有找到被注入的字段,  fieldName: " + name);
		Class<?> clazz = object.getClass();
		Field baseDaoField = ReflectionUtils.findField(clazz, name);
		if (Objects.isNull(baseDaoField)) {
			Class<?> superclass = clazz.getSuperclass();
			if (Object.class.getTypeName().equals(superclass.getTypeName())) {
				return null;
			}
			return getClassByName(clazz.getSuperclass(), name);
		}
		return baseDaoField;
	}

	/**
	 * 获取代理类
	 *
	 * @param obj 被代理类
	 * @return 代理对象
	 * @throws Exception 异常
	 */
	public Object getProxyObject(Object obj) throws Exception {
		Field cglib$CALLBACK_0 = obj.getClass().getDeclaredField("CGLIB$CALLBACK_0");
		cglib$CALLBACK_0.setAccessible(true);
		Object cglibSupperObject = cglib$CALLBACK_0.get(obj);

		Field advised = cglibSupperObject.getClass().getDeclaredField("advised");
		advised.setAccessible(true);
		Object target = ((AdvisedSupport) advised.get(cglibSupperObject)).getTargetSource().getTarget();

		if (AopUtils.isCglibProxy(target)) {
			return this.getProxyObject(target);
		}
		return target;
	}

}
@Inherited
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface AutomaticInjectionClassProvider {

	Class<?> value();

	String fieldName();

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 好的,以下是中文回复: 首先,我们需要定义一个类作为类,可以命名为"Animal",其中包含一个方法"eat": ``` class Animal: def eat(self): print("动物在进食") ``` 接着,我们定义一个继承自"Animal"的子类,可以命名为"Cat",并在其中重写"eat"方法: ``` class Cat(Animal): def eat(self): super().eat() print("猫在吃鱼") ``` 在"Cat"类中,我们使用了"super()"关键字来调用类的"eat"方法,然后再添加了一行代码来输出"猫在吃鱼"。 最后,我们可以测试一下"Cat"类的功能,如下所示: ``` cat = Cat() cat.eat() ``` 输出结果为: ``` 动物在进食 猫在吃鱼 ``` 可以看到,"Cat"类成功地继承了"Animal"类的方法,并且在重写方法使用了"super()"关键字来调用类的方法,实现了我们的需求。 ### 回答2: 定义一个类作为类,可以将其命名为Animal,该类包含一个名为speak()的方法。在speak()方法中,输出一句话 "动物会发出声音"。 ``` class Animal: def speak(self): print("动物会发出声音") ``` 接下来,我们定义一个继承自Animal的子类,可以将其命名为Dog。子类中重写了类的speak()方法,输出一句话 "狗会汪汪叫",并使用super关键字调用类的speak()方法。 ``` class Dog(Animal): def speak(self): super().speak() print("狗会汪汪叫") ``` 现在我们可以进行功能测试了。创建一个Dog类的实例,并调用其speak()方法。 ``` dog = Dog() dog.speak() ``` 运行程序后的输出结果为: ``` 动物会发出声音 狗会汪汪叫 ``` 可以看到,子类中重写了类的方法speak(),并在内部使用super关键字调用了类的speak()方法。这样,子类就继承类的方法,并添加了自己独特的功能。运行后,我们首先可以看到类的方法被调用输出了"动物会发出声音"这句话,接着子类的方法输出了"狗会汪汪叫"。这证明了子类中重写方法使用super关键字调用方法的功能是有效的。 ### 回答3: 类(类)是面向对象编程中的概念,它是其他类的础,其他类通过继承类来获得类的属性和方法继承的子类可以对类的方法进行重写或扩展。 如果要定义一个类作为类,并定义一个继承类的子类,并在子类中重写类的方法,并使用super关键字调用类的方法,可以按照以下步骤进行: 1. 定义类(类): ```python class Parent: def __init__(self, name): self.name = name def say_hello(self): print("Hello, " + self.name + "!") ``` 2. 定义子类并继承类: ```python class Child(Parent): def __init__(self, name, age): super().__init__(name) # 使用super关键字调用类的初始化方法 self.age = age def say_hello(self): # 重写类的say_hello方法 super().say_hello() # 使用super关键字调用类的say_hello方法 print("I'm " + str(self.age) + " years old!") ``` 3. 测试功能: ```python parent = Parent("Tom") parent.say_hello() # 输出:Hello, Tom! child = Child("Jerry", 5) child.say_hello() # 输出:Hello, Jerry! I'm 5 years old! ``` 在这个例子中,类Parent定义了一个名为say_hello的方法,子类Child继承类的属性和方法,并对类的say_hello方法进行了重写。在重写的say_hello方法中,使用了super关键字调用了类的say_hello方法,并添加了自己的输出语句。测试中,分别创建了一个类对象和一个子类对象,并调用了它们的say_hello方法,可以看到类对象输出类定义的输出语句,子类对象输出子类重写的输出语句和类输出的语句。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值