SpringMVC bean加载控制

软件发布|下载排行|最新软件

当前位置:首页IT学院IT技术

SpringMVC bean加载控制

mzz124   2022-09-29 我要评论

1、Spring配置类排除加载SpringMVC的bean

SpringMVC 通常只需要加载 controller 包内的 bean,而 Spring 需要加载 dao 和 service 包内的 bean,为了省事,Spring 配置类经常设置扫描的包为一个大范围的包(包含 dao 和 service 在内的包),此时 Spring 会错误或者多余地加载到 controller 包内的 bean

使 Spring 排除加载 SpringMVC 的 bean 一般有两种方法

  • 方法 1:扫描包时精确扫描需要的包,而不用大范围的扫描
  • 方法 2:大范围扫描,但是将不需要的 bean 过滤掉

方法 1 示例:

@Configuration
// 使 Spring 只扫描 dao 与 service 包
@ComponentScan({"com.mzz.dao", "com.mzz.service"})
public class SpringConfig {
}

方法 2:在 @ComponentScan 注解中可以设置 excludeFilters 属性排除掉 不需要的 bean,excludeFilters 属性的值是一个 Filter 注解,这个注解也在 ComponentScan 内,需要设置 type(过滤类型) 和 classes

下面的示例表示 Spring 的扫描范围为 com.mzz 但是按注解类型过滤掉所有注解了 @Controller 的 bean

@Configuration
@ComponentScan(value = "com.mzz",
    excludeFilters = @ComponentScan.Filter(
            type = FilterType.ANNOTATION,   // 按照 注解 来进行过滤
            classes = {Controller.class}    // 过滤所有注解了 @Controller 的 bean
    )
)
public class SpringConfig {
}

由于 Filter 注解中 type 的默认值就是 FilterType.ANNOTATION,所以上面的代码可以简写成这样:

@Configuration
@ComponentScan(value = "com.mzz",
    excludeFilters = @ComponentScan.Filter({Controller.class})
)
public class SpringConfig {
}

[补充]

当 Spring 和 SpringMVC 的配置类同时存在时,上述的方法二可能不会生效,原因是 Spring 进行大范围扫描时可以扫描到 SpringMVC 的配置类,于是 SpringMVC 中的 bean 也被加载到 Spring 中

此时也有两种解决方法

一是将 SpringMVC 的配置类移动到 Spring 扫描不到的包下(比如 com 包下)

二是 Spring 过滤时也将 SpringMVC 的配置类过滤掉,SpringMVC 的配置类会注解 @Configuration 和 @ComponentScan,可以对其进行过滤

@Configuration
@ComponentScan(value = "com.mzz",
        excludeFilters = @ComponentScan.Filter({Controller.class, ComponentScan.class})
)
public class SpringConfig {
}

2、Servlet容器配置类简洁开发

public class ServletContainerInitConfig extends AbstractDispatcherServletInitializer {
    // 加载 springMVC 容器配置
    @Override
    protected WebApplicationContext createServletApplicationContext() {
        AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
        ctx.register(SpringMvcConfig.class);
        return ctx;
    }
    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
    }
    // 加载 spring 容器配置
    @Override
    protected WebApplicationContext createRootApplicationContext() {
        AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
        ctx.register(SpringConfig.class);
        return ctx;
    }
}

上面的类继承自 AbstractDispatcherServletInitializer 抽象类,改为继承 AbstractAnnotationConfigDispatcherServletInitializer 抽象类。则只需将配置类 class 返回即可,而不需要手动创建 Context,如下

public class ServletContainerInitConfig extends AbstractAnnotationConfigDispatcherServletInitializer {
    @Override
    protected Class<?>[] getRootConfigClasses() {
        return new Class[]{SpringConfig.class};
    }
    // 加载 springMVC 容器配置
    @Override
    protected Class<?>[] getServletConfigClasses() {
        return new Class[]{SpringMvcConfig.class};
    }
    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
    }
}

视频链接:传送门

Copyright 2022 版权所有 软件发布 访问手机版

声明:所有软件和文章来自软件开发商或者作者 如有异议 请与本站联系 联系我们