一 Spring
1.1 Spring IOC,AOP你的理解讲一下?
IOC,也经常被称为IOC容器,将之前手动new对象和手动对对象引用赋值,现在完全交给spring工厂。降低组件之间的耦合性。
AOP,面向切面编程。是对面向对象的一种补充,将一些重复性代码、业务无关的代码抽取,提高代理利用率,减少对业务代码的侵入。
1.2 Spring中Bean的生命周期(创建流程)是什么?
在容器进行初始化时候,首先将definetion转化为实例,然后进行依赖注入(包含属性和对象),然后有前置处理器执行前置处理器,如果配置了innit方法,执行init方法。有后置处理器,然后执行后置处理器。配置销毁方法,执行销毁方法。
1.3 Spring的作用域有哪些?
单例、多例、一次请求创建一个对象、每次会话创建一个对象。
1.4 Spring对象属性注入方式有哪些?
构造方法、set方法、静态工厂
1.5 Spring怎么解决循环依赖(对象在三级缓存中的迁移过程)?
比如a和b进行循环依赖了。
首先a进行实例化,执行get、doget、create、doCreate四个方法,进行对象a实例化,完成后将a放入三级缓存中,然后进行依赖注入,判断是否存在循环依赖。进行对象B的注入,执行get、doget。判断是否存在闭环,如果存在闭环,将对象a放进二级缓存中,并将三级缓存中a对象销毁。然后create、doCreate创建对象b,将对象a注入。将对象b放进一级缓存中。然后回退到a对象注入b对象,将a对象移到一级缓存中,这样a对象依赖b对象,b对象依赖a对象。
1.6 Spring缓存是怎么实现的?
主要是2个注解 @Cacheable 和@CacheEvict 。@Cacheable 可以用作类上和方法上。作用类上,将所有方法返回值放入缓存中,作用在方法上,将方法的返回值放到缓存中。以参数为key
1.7 静态代理和动态代理的区别,各自的优点和缺点?
静态代理是在编辑器就创建代理对象,执行时候效率高,但是灵活度差。动态代理在运行时候,在内存中创建代理对象,灵活度高,执行时候慢。
1.8为什么JDK动态代理需要实现一个接口?
因为JDK动态代理的class文件已经继承了Proxy,但是因为Java中单继承,所以必须实现一个接口。
1.9 JDK动态代理和cglib动态代理的区别?既然有没有实现接口都可以用CGLIB,为什么Spring还要使用JDK动态代理?
JDK是基于接口实现动态代理,cglib是基于实现类进行动态代理。
jdk动态代理是基于反射产生代理对象,cglib是基于AMS库产生字节码生成代理对象,在jdk1.6之前基于字节码的动态代理效率高于cglib动态代理。但是jdk1.7和jdk1.8对jdk动态代理进行优化,在执行次数较少情况下,反射产生动态代理比cglib产生动态代理效率高。
1.10 除了Spring AOP还有什么AOP的实现方案吗?
可以基于aspectj进行动态代理,是在编译器就生成动态代理,启动时候慢,但是执行时候效率高。
1.11 Spring事务有几种,实现的机制(@Transaction的底层原理)是什么,如果事务失败后怎么办?
两种:编程式事务、声明式事务。声明式事务是基于aop,拦截执行方法前后,将事务编织到执行方法中。在执行方法前加上事务,执行方法后加上事务提交和回滚。
@Transaction在bean进行初始化,扫描到注解,利用aop,拦截执行方法前后,将事务编织到执行方法中。
1.12 Spring事务传播机制有哪些?以及每一种机制事务管理员对事务协调员的影响?
7种:
required :事务管理员有事务,事务协调员执行事务。管理员无,协调员自己创建。
requires_new:事务管理员是否有事务,协调员自己创建事务。
support:管理员有事务协调员就用,管理员无事务协调员就不用。
no_support:管理员是否有事务,协调员都不用事务。
mandatory:管理员有事务就协调员使用事务,管理员无事务就报错
never: 管理员没有事务协调员不用事务,管理员有事务报错
nested: 回滚到指定位置。
1.13 FactoryBean 和 BeanFactory 区别?
FactoryBean 是一个对象,对bean进行装饰。BeanFactory是spring工厂的顶级父类。
1.14 SpringMVC的工作原理?
浏览器发送url,前端控制器接受请求,然后转发到执行器映射器,然后执行链给前端控制器,前端控制器请求给处理器适配器,适配器交给处理器进行处理,然后modulAndView给前端控制器,前端控制器将modulAndView传给视图解析器,视图解析器解析响应给前端控制器View。前端控制器将view传给视图渲染,视图渲染好,通过前端控制器响应给浏览器。
1.15 BeanFactory和Application区别
BeanFactory是工厂的顶级父类,定义了bean文件读取、实例化、维护对象的关系等。ApplicationContext除了继承上述功能还包含国际化、时间的监听等功能。
BeanFactory产生对象是懒加载,在获取对象时候,才开始去实例化对象,如果配置文件有问题,在获取时候才能发现问题。Application在初始化时候就创建对象。编译时候对配置文件进行加检查、启动慢,执行效率好。
二 Mybatis
2.1 MyBatis 中${}和#{}的区别?
#{}能防止sql注入问题,${}不能防止sql注入问题。因为#{}在sql进行解析时候,进行预编译,#{}替换层?占位符。${}在sql进行解析时候,直接进行参数替换。
2.2 既然${}不安全,为什么还需要${},什么时候会用到它?
但是在sql中表名词和order by进行排序等,需要的表名和字段名不需要’’包裹,必须用${}。
2.3 为什么预编译后能阻止sql注入?
进行预编译的时候,将mybatis将含有占位符的sql交给mysql进行处理,对sql进行解析和优化。然后再进行参数赋值。然后执行,即在执行时候不会对sql进行编译。
2.4 MyBatis的接口是怎么和XML文件联系上的?
mybatis开始加载时,xml配置文件加载成流,利用XParamPath解析器将流转化为document对象。然后开始解析document对象,解析配置文件放到cofigeration。然后获取sql的xml文件,加载流,然后将流转化为document。然后对sql的xml文件进行解析,并将sql封装成Sqlsource对象,并封装到cofigeration。然后SqlSessionFactoryBuiler根据cofigeration创建SqlSessionFactory,SqlSessionFactory再产生SqlSession对象。接口的查询方法,就是执行代理对象的invork方法,最终执行SqlSession的查询方法即接口和xml文件建立联系。
2.5 MyBatis 执行一个 Select 查询的流程?
实行查询方法,进入Executor的查询方法,取出SqlBould,如果开启了二级缓存,使用CachingExecutor的查询方法,先看二级缓存中是否有数据,有直接返回,如果没有调用被装饰的SimpleExecutor,然后插叙一级缓存,如果一级缓存中有数据直接响应数据,无数据然后进行数据库查询。首先使用parameHandler对参数进行设置,然后StatementHandler执行sql,ResultSetHandler对结果映射,响应list集合。
2.6 Mybatis如何不使用MySQL语句,如何实现分页?
使用RowBounds对象进行分页,对查询的结果集再内存中分页,即是非物理分页。(RowBounds对象考察)
2.7 Mybatis如何实现一个插件?
实现Mybatis的Interceptor接口,重写Interceptor方法。
2.8 Mybatis插件实现原理是什么?
在创建ParameHandler、ResultSetHandler、StatementHandler、Executor等创建对象时,先获取所有实现Interceptor接口,插件机制为四大对象创建代理对象,代理对象就可以拦截四大对象的每一个执行。
2.9 Mybatis有哪些Executor?
CachingExecutor缓存器处理器(二级缓存)
BaseExecutor基础处理器(一级缓存)
SimpleExecutor简单处理器(默认处理器)
ReseseExecutor可重用处理器
BathExecutor批量处理器。
2.10 Mybatis二级缓存机制?
一级缓存器基于hashMap,作用域是session,默认时开启的。
二级处理器基于hashMap,作用域是namespace,默认是关闭的。在sql的xml文件配置
在进行新增、编辑、删除会清理缓存。
但是二级缓存和一级缓存和JVM内存在一个地方,所以缓存过多影响性能。而且缓存在本地虚拟机中,会有分布式数据不一致问题。
2.11 Mybatis是否支持延迟加载?如果支持,它的实现原理是什么?
支持延迟加载,如关联对象的查询和关联对象集合查询。
利用cglib生成代理对象,调用目标方法,先进入目标方法的invork方法。比如a.getB().getName(),执行a.getB(),发现b对象为空,然后获取之前保存的sql,进行查询,然后对b进行赋值,再调用getName()。完成。