Spring框架getBean()方法返回对象为什么只能转成接口对象,转换成接口的实例会报错?

问题的出现是因为有人问我,为什么他在学习Spring框架的时候,他在xml文件当中定义了一个Bean,最后在调用getBean()方法获取这个Bean的时候,必须转换成这个Bean对应的接口,而不能转换成这个接口的实现类。

我在网上一查,发现也有对应的问题,但是感觉对应的答案都不够正确,或者不够解答我的疑惑

 

 

 现在我们开始重现这个问题

第一步:创建接口和对应的实现类

//对应接口
public interface ICar {
    void move();
}

//对应的实现类
@Transactional
public class MyBenz implements ICar {

    @Override
    public void move() {
        System.out.println("my car");
    }
}

第二步:编写对应的配置文件

<?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:myname="http://www.example.org/schema/user" xmlns:tx="http://www.springframework.org/schema/tx"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans-4.2.xsd http://www.example.org/schema/user http://www.example.org/schema/user.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">


    <!-- 配置HikariDataSource数据源 -->
    <bean id = "dataSource" class = "com.zaxxer.hikari.HikariDataSource">
        <property name="username" value="root"></property>
        <property name="password" value="123456"></property>
        <property name="driverClassName" value="com.mysql.cj.jdbc.Driver"></property>
        <property name="jdbcUrl" value="jdbc:mysql://localhost:3307/my_oracle?serverTimezone=GMT"></property>
    </bean>

    <!-- 定义事务 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource" /><!-- ref:引入数据源  -->
    </bean>

    <!-- 配置 Annotation 驱动,扫描@Transactional注解的类定义事务启动事物注解 transaction-manager的值必须和上面这个bean的id一样-->
    <tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="false" />

    <!--配置bean,class属性我们设置为实现类-->
    <bean id="myBenz" class="com.fsl.springbootjunit.spring.test3.MyBenz"></bean>

</beans>

第三步:进行测试

public static void main(String[] args) {
        ApplicationContext bf = new ClassPathXmlApplicationContext("spring/applicationContext-beans.xml");
        MyBenz myBenz = (MyBenz)bf.getBean("myBenz");
        myBenz.move();
    }

结果如下,会出现java.lang.ClassCastException:

我们在配置文件当中配置的Bean的class属性明明就是com.fsl.springbootjunit.spring.test3.MyBenz,但是为什么我们getBean()的时候,转成对应的类型会报类型转换错误呢?

 

问题解答

首先我们要知道的是,Spring的AOP的实现底层是使用的代理模式,它就是通过创建对应的代理对象,通过对代理对象的方法运行前后进行处理来达到目的。我们使用的事务也是如此,给创建一个代理对象,在这个方法运行之前,Spring帮助开启事务,方法运行之后,Spring帮助我们提交事务或者回滚事务。

Spring创建代理对象的代码如下:

@Override
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
    //在创建代理对象之前进行一系列判断,来决定是使用JDK代理还是CGLIB代理
    //isOptimize:使用cglib代理是否使用激进的优化策略
    //isProxyTargetClass:是否对目标本身进行代理,而不是对目标类的接口进行代理
    //hasNoUserSuppliedProxyInterfaces:是否不存在代理接口
    if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
	    Class<?> targetClass = config.getTargetClass();
	    if (targetClass == null) {
			throw new AopConfigException("TargetSource cannot determine target class: " +
						"Either an interface or a target is required for proxy creation.");
		}
		if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {
			return new JdkDynamicAopProxy(config);
		}
		return new ObjenesisCglibAopProxy(config);
	}
	else {
		return new JdkDynamicAopProxy(config);
	}
}

在创建对象开始之前,会进行判断来决定使用JDK动态代理还是CGLIB动态代理,因为这两种代理模式对代理对象的要求不同,比如JDK动态代理就要求被代理的对象实现了某个接口。

其中有一个属性很重要,isProxyTargetClass,这个就用来判断,是否对目标本身进行代理,而这个属性,默认是false

现在我们就能够知道,原来代理对象在生成的过程当中,是去找到被代理对象的接口,依据那个接口来生成的对象,而不是依据目标本身生成的。

 ApplicationContext bf = new ClassPathXmlApplicationContext("spring/applicationContext-beans.xml");
        Object myBenz = bf.getBean("myBenz");
        //这里返回true,因为是依据这个接口生成的代理对象
        System.out.println(myBenz instanceof ICar);
        //这里返回false,因为这个是代理目标类,但是Spring默认的不以这个去生成代理对象
        System.out.println(myBenz instanceof MyBenz);

那假如我们就想以目标类去生成代理对象呢?

通过上面的分析,大家肯定也能够回答出来,我把proxyTargetClass属性设置为true就可以了。

<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true" />

在执行下面的这个代码,我们就能够看到对应的效果

ApplicationContext bf = new ClassPathXmlApplicationContext("spring/applicationContext-beans.xml");
Object myBenz = bf.getBean("myBenz");
//返回true
System.out.println(myBenz instanceof ICar);
//返回true
System.out.println(myBenz instanceof MyBenz);

 

 

  • 29
    点赞
  • 37
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
Spring中,我们可以通过ApplicationContext的getBean方法来获取已注入的对象方法实例。 首先,在我们的Spring配置文件中,我们需要定义并配置我们要注入的对象以及它们的依赖关系。这可以通过使用Bean标签和相关属性来实现。例如,我们可以使用<bean>标签定义一个名为"exampleService"的Bean,并指定它的class属性为我们所需的。我们还可以通过使用<property>标签来定义和注入该的属性依赖关系。 接下来,在我们的应用代码中,我们需要获取我们所注入对象方法实例。这可以通过使用ApplicationContext的getBean方法实现。这个方法接收一个参数,即我们所要获取的Bean的名称,然后返回与该名称关联的Bean的实例。 例如,如果我们想获取名为"exampleService"的Bean的实例,我们可以调用ApplicationContext的getBean方法,并将"exampleService"作为参数传递进去。得到的返回结果就是"exampleService"的实例。 在获取到Bean的实例后,我们就可以使用它来调用所需的方法了。例如,如果"exampleService"具有一个名为"doSomething"的方法,我们可以简单地调用该方法来执行所需的操作。 总而言之,通过ApplicationContext的getBean方法可以方便地获取已注入的对象方法实例。我们只需要提供Bean的名称,然后就可以获得相应的实例,并使用它来调用相应的方法。这种方式可以使我们的应用具有更好的灵活性和可扩展性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值