[记录]Spring注入的对象类型

开篇

之前,在用spring编码调试的时候,有时候发现被自动注入的对象是原始类的对象,有时候是代理类的对象,那什么时候注入的原始类对象呢,有什么时候注入的是代理类的对象呢?心里就留下了这个疑问。后来再次看spring aop的时候变有了大胆的想法。

案例

先添加springboot依赖
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter</artifactId>
    <version>2.0.2.RELEASE</version>
</dependency>
添加测试的类
  • 添加Service1
package beldon.service;
public interface DemoService {
}
package beldon.service.impl;
import beldon.service.DemoService;
import org.springframework.stereotype.Service;
@Service
public class DemoServiceImpl implements DemoService {
}
  • 添加Service2
package beldon.service;
public interface Demo2Service {
    void asyncDemo();
}
package beldon.service.impl;
import beldon.service.Demo2Service;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
@Service
public class Demo2ServiceImpl implements Demo2Service {
    @Override
    @Async
    public void asyncDemo() {
        System.out.println("Demo2Service:" + Thread.currentThread().getName());
    }
}
  • 添加Service3
package beldon.service.impl;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
@Service
public class Demo3Service {
    @Async
    public void asyncDemo() {
        System.out.println("Demo3Service:" + Thread.currentThread().getName());
    }
}
  • Application
package beldon.proxycheck;
import beldon.service.Demo2Service;
import beldon.service.DemoService;
import beldon.service.impl.Demo3Service;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;
import javax.annotation.PostConstruct;

@SpringBootApplication
@EnableAsync
public class CheckApplication {
    @Autowired
    private DemoService demoService;
    @Autowired
    private Demo2Service demo2Service;
    @Autowired
    private Demo3Service demo3Service;
    @PostConstruct
    public void init() {
        System.out.println("------------");
        System.out.println("DemoService:"+demoService.getClass().getName());
        System.out.println("Demo2Service:"+demo2Service.getClass().getName());
        System.out.println("Demo3Service:"+demo3Service.getClass().getName());
        System.out.println("------------");
        demo2Service.asyncDemo();
        demo3Service.asyncDemo();
        System.out.println("CheckApplication:"+Thread.currentThread().getName());
    }
    public static void main(String[] args) {
        SpringApplication.run(CheckApplication.class);
    }
}
代码描述
  • 添加了3个service,DemoServiceDemo2Service是接口,有实现类。Demo3Service是没有接口,只有单一的类
  • Demo2ServiceDemo3ServiceasyncDemo()方法上有@Async注解
  • CheckApplication方法上有@EnableAsync,用来开启异步
运行结果
------------
DemoService:beldon.proxycheck.service.impl.DemoServiceImpl
Demo2Service:com.sun.proxy.$Proxy37
Demo3Service:beldon.proxycheck.service.impl.Demo3Service$$EnhancerBySpringCGLIB$$b4ca4e7c
------------
Demo2Service:SimpleAsyncTaskExecutor-1
CheckApplication:main
Demo3Service:SimpleAsyncTaskExecutor-2

结果可以看出DemoService是被注入的是原始类的对象,Demo2Service被注入的对象是jdk代理的对象,Demo3Service被注入的对象是cglib的代理对象

将注入的demo2Service改为实现类注入
@Autowired
private Demo2ServiceImpl demo2Service;

运行结果如下:

***************************
APPLICATION FAILED TO START
***************************

Description:

The bean 'demo2ServiceImpl' could not be injected as a 'beldon.service.impl.Demo2ServiceImpl' because it is a JDK dynamic proxy that implements:
    beldon.service.Demo2Service

Action:

Consider injecting the bean as one of its interfaces or forcing the use of CGLib-based proxies by setting proxyTargetClass=true on @EnableAsync and/or @EnableCaching.

上面错误描述的是demo2ServiceImpl是实现Demo2Service接口的一个jdk动态代理,不能直接被注入

强制使用cglib

修改CheckApplication中的@EnableAsync如下

@EnableAsync(proxyTargetClass = true)

运行结果如下:

------------
DemoService:beldon.proxycheck.service.impl.DemoServiceImpl
Demo2Service:beldon.proxycheck.service.impl.Demo2ServiceImpl$$EnhancerBySpringCGLIB$$c39af2f2
Demo3Service:beldon.proxycheck.service.impl.Demo3Service$$EnhancerBySpringCGLIB$$b074e2af
CheckApplication:main
Demo2Service:SimpleAsyncTaskExecutor-1
Demo3Service:SimpleAsyncTaskExecutor-2

上面结果是Demo2ServiceDemo3Service被注入的都是cglib代理类

结论

spring很多功能都是通过aop来实现,如果事务,缓存注解,异步、还有一些自定义的aop等等,而aop是通过动态代理来实现的,spring主要用到的动态代理有jdk的动态代理和cglib。

  • Spring 在没有使用aop的时候自动注入的时候是原始类型对象
  • 在发生aop的时候,若代理对象有实现接口,则默认会使用jdk动态代理
  • 在发生aop的时候,若代理对象没有实现接口,则默认会使用cglib动态代理
  • jdk动态代理必须有实现接口
  • 可以强制使用cglib来做spring动态代理

转载于:https://www.cnblogs.com/beldon/p/9270324.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Spring框架中,对象的创建是由Spring容器负责的。Spring容器会在启动时读取配置文件,根据配置文件中的信息创建对象,并将这些对象存储在容器中。Spring容器会根据配置文件中定义的创建规则,即Bean定义,来创建对应的对象。 在Spring中,对象的创建有以下几种方式: 1. 构造函数注入:通过调用对象的构造函数创建对象,并将依赖的其他对象或值通过构造函数的参数传递进来。 2. Setter方法注入:通过调用对象的Setter方法创建对象,并将依赖的其他对象或值通过Setter方法设置进来。 3. 工厂方法注入:通过调用工厂方法来创建对象,并将依赖的其他对象或值通过工厂方法的参数传递进来。 4. 静态工厂方法注入:与工厂方法注入类似,不同的是工厂方法是静态方法。 Spring容器在创建对象时,会根据配置文件中的信息来确定使用哪种方式来创建对象。一旦对象被创建并存储在容器中,其他对象可以通过依赖注入的方式来使用它。 ### 回答2: Spring是一个开源的Java企业级应用开发框架,它提供了一个强大的IOC(控制反转)容器,这个容器可以帮助我们管理和创建对象。 在使用Spring创建对象时,我们首先需要在配置文件中定义对象的信息,包括对象的类名、属性以及依赖关系等。然后,Spring容器会根据这些配置信息来创建对象Spring使用反射机制来实现对象的创建。当我们需要一个对象时,Spring会根据对象的类名使用反射机制来实例化一个对象,并将对象的属性自动注入。 通过Spring创建的对象,具有很高的灵活性和可扩展性。我们可以通过配置文件来动态地改变对象的属性值,而不需要修改源代码。这使得我们可以更方便地对对象进行管理和维护。 此外,Spring的IOC容器还提供了对象的依赖注入功能。它可以自动解决对象之间的依赖关系,并将依赖的对象注入到目标对象中。这使得对象之间的协作变得更加简单和松耦合。 总之,Spring是一个强大的对象创建和管理框架。它通过IOC容器实现对象的创建和管理,帮助我们更好地组织和维护对象。它的灵活性和可扩展性使得我们可以轻松地管理大量的对象,并且能够方便地修改和扩展对象的属性和行为。 ### 回答3: Spring创建对象是指通过Spring框架实现对象的创建和管理。在Spring中,对象的创建和管理不再由应用程序自己完成,而是交给Spring容器来管理。 首先,对象的创建是通过依赖注入实现的。在Spring中,对象之间的依赖关系是通过依赖注入来建立的,即通过配置文件或注解将对象之间的依赖关系告知Spring容器,然后由Spring容器根据依赖关系来创建和管理对象。这样,对象的创建和依赖关系将变得更加灵活和可配置。 其次,Spring容器采用单例模式来管理对象。在Spring容器中,对象一般是单例的,即每个对象在容器中只会创建一次,并且可以在整个应用程序中共享。这样可以减少对象的创建和销毁的开销,提高系统的性能和效率。 除了创建对象Spring还提供了AOP(面向切面编程)和IoC(控制反转)等功能。AOP可以在不修改原有代码的情况下,通过横向切割将一些相同的功能模块提取出来,例如日志记录、权限控制等,提高系统的可维护性和可重用性。而IoC则是将对象的创建和依赖关系的管理交给Spring容器来处理,通过配置文件或注解来实现对象的创建和依赖关系的注入。 综上所述,Spring创建对象是通过依赖注入实现的,采用单例模式来管理对象,并且提供了AOP和IoC等功能,使得对象的创建和管理更加灵活和可配置。这种方式能够提高系统的性能和效率,提高代码的可维护性和可重用性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值