浅谈Spring框架

一、什么是Spring?

        Spring是一个开源框架,可以降低开发复杂度,提高开发效率,轻量级低耦合的框架。由于Spring的分层架构,可以自己选择整合其他组件,灵活性高

二、什么是IOC?

       IOC 叫做控制反转,指的是通过Spring来管理对象的创建、配置和生命周期,这样相当于把控制权交给了Spring,不需要人工来管理对象之间复杂的依赖关系,这样做的好处就是解耦。在Spring里面,主要提供了 BeanFactory 和 ApplicationContext 两种 IOC 容器,通过他们来实现对 Bean 的管理。

        BeanFactory采用延迟加载机制,初始化时间短,适用于资源有限或需要延迟加载的场景。ApplicationContext在启动时预先加载所有Bean对象,初始化时间较长,但在应用运行时能够更快地获取Bean对象,并提供了更多的功能和特性,适用于大多数应用场景

  1. BeanFactory(Bean工厂,顶层接口): BeanFactory是Spring框架中最基本的IOC容器接口。它是一种轻量级的容器,采用延迟加载(懒加载)机制,即在需要获取Bean时才进行实例化和初始化。由于延迟加载的特性,BeanFactory的初始化时间较短,节约了系统资源。BeanFactory提供了基本的IOC功能,包括Bean的实例化、配置、装配以及管理Bean之间的依赖关系。

  2. ApplicationContext: ApplicationContext是BeanFactory的子接口,也是Spring框架中更高级、功能更丰富的IOC容器。与BeanFactory相比,ApplicationContext在启动时会预先加载所有Bean对象,进行实例化和初始化。这使得ApplicationContext在应用运行时能够更快地获取Bean对象,提高了应用的性能。除了BeanFactory的功能外,ApplicationContext还提供了更多的特性,如国际化支持、事件传播、资源加载、AOP等。

三、什么是AOP?

        AOP即面向切面编程,可以在不改变原有代码的基础上对目标方法进行无侵入式增强。AOP 基于动态代理的方式实现,如果是实现了接口的话就会使用 JDK 动态代理,反之则使用 CGLIB 代理,Spring中 AOP 的应用主要体现在 事务、日志、异常处理等方面,通过在代码的前后做一些增强处理,可以实现对业务逻辑的隔离,提高代码的模块化能力,同时也是解耦。

四、JDK动态代理和CGliB代理区别有哪些?

Spring提供了两种方式来实现动态代理:JDK动态代理和CGLIB代理

JDK动态代理:JDK动态代理是基于接口的代理,通过反射机制动态生成代理类。当目标对象实现了接口时,Spring会使用JDK动态代理来创建代理对象。JDK动态代理通过Proxy类和InvocationHandler接口实现。在运行时,通过Proxy类的newProxyInstance()方法生成代理对象,同时传入一个实现了InvocationHandler接口的代理处理器对象,用于处理代理对象的方法调用。

// 定义UserService接口,包含addUser方法
public interface UserService {
    void addUser();
}

// UserServiceImpl实现了UserService接口,实现了addUser方法
public class UserServiceImpl implements UserService {
    @Override
    public void addUser() {
        System.out.println("Add user");
    }
}

// MyInvocationHandler实现了JDK动态代理的InvocationHandler接口,用于对目标方法进行增强
public class MyInvocationHandler implements InvocationHandler {
    private Object target; // 目标对象

    public MyInvocationHandler(Object target) {
        this.target = target; // 初始化目标对象
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("Before method call"); // 在方法调用前输出信息
        Object result = method.invoke(target, args); // 调用目标对象的原始方法
        System.out.println("After method call"); // 在方法调用后输出信息
        return result;
    }
}

public class Main {
    public static void main(String[] args) {
        UserService target = new UserServiceImpl(); // 创建目标对象
        MyInvocationHandler handler = new MyInvocationHandler(target); // 创建代理处理器
        UserService proxy = (UserService) Proxy.newProxyInstance(
                target.getClass().getClassLoader(), // 类加载器
                target.getClass().getInterfaces(), // 目标对象实现的接口
                handler // 代理处理器
        );
        proxy.addUser(); // 调用代理对象的方法,实际会执行增强后的逻辑
    }
}

CGLIB代理:CGLIB代理是基于继承的代理,当目标对象没有实现接口时,Spring会使用CGLIB代理来创建代理对象。CGLIB代理通过字节码技术生成目标对象的子类,并重写父类的方法来实现增强功能。

// UserService类定义了一个简单的方法 addUser
public class UserService {
    public void addUser() {
        System.out.println("Add user");
    }
}

// MyMethodInterceptor实现了CGLIB的MethodInterceptor接口,用于对目标方法进行增强
public class MyMethodInterceptor implements MethodInterceptor {
    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("Before method call"); // 在方法调用前输出信息
        Object result = proxy.invokeSuper(obj, args); // 调用被代理对象的原始方法
        System.out.println("After method call"); // 在方法调用后输出信息
        return result;
    }
}

public class Main {
    public static void main(String[] args) {
        Enhancer enhancer = new Enhancer(); // 创建Enhancer对象,用于生成代理类
        enhancer.setSuperclass(UserService.class); // 设置父类为UserService,生成的代理类会继承UserService
        enhancer.setCallback(new MyMethodInterceptor()); // 设置方法拦截器,用于对目标方法进行增强
        UserService proxy = (UserService) enhancer.create(); // 创建代理对象
        proxy.addUser(); // 调用代理对象的方法,实际会执行增强后的逻辑
    }
}

五、Spring中有哪些设计模式?

1.代理模式:

        所谓代理,是指它与被代理对象实现了相同的接口,客户端必须通过代理才能与被代理的目标类进行交互,而代理一般在交互的过程中(交互前后),进行某些特定的处理,比如在调用这个方法前做前置处理,调用这个方法后做后置处理。代理又分为静态代理和动态代理两种方式,Spring 的 AOP 采用的是动态代理的方式Spring 通过动态代理对类进行方法级别的切面增强,动态生成目标对象的代理类,并在代理类的方法中设置拦截器,通过执行拦截器中的逻辑增强了代理方法的功能,从而实现 AOP。

2.单例模式:

        单例模式是指一个类在整个系统运行过程中,只允许产生一个实例。在Spring中,Bean 可以被定义为两种模式:Prototype(多例)和Singleton(单例),Spring 默认是单例模式。那么Spring是如何实现单例模式的呢?答案是通过单例注册表的方式,具体来说就是使用了HashMap

public class DefaultSingletonBeanRegistry {
    
    //使用了线程安全容器ConcurrentHashMap,保存各种单实例对象
    private final Map singletonObjects = new ConcurrentHashMap;

    protected Object getSingleton(String beanName) {
    //先到HashMap中拿Object
    Object singletonObject = singletonObjects.get(beanName);
    
    //如果没拿到通过反射创建一个对象实例,并添加到HashMap中
    if (singletonObject == null) {
      singletonObjects.put(beanName,
                           Class.forName(beanName).newInstance());
   }
   
   //返回对象实例
   return singletonObjects.get(beanName);
  }
}

3.模板模式:

        主要是一些对数据库操作的类用到,比如 JdbcTemplate、JpaTemplate,因为查询数据库的建立连接、执行查询、关闭连接几个过程,非常适用于模板方法。

六、Spring为什么默认是单例?如何保证单例Bean线程安全?

1、为什么Spring默认是单例的?

        Spring默认采用延迟初始化(懒汉式加载)策略,即只有在第一次被请求时才会创建Bean实例。这样可以避免在容器启动过程中创建大量的对象,提高应用程序的启动性能

2、Spring中如何保证单例Bean的线程安全?

1.改变Bean的作用域

   通过将Bean的作用域改为原型,可以保证每次获取Bean都会返回一个新的实例,从而避免了多线程之间共享一个Bean实例的问题。具体的做法就是在Bean类上的@Bean注解中使用@scope("prototype")来指定Bean的作用域。

2.线程安全同步机制(加锁)

    对于一些需要共享状态的Bean,可以采取同步机制来保护共享状态,避免多个线程同时修改同一个Bean实例的状态。一般来说使用synchronized、Reentrantlock等关键字或锁对象来同步存取共享状态的代码。

3.使用ThreadLocal

   如果只有某一个Bean类的一部分需要进行同步,可以考虑将这部分“状态”信息移除到清秀级别的Threadlocal中,这样每个线程都会有自己的副本,避免多线程之间共享状态的问题。

4.使用并发集合类

  Java并发包中提供了很多线程安全的Concurrent类,例如ConcurrentHashMap、CopyOnWriteArrayList等。这些类内部实现了各种同步机制,可以保证多线程下的正确性。所以,在处理多线程环境下的共享Bean时,也可以使用这些线程安全类来替代普通的集合类,从而避免多个线程访问同一个Bean实例时产生的并发问题。

  • 17
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值