java 注入为空_@Autowired注入为null问题分析

问题说明

最近看到Spring事务,在学习过程中遇到一个很苦恼问题

搭建好Spring的启动环境后出现了一点小问题

在启动时候却出现[java.lang.NullPointerException]

不过因为当时一个小小的疏忽 很low的问题 请往下看 ...

工程结构

AAffA0nNPuCLAAAAAElFTkSuQmCC

代码片段

spring.xml

AAffA0nNPuCLAAAAAElFTkSuQmCC

AAffA0nNPuCLAAAAAElFTkSuQmCC

AAffA0nNPuCLAAAAAElFTkSuQmCC1 <?xml  version="1.0" encoding="UTF-8"?> 2 10 11     12     13 14     15     17         18         20         21         22     23 24     25     27         28     29 30     31     33         34     35 36 

AAffA0nNPuCLAAAAAElFTkSuQmCC

Spring.xml

Test.java

AAffA0nNPuCLAAAAAElFTkSuQmCC1 public class Test {2     public static void main(String[] args) {3         ClassPathXmlApplicationContext classPathXmlApplicationContext = new ClassPathXmlApplicationContext(4                 "spring.xml");5         ServiceIF service = (ServiceIF) classPathXmlApplicationContext.getBean("serviceImpl");6         service.add("小王", 23);7     }8 }

AAffA0nNPuCLAAAAAElFTkSuQmCC

TransactionUtil.java

AAffA0nNPuCLAAAAAElFTkSuQmCC

AAffA0nNPuCLAAAAAElFTkSuQmCC

AAffA0nNPuCLAAAAAElFTkSuQmCC1 @Component("transactionUtil") 2 public class TransactionUtil { 3  4     /** 5      * 初始化数据源 6      */ 7     @Autowired 8     private DataSourceTransactionManager dataSourceTransactionManager; 9 10     /**11      * 开启事务12      * 

13      * @return14      */15     public TransactionStatus begin() {16         TransactionStatus transaction = dataSourceTransactionManager.getTransaction(new DefaultTransactionDefinition());17         System.out.println(" 开启事务成功 ");18         return transaction;19     }20 21     /**22      * 提交事物23      *

24      * @param transaction25      */26     public void commit(TransactionStatus transaction) {27         dataSourceTransactionManager.commit(transaction);28         System.out.println(" 事物提交成功 ");29     }30 31     /**32      * 回滚事务33      *

34      * @param transaction35      */36     public void rollback(TransactionStatus transaction) {37         dataSourceTransactionManager.rollback(transaction);38         System.err.println(" 事物进行回滚 ");39     }40 }

AAffA0nNPuCLAAAAAElFTkSuQmCC

TransactionUtil.java

ServiceImpl.java

AAffA0nNPuCLAAAAAElFTkSuQmCC1 @Service("serviceImpl") 2 public class ServiceImpl implements ServiceIF { 3  4     @Autowired 5     TransactionUtil transactionUtil; 6  7     private TransactionStatus transactionStatus = null; 8  9     @Override10     public void add(String name, Integer age) {11         transactionStatus = transactionUtil.begin();12         try {13             new DaoImpl().add(name, age);14             transactionUtil.commit(transactionStatus);15         } catch (Exception e) {16             System.err.println("ERROR >>> 执行出现异常 即将进行回滚操作");17             transactionUtil.rollback(transactionStatus);18         }19     }20 }

AAffA0nNPuCLAAAAAElFTkSuQmCC

DaoImpl.java

AAffA0nNPuCLAAAAAElFTkSuQmCC1 public class DaoImpl implements DaoIF{ 2 

3     /** 4      * 注入jdbc模板类 5      */ 6     @Autowired 7     private JdbcTemplate jdbcTemplate; 8

9     /**10      * 第一条插入语句11      */12     private final String SQL_INSERT_01 = "insert into user values (?,?)";13

14     /**15      * 添加sql执行16      *

17      * @param name18      * @param age19      */20     public void add(String name, Integer age) {21         jdbcTemplate.update(SQL_INSERT_01, name, age);22     }23 }

AAffA0nNPuCLAAAAAElFTkSuQmCC

运行结果

AAffA0nNPuCLAAAAAElFTkSuQmCC

问题分析

AAffA0nNPuCLAAAAAElFTkSuQmCC

解决思路

我在想 为什么会没有注入进来呢 我明明加了@Autowired注解

后来猜到可能是Spring.xml配置的问题

看完也没有问题 我就从Java Source一步一步看 发现....

AAffA0nNPuCLAAAAAElFTkSuQmCC

我靠 我就猜测是不是如果用「new Object()」的方式创建实例后其class中的Bean的注解会失效呢?

然后我尝试在ServiceImpl.java中以注解的方式把DaoIF的实例注入到ServiceImpl,

并在DaoImpl.java的类上面添加@Repository,

把ServiceImpl.java中new DaoImpl()替换成注入的daoImpl。

改修代码

ServiceImpl.java修改后

AAffA0nNPuCLAAAAAElFTkSuQmCC

DaoImpl.java修改后

AAffA0nNPuCLAAAAAElFTkSuQmCC

改修后调试

AAffA0nNPuCLAAAAAElFTkSuQmCC

AAffA0nNPuCLAAAAAElFTkSuQmCC

其实我懂得也不太多 Spring注入的流程那

首先他会把项目中target -> classes 目录下的「.class」文件进行解析

通过Spring.xml中的「context:component-scan」进行注解扫描

如果这个路径下的「.class」文件的类上面是否存在@Component声明的注解

如果被此类注解修饰,Spring会把所有被注解修饰的bean进行实例化操作  供给@Autowired进行注入

(在spring注解的源码中@Service和@Repository等等都继承了@Component注解)

结论

在使用Spring的Bean容器时 千万要确保

配置的注解扫描路径正确

Jar的依赖是否存在

是否在bean的上面加「@Service @Repository @Component … 」

要细心 遇到异常不要紧 慢慢分析!!!

### 解决@Autowired注解导致Spring服务注入问题 当遇到`@Autowired`注解无法成功注入Service对象,即注入的结果为`null`的情况时,通常是因为以下几个原因造成的: #### 1. Bean未被扫描到 确保带有@Service或其他组件注解的类位于配置文件中定义的基础包及其子包下。只有这样,这些类才能被Spring上下文识别并注册成为Bean实例[^1]。 #### 2. 配置缺失@ComponentScan 如果自定义了基础包路径或者采用了模块化开发模式,则需确认已适当设置了`@ComponentScan`注解以覆盖所有需要扫描的包范围。 #### 3. 多个相同类型的Bean存在冲突 在同一作用域内如果有两个以上同类型的Bean,并且没有特别指明哪个应该优先考虑的话,可能会造成不确定的行为甚至失败。此时可以通过添加`@Primary`或联合使用`@Qualifier`来解决问题[^3]。 #### 4. 循环依赖问题 循环依赖是指A依赖于B而B又反过来依赖于A的情形,在某些版本的Spring中这可能导致初始化阶段出现问题。对于这种情况,建议重构设计减少不必要的相互依赖关系;也可以尝试调整加载顺序或是采用setter方法代替构造器来进行依赖注入[^2]。 #### 5. 使用不当的作用域 检查所使用的Scope是否合适,默认情况下singleton是最常见的选择。但如果选择了prototype或者其他特殊scope,则需要注意其生命周期管理以及如何正确获取最新的bean实例。 ```java // 正确示例:确保Service类上有相应注解并且处于正确的包结构之下 package com.example.service; import org.springframework.stereotype.Service; @Service public class MyServiceImpl implements MyService { // ... } ``` ```java // Controller层中的正确写法 package com.example.controller; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.RestController; import com.example.service.MyService; @RestController public class MyController { private final MyService myService; @Autowired public MyController(MyService myService){ this.myService = myService; } // 或者使用字段注入(虽然推荐构造函数注入) //@Autowired //private MyService myService; //... } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值