Spring Core
在生命周期长的Bean中指定一个短的
- xml配置方式
proxy-target-class=true:强制使用 CGLib 代理
proxy-target-class=false:Spring 会根据情况自动选择代理方式.
两种代理方式的区别:JDK 动态代理:只能代理实现了接口的类,基于接口创建代理对象,性能较好,但有接口限制
CGLib 代理:可以代理没有接口的类,通过继承目标类创建代理对象,功能更强大,但性能相对较低
xml配置
<!-- DefaultUserPreferences implements the UserPreferences interface -->
<bean id="userPreferences" class="com.stuff.DefaultUserPreferences" scope="session">
<aop:scoped-proxy proxy-target-class="false"/>
</bean>
<bean id="userManager" class="com.stuff.UserManager">
<property name="userPreferences" ref="userPreferences"/>
</bean>
- 注解方式
// CGLIB 方式
@Scope(value="request", proxyMode = ScopedProxyMode.TARGET_CLASS)
// JDK动态代理 方式
@Scope(value="request", proxyMode = ScopedProxyMode.INTERFACES)
单例Bean中的有原型bean
方法一:lookup
,通过lookup会自动去查找类型一样的bean进行注入;注意使用lookup的方法必须为抽象方法;
@Component("AAA")
@Scope(value = "prototype")
public class AAA {
@Bean
public AAA createAAA(A a,B b) {
return this;
}
}
@Component
public abstract class ControllerManager {
private AAA aaa;
@Lookup
public abstract AAA createAAA() ;
public void test() {
this.aaa = createAAA();
}
}
方法二:使用ObjectFactory,请看这里
@Autowired
field底层实现
@Autowired
先会按类型注入,如果有多个类型,则按照名字注入;在源码中设置值是通过反射实现org.springframework.beans.DirectFieldAccessor.FieldPropertyHandler#setValue
@Override
public void setValue(@Nullable Object value) throws Exception {
try {
ReflectionUtils.makeAccessible(this.field);
this.field.set(getWrappedInstance(), value);
} catch (IllegalAccessException ex) {
throw new InvalidPropertyException(getWrappedClass(), this.field.getName(),
"Field is not accessible", ex);
}
}
static @Bean
static @Bean 示例
@Configuration
public class InfraConfig {
@Bean
public static PropertySourcesPlaceholderConfigurer placeholderConfigurer() {
return new PropertySourcesPlaceholderConfigurer();
}
@Bean
public static MyService myService(Dependency dep) {
// 通过方法参数注入
return new MyService(dep);
}
}
特点:
- 方法不依赖配置类实例,可以在配置类实例化之前,被调用。
- 不能使用配置类字段注入。
- 默认仍是单例,Spring会缓存对象,下次获取直接复用。
- 静态方法减少CGLIB开销。
- 只能使用构造方法或者
ApplicationContext.getBean(...)
来获取。
适用场景:
早基础设施Bean 如PropetySourcePlaceholerConfigurer
避免循环依赖
轻量初始化 不使用CBLIB,不依赖配置类字段
特性 | 普通 @Bean | static @Bean |
---|---|---|
配置类依赖 | 依赖实例化后的配置类字段 | 不依赖配置类实例 |
可用字段注入 | ✅ 可以 | ❌ 不可以 |
调用时机 | 配置类实例化后 | 配置类实例化前 |
Spring 单例缓存 | ✅ 单例 | ✅ 单例 |
适用场景 | 业务 Bean、普通依赖 Bean | 基础设施 Bean、早期初始化、避免循环依赖 |
CGLIB 代理开销 | 有 | 没有 |
@Autowired
或者@Resource
指定实现类
- 在某一个子实现类上使用
@Primary
指定要注入的Bean为当前的bean, 以下注入方式二选一
@Primary
@Component
public class ProtoBeanImpl implements ProtoBean{
}
@Primary
@Bean
public ProtoBeanImpl protoBean(){
return new ProtoBeanImpl();
}
- 使用
@Qualifier("xxx")
注解 指定要注入的bean的类型
@Qualifier("protoBeanImpl2")
@Autowired
private ProtoBean bean;
@Autowired
public SingleBean(@Qualifier("protoBeanImpl2") ProtoBean proto) {
this.protoBeanImpl2 = (ProtoBeanImpl2) proto;
}
- 使用
@Resource
@Resource(name="protoBeanImpl2")
private ProtoBean protoBean;
// 和下面写法等同,不指定name,默认为变量名
@Resource
private ProtoBean protoBeanImpl2;
构造器注入
-
using constructor inject,do not need other annotation if all the properties is base type ,using
@ConstructorProperties({"xxx","xxx",...})
to inject the value -
if just one constructor here, need' not
@Autowired
-
only one multi-argument constructor can be set
@Autowired(required = true)
-
if one more constructor are annotationed with
@Autowired(required = false)
The constructor with the greatest number of dependencies that can be satisfied by matching beans in the Spring container will be chosen -
The
@Autowired
,@Inject
,@Value
, and@Resource
annotations are handled by Spring BeanPostProcessor implementations
使用AspectJ注解进行AOP配置好还是xml
使用xml配置时,将AOP配置分散了,一部分在xml中,一部分在后台的class类中。不符合DRY原则。
使用@AspectJ
,则将整个AOP的配置放在一个配置类中,@AspectJ
支持额外的实例模型更丰富的组合,是每个切面成为一个模型单元。
同时,@AspectJ
能被 Spring AOP 和AspectJ 解析,你可以使用AspectJ的语法去实现更加复杂的切面逻辑
闭包 和 回调
闭包:闭包和匿名函数经常被用作同义词。但严格来说,匿名函数就是字面意义上没有被赋予名称的函数,而闭包则实际上是一个函数的实例,也就是说是存在于内存里的某个结构体。如果从实现上来看的话,匿名函数如果没有捕捉自由变量,那么它其实可以被实现为一个函数指针,或者直接内联到调点,如果它捕捉了自由变量那么它将是一个闭包;而闭包则意味着同时包括函数指针和环境两个关键元素。参考出处 Closure Sample
public interface Adder {
int add(int x);
}
public static Adder makeAdder(final int n) {
return new Adder() {
public int add(int x) {
return x + n;
}
};
}
回调:在计算机程序设计中,回调函数,或简称回调(Callback 即call then back 被主函数调用运算后会返回主函数),是指通过参数将函数传递到它代码的,某一块可执行代码的引用参考出处 Callback Sample
class RemoteClass {
private OnChangeListener mOnChangeListener;
void makeSomeChanges() {
/*
.. do something here and call callback
*/
mOnChangeListener.onChanged(this, 1);
}
public void setOnChangeListener(OnChangeListener listener) {
mOnChangeListener = listener;
}
public interface OnChangeListener {
void onChanged(RemoteClass remoteClass, int test);
}
}
class Test {
public static void main(String[] args) {
RemoteClass obj = new RemoteClass();
obj.setOnChangeListener(demoChanged);
obj.makeSomeChanges();
}
// this case remind me of spring framework lots of anonymous ObjectFactory call back
private static RemoteClass.OnChangeListener demoChanged = new RemoteClass.OnChangeListener() {
@Override
public void onChanged(RemoteClass remoteClass, int incoming) {
switch (incoming) {
case 1:
System.out.println("I will take appropriate action!");
break;
default:
break;
}
}
};
}
// callback in springframework 4.3.x
// @see org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean
// Create bean instance.
if (mbd.isSingleton()) {
sharedInstance = getSingleton(beanName, new ObjectFactory<Object>() {
@Override
public Object getObject() throws BeansException {
try {
return createBean(beanName, mbd, args);
}
catch (BeansException ex) {
// Explicitly remove instance from singleton cache: It might have been put there
// eagerly by the creation process, to allow for circular reference resolution.
// Also remove any beans that received a temporary reference to the bean.
destroySingleton(beanName);
throw ex;
}
}
});
bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
}
@Configurable
@Configuration
@Configuration
是配置类
@Configurable
它是一个bean所依赖的Aspect,并且不被Spring管理,但是可以自动注入
Spring Boot自定义一个Event发布
本案例使用到了SpEL
(Spring Express Language)表达式
- 定义自己的事件
MyEvent
/**
* @Author: WhaleFall541
* @Date: 2021/7/12 21:38
*/
public class MyEvent extends ApplicationEvent {
private final String address;
private final String content;
public MyEvent(Object source, String address, String content) {
super(source);
this.address = address;
this.content = content;
}
// getter and setter omit
@Override
public String toString() {
return "MyEvent{" +
"address='" + address + '\'' +
", content='" + content + '\'' +
'}';
}
}
- 自定义事件发布类
MyEventPublish
/**
* @Author: WhaleFall541
* @Date: 2021/7/12 21:41
*/
@Component
public class MyEventPublish implements ApplicationEventPublisherAware {
private ApplicationEventPublisher publisher;
@Override
public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {
this.publisher = applicationEventPublisher;
}
public void publish(String address, String content) {
if ("aaa".equals(address)) {
publisher.publishEvent(new MyEvent(this, address, content));
return;
}
}
}
- 编写事件监听类
/**
* @Author: WhaleFall541
* @Date: 2021/7/12 21:38
*/
@Component
public class MyListener implements ApplicationEventPublisherAware {
private Log log = LogFactory.getLog(MyListener.class);
// SpEL #a0 代表第一个入参
// @EventListener(condition = "#a0.content == 'foo'")
// SpEL #event 代表名称相同的参数
// @EventListener(condition = "#event.content == 'foo'")
// 如果要表示一个对象要使用@XXX testMethod为容器中bean的名字
@EventListener(condition = "@testMethod.test().equals(#a0.content)")
// @EventListener({ContextStartedEvent.class, ContextRefreshedEvent.class})
public void process(MyEvent event) {
System.err.println("event test is ok " + event);
}
}
TestMethod
类
@Component
public class TestMethod {
public List<String> test() {
List<String> al = new ArrayList();
al.add("foo");
al.add("fool");
return al;
}
}
- 测试类
注意要引入依赖,此处列举gradle
依赖配置 testImplementation 'org.springframework.boot:spring-boot-starter-test'
@SpringBootTest
class Charter1Tests implements ApplicationContextAware {
@Test
void testApplicationEvent() {
MyEventPublish bean = context.getBean(MyEventPublish.class);
bean.publish("aaa", "foo");
}
}
- 署名:在原有代码和衍生代码中,保留原作者署名及代码来源信息。
- 保留许可证:在原有代码和衍生代码中,保留Apache 2.0协议文件。
- 署名:应在使用本文档的全部或部分内容时候,注明原作者及来源信息。
- 非商业性使用:不得用于商业出版或其他任何带有商业性质的行为。如需商业使用,请联系作者。
- 相同方式共享的条件:在本文档基础上演绎、修改的作品,应当继续以知识共享署名 4.0国际许可协议进行许可。