解决spring懒加载以及@PostConstruct结合的坑

2022-07-19,,,

目录
  • spring加载及@postconstruct的坑
    • 下面是一个初始化数据的组件
  • 遗留问题 @postconstruct注入不成功
    • 直接先说原因吧
      • 1.忽略ssm本身对注解是通过扫包才让注解有效的
      • 2.忽略@service的注解
      • 3.注意扫包区间
      • 4.@postcoustruct注解用于

spring懒加载及@postconstruct的坑

举例说明:

下面是一个初始化数据的组件

@component
public class initdata {
    /**
     * 初始化加载bean
     */
    @postconstruct
    public void init() {
        map<string, string> map = new hashmap<string, string>();
        for (int i=0;i<10;i++) {
            map.put(i+"", i+"");
        }
        //模拟加载一些别单例模式bean的数据初始化
        errormsgutil1.getinstance().setmap(map);
        errormsgutil2.getinstance().setmap(map);
    }

好了,如果你开启了spring的懒加载模式,而且 initdata这个bean只是被扫描而没有被注入,那么errormsgutil里的map永远是空的。

@postconstruct实在bean初始化的时候被创建的,开启了懒加载显然如果initdata没有被用到那么就一直不执行了。

此坑已踩,小弟还是对spring理解不深,继续学习。

ps:如何开启spring的懒加载模式,在spring.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" xmlns:p="http://www.springframework.org/schema/p"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemalocation="  
    http://www.springframework.org/schema/beans   
    http://www.springframework.org/schema/beans/spring-beans-3.0.xsd  
    http://www.springframework.org/schema/context  
    http://www.springframework.org/schema/context/spring-context-3.0.xsd  
    " default-lazy-init="true">

遗留问题 @postconstruct注入不成功

前两天做了个纯java代码的rabbitmq监听多个ip的客户端功能,由于用的不是配置方式的listener方式—博文中有这一节,无法自动启动。就用@postconstruct来项目启动时运行监听mq,但是老遇到调用业务逻辑层方法时,注入不成功导致空指针异常。今天排查了一下,发现主要问题是框架扫包忽略了。

直接先说原因吧

1.忽略ssm本身对注解是通过扫包才让注解有效的

<!-- 自动扫描该包,支持注解的层限制,把api这个controller层排除在外了。另外多个包中间用逗号或者分号隔开都可以。 -->
<context:component-scan base-package="com.**.service,com.**.action,com.**.common" >
    <context:include-filter type="annotation" expression="org.springframework.stereotype.controller"/>
    <context:exclude-filter type="annotation" expression="org.springframework.stereotype.api"/>
</context:component-scan>

2.忽略@service的注解

由于扫包是扫service层和action层(相当于service层),common工具层。所以在api层(相当于controller层)用@service和不用注解都是错误的,都会导致注入失败。

3.注意扫包区间

出了这个范围@postconstruct是无效的。应用在启动时是不会走带有这个注解的方法的。

4.@postcoustruct注解用于

在依赖关系注入完成之后需要执行的方法上,以执行任何初始化。此方法所在的类必须放入服务之前调用。也就是该注解的类上不能随便注解:经验总结是能用@service注解,不能用@controller注解,否则启动不会走这个方法。这个类定位为服务层/业务层。而不是控制层(web层)

有了上面说的注意点。我重新在工具类包common包中写了个测试类。然后spring配置文件上扫包范围增加了这个common包。代码如下:com.zhanglf.common.cache.commoncachemap.java

package com.zhanglf.common.cache;
import javax.annotation.postconstruct;
import javax.annotation.resource;
import org.springframework.stereotype.service;
import com.zlf.bo.staffbo;
import com.zlf.service.istaffservice;
@service("commoncachemap")
public class commoncachemap {
    @resource
    private istaffservice staffservice;
    @postconstruct
    public void getonestaff(){
        staffbo staffbo = staffservice.selectbyprimarykey("s01");
        system.out.println(staffbo.getname());
    }
}

结果是注入成功,运行结果如下:

这样@postconstruct注入问题就解决了。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。

《解决spring懒加载以及@PostConstruct结合的坑.doc》

下载本文的Word格式文档,以方便收藏与打印。