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

文章介绍了如何使用SpringBoot的ApplicationRunner接口在ApplicationContext加载完成后执行自定义逻辑,确保自动配置完成后进行特定的注入操作。通过@PostConstruct注解和@Order控制执行顺序。同时,对比了ApplicationRunner和CommandLineRunner的区别。文章还展示了如何处理CGLib代理对象的注入问题。
  • 继承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();

}

### C# 子类继承父类后如何在函数中直接调用父类方法 在 C# 中,子类可以通过 `base` 关键字显式调用父类方法。这种方法调用仅限于子类能够访问的父类成员,即父类方法的访问修饰符必须为 `public` 或 `protected`[^1]。以下是一个详细的说明和示例代码: #### 1. 使用 `base` 调用父类方法 当子类重写了父类的虚方法时,可以通过 `base` 调用父类的原始实现。例如: ```csharp public class ParentClass { public virtual void Display() { Console.WriteLine("Parent class method."); } } public class ChildClass : ParentClass { public override void Display() { base.Display(); // 调用父类的 Display 方法 Console.WriteLine("Child class method."); } } ``` 在上述代码中,`ChildClass` 的 `Display` 方法通过 `base.Display()` 调用了父类的 `Display` 方法。 #### 2. 父类方法的访问修饰符限制 子类是否能够调用父类方法取决于父类方法的访问修饰符: - 如果父类方法是 `public` 或 `protected`,子类可以直接通过 `base` 调用。 - 如果父类方法是 `private`,则子类无法访问或调用该方法[^2]。 #### 3. 示例:隐藏与重写方法的区别 如果子类使用 `new` 关键字隐藏了父类方法,则需要显式使用 `base` 调用父类方法。例如: ```csharp public class ParentClass { public void Show() { Console.WriteLine("Parent class Show method."); } } public class ChildClass : ParentClass { public new void Show() { base.Show(); // 调用父类的 Show 方法 Console.WriteLine("Child class Show method."); } } ``` 在此示例中,`ChildClass` 的 `Show` 方法隐藏了父类的 `Show` 方法,并通过 `base.Show()` 显式调用了父类的实现[^1]。 #### 4. 构造函数中的 `base` 调用 在子类的构造函数中,可以通过 `base` 调用父类的带参数构造函数,从而初始化继承父类的成员变量。例如: ```csharp public class Person { public Person(string name) { Console.WriteLine($"Person constructor called with name: {name}"); } } public class Student : Person { public Student(string name) : base(name.ToUpper()) // 调用父类构造函数并转换参数 { Console.WriteLine("Student constructor called."); } } ``` 在上述代码中,`Student` 类的构造函数通过 `base` 调用了父类 `Person` 的构造函数,并对参数进行了处理[^3]。 #### 5. 接口实现与基类方法调用 如果子类同时实现了接口并且继承父类父类方法不能通过 `base` 调用接口方法的实现。这是因为接口方法的实现属于子类的一部分,而非父类的一部分[^4]。 ### 总结 C# 中子类可以直接调用父类方法,但需要遵循以下规则: - 使用 `base` 关键字显式调用父类方法。 - 父类方法的访问修饰符必须允许子类访问。 - 在构造函数中,可以通过 `base` 调用父类的带参数构造函数。 - 区分隐藏方法和重写方法的不同调用方式。 - 接口实现与基类方法调用之间存在限制。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值