springboot实战电商项目mall4j (https://gitee.com/gz-yami/mall4j)

java商城系统源码

1. bean注册

我们有个Person类

public class Person {

    private Integer age;
    
    private String name;

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Person [age=" + age + ", name=" + name + "]";
    }
}

1.1 传统bean注册

新建beans.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    
    <bean id="person" class="com.opgame.spring.Person">
        <property name="age" value="18"/>
        <property name="name" value="张三"/>
    </bean>

</beans>

使用person对象进行测试

@Test
public void testBean() {
    ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
    Person person = (Person)context.getBean("person");
    System.out.println(person);
}

1.2 使用annotation代替xml注册

创建一个带有@Configuration的java类,使用 @Bean进行注册

@Configuration // 告诉spring这个是一个config类
public class BeansConfig {

    // 给容器注册一个bean,类型为返回值类型,默认id为方法名,在bean注解中设置id
    @Bean("person")
    public Person person() {
        Person person = new Person();
        person.setAge(20);
        person.setName("李四");
        return person;
    }
}

使用person对象进行测试

@Test
public void testBean() {
    ApplicationContext context = new AnnotationConfigApplicationContext(BeansConfig.class);
    Person person = (Person)context.getBean("person");
    System.out.println(person);
}

2 exclude、include filter

在传统的spring mvc项目中为了使事务起效,我们通常在context.xml 进行exclude @Controller类,在mvc.xml 进行include @Controller,当我们使用一些第三方类库的时候,有时也需要对某些类进行排除扫码,那该怎么办呢?

2.1 传统xml排除与导入扫描类

context.xml(需要context命名空间)

<context:component-scan base-package="com.opgame">
    <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

mvx.xml

<!-- use-default-filters默认ture,默认扫描@Component @Repository @Service @Controller -->
    <context:component-scan base-package="com.opgame" use-default-filters="false">
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

2.2使用注解的形式排除与导入扫描类

@ComponentScan 注解使用

  • value:指定要扫描的包
  • excludeFilters = Filter[] :指定扫描的时候按照什么规则排除那些组件
  • includeFilters = Filter[] :指定扫描的时候只需要包含哪些组件
  • useDefaultFilters:是否使用默认的Filters

Filter

  • FilterType.ANNOTATION:按照注解
  • FilterType.ASSIGNABLE_TYPE:按照给定的类型
  • FilterType.ASPECTJ:使用ASPECTJ表达式
  • FilterType.REGEX:使用正则指定
  • FilterType.CUSTOM:使用自定义规则
  • classes:ANNOTATION、ASSIGNABLE_TYPE、CUSTOM 给定的类
  • pattern:ASPECTJ、REGEX 给定的表达式
@Configuration // 告诉spring这个是一个config类
@ComponentScan(value="com.opgame",useDefaultFilters=false,
includeFilters= {
        @Filter(type=FilterType.ANNOTATION,classes= {Controller.class})
},
excludeFilters= {
        @Filter(type=FilterType.ASSIGNABLE_TYPE,classes= {PersonController.class})
})
public class BeansConfig{
    
} 

自定义规则

// 实现org.springframework.core.type.filter.TypeFilter 接口
public class BeanTypeFilter implements TypeFilter{

    /**
     * @param metadataReader 获取被spring扫描到的当前类的元数据
     * @param metadataReaderFactory 获取元数据的工厂,用于获取其他类的信息
     */
    public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory)
            throws IOException {
        // 获取当前类注解元信息
        AnnotationMetadata annotationMetadata = metadataReader.getAnnotationMetadata();
        // 获取当前类的类信息
        ClassMetadata classMetadata = metadataReader.getClassMetadata();
        // 获取当前类的资源(类路径)
        Resource resource = metadataReader.getResource();
        
        // 如果当前类名含有person则配成功
        if(classMetadata.getClassName().contains("person")) {
            return true;
        }
        return false;
    }
}

3. 作用域与懒加载

3.1 xml

<!-- 默认scope="singleton" 单例-->
<!-- lazy-init="true" 当需要用到的时候才创建对象,默认为spring初始化时创建-->
<bean id="person" class="com.opgame.spring.Person" scope="singleton" lazy-init="true">
        <property name="age" value="18"/>
        <property name="name" value="张三"/>
</bean>

3.2 注解

@Lazy
@Scope("singleton")
@Bean("person")
public Person person() {
    Person person = new Person();
    person.setAge(20);
    person.setName("李四");
    return person;
}

4 按条件注册

在有的时候,我们编写框架,又或者编写代码,在不同环境时需要切换到不同的实现方法。那么在何时加载那种类的实现,spring为我们提供了这个注解

在spring中使用@Conditional 注解实现按需加载类。

@Conditional({
    LinuxCondition.class
})
public Person person1() {
    Person person = new Person();
    person.setAge(18);
    person.setName("李小四");
    return person;
}
// 实现 org.springframework.context.annotation.Condition 接口
public class LinuxCondition  implements Condition{
    
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        // 获取ioc使用的beanfactory
        ConfigurableListableBeanFactory beanFactory = context.getBeanFactory();
        // 获取类加载器
        ClassLoader classLoader = context.getClassLoader();
        // 获取当前环境
        Environment environment = context.getEnvironment();
        // 获取bean注册类
        BeanDefinitionRegistry registry = context.getRegistry();
        
        // 判断是否包含person对象
        boolean hasPerson = registry.containsBeanDefinition("person");
        if (!hasPerson) {
            return false;
        }
        
        // 判断是否属于linux 环境
        if(environment.getProperty("os.name").contains("linux")) {
            return true;
        }
        return false;
    }
}

在启动程序时设置 vm arguments 也可以查看Environment相关的笔记

-Dos.name=linux

也可以在

在spring boot中,使用@ConditionalOnMissingBean(name="xxx") @ConditionalOnMissingBean(xxx.class) 也可以判断当没有某些类或者方法的时候进行注册。

5. 使用import进行注册

现有多个类

class Blue {}

class Color {}

class Red {}

5.1 进行简单的注册

当我们需要无参注册的时候,可以使用@import 进行注册如:

@Configuration
@Import({Blue.class,Red.class})
public class BeansConfig {
    
}

其中bean id为类全名称 如com.opgame.spring.bean.Blue com.opgame.spring.bean.Red

相当于

@Configuration
public class BeansConfig {
    @Bean("com.opgame.spring.bean.Blue")
    public Blue blue() {
        return new Blue();
    }
    
    @Bean("com.opgame.spring.bean.Red")
    public Red red() {
        return new Red();
    }
}

5.2 使用ImportSelector 进行类名注册

// 自定义逻辑返回需要导入的组件
// 实现org.springframework.context.annotation.ImportSelector 接口
public class BeanImportSelector implements ImportSelector{

    /**
     * importingClassMetadata: 可以获取当前被标注@Import注解的所有注解元信息
     *     即当前例子中 BeansConfig.class的所有注解的元信息
     */
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        // 不允许返回null
        return new String[] {"com.opgame.spring.bean.Blue","com.opgame.spring.bean.Red"};
    }
}
@Configuration
@Import({BeanImportSelector.class})
public class BeansConfig {
    
}

5.3 ImportBeanDefinitionRegistrar 注册一个bean到容器中

此时注册的是BeanDefinition,也就是bean的定义信息,而spring 会根据bean的定义信息,进行bean的注册!也就是此时并没有注册bean!!!
// 实现org.springframework.context.annotation.ImportBeanDefinitionRegistrar接口
public class BeansImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar{

    /**
     * importingClassMetadata:当前config类的注解信息(beansConfig)
     * registry:Bean注册类,可以在此把所有需要添加到容器中的bean调用 BeanDefinitionRegistry.registerBeanDefinition()方法进行注册
     */
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
        boolean hasBlue = registry.containsBeanDefinition("com.opgame.spring.bean.Blue");
        boolean hasRed = registry.containsBeanDefinition("com.opgame.spring.bean.Red");
        // 如果有红色和蓝色,则注册一个id为color的 Color对象
        if (hasBlue && hasRed) {
            RootBeanDefinition beanDefinition = new RootBeanDefinition(Color.class);
            registry.registerBeanDefinition("color", beanDefinition);
        }
    }

}
@Configuration
@Import({Blue.class,Red.class,BeansImportBeanDefinitionRegistrar.class})
public class BeansConfig {
    
}

6 实现FactoryBean<?>接口注册bean

// 实现org.springframework.beans.factory.FactoryBean接口
public class PersonFactoryBean implements FactoryBean<Person>{

    // 构造person对象
    public Person getObject() throws Exception {
        return new Person();
    }

    // 获取当前对象的类型
    public Class<?> getObjectType() {
        return Person.class;
    }

    // 是否为单例
    public boolean isSingleton() {
        return true;
    }
}
@Configuration
public class BeansConfig {
    @Bean
    public PersonFactoryBean personFactoryBean() {
        return new PersonFactoryBean();
    }
}

使用person对象进行测试

@Test
public void testBean() {
    ApplicationContext context = new AnnotationConfigApplicationContext(BeansConfig.class);
    context.getBean("personFactoryBean").getClass(); // com.opgame.spring.Person
    
    // 获取该工厂构造除的bean对象需要&符号
    Person person = (Person)context.getBean("&personFactoryBean"); 
    
    // 不使用&则获取工厂本身
    PersonFactoryBean personFactoryBean = (PersonFactoryBean)context.getBean("personFactoryBean"); 
    
    System.out.println(person);
}

7 通过BeanDefinitionBuilder创建

BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(User.class);
builder.addPropertyValue("id", 1);
builder.addPropertyValue("name", "李四");
registry.registerBeanDefinition("user", builder.getBeanDefinition());

springboot实战电商项目mall4j (https://gitee.com/gz-yami/mall4j)

java商城系统源码


mall4j
26 声望0 粉丝