
<p class="f_center"><img src="https://nimg.ws.126.net/?url=http%3A%2F%2Fdingyue.ws.126.net%2F2024%2F0118%2Fdbd7a354j00s7gii7002dd200rs00ijg00id00c8.jpg&thumbnail=660x&quality=80&type=jpg"/><br/>作 者 | 淘 苏<br/></p><p id="2DO5DVN1">导语:对技术人来说,面试成功的道路只有一条,就是好好准备技术基础。本文是面试系列文章Spring篇,作者把自己的八股文和一些经验总结汇总在一起,供大家参考。</p><p id="2DO5DVN3"><strong>设计思想&Beans</strong></p><p id="2DO5DVN5"><strong>1 IOC 控制反转</strong></p><p id="2DO5DVN7">IoC(Inverse of Control:控制反转)是⼀种设计思想,就是将原本在程序中⼿动创建对象的控制权,交由Spring框架来管理。IoC 在其他语⾔中也有应⽤,并⾮ Spring 特有。</p><p id="2DO5DVN9">IoC 容器是 Spring⽤来实现 IoC 的载体, IoC 容器实际上就是个Map(key,value),Map 中存放的是各种对象。将对象之间的相互依赖关系交给 IoC 容器来管理,并由 IoC 容器完成对象的注⼊。这样可以很⼤程度上简化应⽤的开发,把应⽤从复杂的依赖关系中解放出来。IoC 容器就像是⼀个⼯⼚⼀样,当我们需要创建⼀个对象的时候,只需要配置好配置⽂件/注解即可,完全不⽤考虑对象是如何被创建出来的。</p><p id="2DO5DVNB"><strong>DI 依赖注入</strong></p><p id="2DO5DVND">DI:(Dependancy Injection:依赖注入)站在容器的角度,将对象创建依赖的其他对象注入到对象中。</p><p id="2DO5DVNF"><strong>2 AOP 动态代理</strong></p><p id="2DO5DVNH">AOP(Aspect-Oriented Programming:⾯向切⾯编程)能够将那些与业务⽆关,却为业务模块所共同调⽤的逻辑或责任(例如事务处理、⽇志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可拓展性和可维护性。</p><p id="2DO5DVNJ">Spring AOP就是基于动态代理的,如果要代理的对象,实现了某个接⼝,那么Spring AOP会使⽤JDKProxy,去创建代理对象,⽽对于没有实现接⼝的对象,就⽆法使⽤ JDK Proxy 去进⾏代理了,这时候Spring AOP会使⽤基于asm框架字节流的Cglib动态代理 ,这时候Spring AOP会使⽤ Cglib ⽣成⼀个被代理对象的⼦类来作为代理。</p><p id="2DO5DVNL"><strong>3 Bean 生命周期</strong></p><p id="2DO5DVNN">单例对象:singleton</p><p id="2DO5DVNP">总结:单例对象的生命周期和容器相同</p><p id="2DO5DVNR">多例对象:prototype</p><p id="2DO5DVNT">出生:使用对象时spring框架为我们创建</p><p id="2DO5DVNV">活着:对象只要是在使用过程中就一直活着</p><p id="2DO5DVO1">死亡:当对象长时间不用且没有其它对象引用时,由java的垃圾回收机制回收</p><p class="f_center"><img src="https://nimg.ws.126.net/?url=http%3A%2F%2Fdingyue.ws.126.net%2F2024%2F0118%2F214c1349j00s7gii8003wd200hg0073g00id007g.jpg&thumbnail=660x&quality=80&type=jpg"/><br/></p><p id="2DO5DVO4">IOC容器初始化加载Bean流程:</p><p id="2DO5DVO6"><br/></p><p id="2DO5DVO8">总结:<br/></p><p id="2DO5DVOA"><strong>四个阶段</strong></p><p><ul><li id="2DO5DVVN"></p><p id="2DO5DVOE">实例化 Instantiation</p><p></li><li id="2DO5DVVO"></p><p id="2DO5DVOF">属性赋值 Populate</p><p></li><li id="2DO5DVVP"></p><p id="2DO5DVOG">初始化 Initialization</p><p></li><li id="2DO5DVVQ"></p><p id="2DO5DVOH">销毁 Destruction</p><p></li></ul><br/></p><p id="2DO5DVOI"><strong>多个扩展点</strong></p><p><ul><li id="2DO5DVVR"></p><p id="2DO5DVOM">影响多个Bean</p><p></li><li style="list-style: none; display: inline"></p><p><ul><li id="2DO5DVVS"></p><p id="2DO5DVOP">BeanPostProcessor</p><p></li><li id="2DO5DVVT"></p><p id="2DO5DVOQ">InstantiationAwareBeanPostProcessor</p><p></li></ul></p><p></li><li id="2DO5DVVU"></p><p id="2DO5DVOT">影响单个Bean</p><p></li><li style="list-style: none; display: inline"></p><p><ul><li id="2DO5DVVV"></p><p id="2DO5DVP0">Aware</p><p></li></ul></p><p></li></ul><br/></p><p id="2DO5DVP3"><strong>完整流程</strong></p><p id="2DO5DVP7">1.实例化一个Bean--也就是我们常说的new;</p><p id="2DO5DVP8">2.按照Spring上下文对实例化的Bean进行配置--也就是IOC注入;</p><p id="2DO5DVPB">3.如果这个Bean已经实现了BeanNameAware接口,会调用它实现的setBeanName(String)方法,也就是根据就是Spring配置文件中Bean的id和name进行传递;</p><p id="2DO5DVPC">4.如果这个Bean已经实现了BeanFactoryAware接口,会调用它实现setBeanFactory(BeanFactory)也就是Spring配置文件配置的Spring工厂自身进行传递;</p><p id="2DO5DVPD">5.如果这个Bean已经实现了ApplicationContextAware接口,会调用setApplicationContext(ApplicationContext)方法,和4传递的信息一样但是因为ApplicationContext是BeanFactory的子接口,所以更加灵活;</p><p id="2DO5DVPE">6.如果这个Bean关联了BeanPostProcessor接口,将会调用postProcessBeforeInitialization()方法,BeanPostProcessor经常被用作是Bean内容的更改,由于这个是在Bean初始化结束时调用那个的方法,也可以被应用于内存或缓存技术;</p><p id="2DO5DVPF">7.如果Bean在Spring配置文件中配置了init-method属性会自动调用其配置的初始化方法。</p><p id="2DO5DVPG">8.如果这个Bean关联了BeanPostProcessor接口,将会调用postProcessAfterInitialization(),打印日志或者三级缓存技术里面的bean升级;</p><p id="2DO5DVPH">9.以上工作完成以后就可以应用这个Bean了,那这个Bean是一个Singleton的,所以一般情况下我们调用同一个id的Bean会是在内容地址相同的实例,当然在Spring配置文件中也可以配置非Singleton,这里我们不做赘述。</p><p id="2DO5DVPI">10.当Bean不再需要时,会经过清理阶段,如果Bean实现了DisposableBean这个接口,或者根据spring配置的destroy-method属性,调用实现的destroy()方法</p><p id="2DO5DVPM"><strong>4 Bean 作用域</strong></p><p class="f_center"><img src="https://nimg.ws.126.net/?url=http%3A%2F%2Fdingyue.ws.126.net%2F2024%2F0118%2F64746f18j00s7gii9002hd200jk0075g00it006v.jpg&thumbnail=660x&quality=80&type=jpg"/><br/></p><p id="2DO5DVPP">默认作用域是singleton,多个线程访问同一个bean时会存在线程不安全问题</p><p id="2DO5DVPR"><strong>保障线程安全方法:</strong></p><p id="2DO5DVPV">1.在Bean对象中尽量避免定义可变的成员变量(不太现实);</p><p id="2DO5DVQ0">2.在类中定义⼀个ThreadLocal成员变量,将需要的可变成员变量保存在 ThreadLocal 中;</p><p id="2DO5DVQ3"><strong>ThreadLocal:</strong></p><p id="2DO5DVQ5">每个线程中都有一个自己的ThreadLocalMap类对象,可以将线程自己的对象保持到其中,各管各的,线程可以正确的访问到自己的对象。</p><p id="2DO5DVQ7">将一个共用的ThreadLocal静态实例作为key,将不同对象的引用保存到不同线程的ThreadLocalMap中,然后在线程执行的各处通过这个静态ThreadLocal实例的get()方法取得自己线程保存的那个对象,避免了将这个对象作为参数传递的麻烦。</p><p id="2DO5DVQ9"><strong>5 循环依赖</strong></p><p id="2DO5DVQB">循环依赖其实就是循环引用,也就是两个或者两个以上的 Bean 互相持有对方,最终形成闭环。比如A 依赖于B,B又依赖于A</p><p id="2DO5DVQD">Spring中循环依赖场景有:</p><p><ul><li id="2DO5E000"></p><p id="2DO5DVQH">prototype 原型 bean循环依赖</p><p></li><li id="2DO5E001"></p><p id="2DO5DVQI">构造器的循环依赖(构造器注入)</p><p></li><li id="2DO5E002"></p><p id="2DO5DVQJ">Field 属性的循环依赖(set注入)</p><p id="2DO5DVQK">其中,构造器的循环依赖问题无法解决,在解决属性循环依赖时,可以使用懒加载,spring采用的是提前暴露对象的方法。</p><p></li></ul><br/></p><p id="2DO5DVQM"><strong>懒加载@Lazy解决循环依赖问题</strong></p><p id="2DO5DVQO">Spring 启动的时候会把所有bean信息(包括XML和注解)解析转化成Spring能够识别的BeanDefinition并存到Hashmap里供下面的初始化时用,然后对每个 BeanDefinition 进行处理。普通 Bean 的初始化是在容器启动初始化阶段执行的,而被lazy-init=true修饰的 bean 则是在从容器里第一次进行context.getBean() 时进行触发。</p><p id="2DO5DV"><strong>三级缓存解决循环依赖问题</strong></p><p class="f_center"><img src="https://nimg.ws.126.net/?url=http%3A%2F%2Fdingyue.ws.126.net%2F2024%2F0118%2F07f07b21j00s7giia00d9d200u000dlg00it008i.jpg&thumbnail=660x&quality=80&type=jpg"/><br/></p><p id="2DO5DVQV">1.Spring容器初始化ClassA通过构造器初始化对象后提前暴露到Spring容器中的singletonFactorys(三级缓存中)。</p><p id="2DO5DVR0">2.ClassA调用setClassB方法,Spring首先尝试从容器中获取ClassB,此时ClassB不存在Spring 容器中。</p><p id="2DO5DVR1">3.Spring容器初始化ClassB,ClasssB首先将自己暴露在三级缓存中,然后从Spring容器一级、二级、三级缓存中一次中获取ClassA 。</p><p id="2DO5DVR2">4.获取到ClassA后将自己实例化放入单例池中,实例 ClassA通过Spring容器获取到ClassB,完成了自己对象初始化操作。</p><p id="2DO5DVR3">5.这样ClassA和ClassB都完成了对象初始化操作,从而解决了循环依赖问题。</p><p id="2DO5DVR6"><strong>Spring 注解</strong></p><p id="2DO5DVR8"><strong>1 @SpringBoot</strong></p><p id="2DO5DVRA"><strong>声明bean的注解</strong></p><p id="2DO5DVRC">@Component 通⽤的注解,可标注任意类为 Spring 组件</p><p id="2DO5DVRD">@Service 在业务逻辑层使用(service层)</p><p id="2DO5DVRE">@Repository 在数据访问层使用(dao层)</p><p id="2DO5DVRF">@Controller 在展现层使用,控制器的声明(controller层)</p><p id="2DO5DVRH"><strong>注入bean的注解</strong></p><p id="2DO5DVRJ">@Autowired:默认按照类型来装配注入,@Qualifier:可以改成名称</p><p id="2DO5DVRK">@Resource:默认按照名称来装配注入,JDK的注解,新版本已经弃用</p><p id="2DO5DVRM"><strong>@Autowired注解原理</strong></p><p id="2DO5DVRO">@Autowired的使用简化了我们的开发,</p><p id="2DO5DVRQ">实现 AutowiredAnnotationBeanPostProcessor 类,该类实现了 Spring 框架的一些扩展接口。</p><p id="2DO5DVRR">实现 BeanFactoryAware 接口使其内部持有了 BeanFactory(可轻松的获取需要依赖的的 Bean)。</p><p id="2DO5DVRS">实现 MergedBeanDefinitionPostProcessor 接口,实例化Bean 前获取到 里面的 @Autowired 信息并缓存下来;</p><p id="2DO5DVRT">实现 postProcessPropertyValues 接口, 实例化Bean 后从缓存取出注解信息,通过反射将依赖对象设置到 Bean 属性里面。</p><p id="2DO5DVRU"><strong>@SpringBootApplication</strong></p><p id="2DO5DVS0"><br/></p><p id="2DO5DVS2">@SpringBootApplication注解等同于下面三个注解:<br/></p><p><ul><li id="2DO5E003"></p><p id="2DO5DVS6">@SpringBootConfiguration: 底层是Configuration注解,说白了就是支持JavaConfig的方式来进行配置</p><p></li><li id="2DO5E004"></p><p id="2DO5DVS7">@EnableAutoConfiguration:开启自动配置功能</p><p></li><li id="2DO5E005"></p><p id="2DO5DVS8">@ComponentScan:就是扫描注解,默认是扫描当前类下的package</p><p></li></ul><br/></p><p id="2DO5DVSA">其中@EnableAutoConfiguration是关键(启用自动配置),内部实际上就去加载META-INF/spring.factories文件的信息,然后筛选出以EnableAutoConfiguration为key的数据,加载到IOC容器中,实现自动配置功能!</p><p id="2DO5DVSC">它主要加载了@SpringBootApplication注解主配置类,这个@SpringBootApplication注解主配置类里边最主要的功能就是SpringBoot开启了一个@EnableAutoConfiguration注解的自动配置功能。</p><p id="2DO5DVSE"><strong>@EnableAutoConfiguration作用:</strong></p><p id="2DO5DVSG">它主要利用了一个</p><p id="2DO5DVSI">EnableAutoConfigurationImportSelector选择器给Spring容器中来导入一些组件。</p><p><br/><br/></p><p id="2DO5DVSL"><strong>2 @SpringMVC</strong></p><p><br/><br/></p><p id="2DO5DVSN"><strong>SpringMVC 原理</strong></p><p class="f_center"><img src="https://nimg.ws.126.net/?url=http%3A%2F%2Fdingyue.ws.126.net%2F2024%2F0118%2F207e3bb3j00s7giib00cyd200sd00evg00it009u.jpg&thumbnail=660x&quality=80&type=jpg"/><br/></p><p id="2DO5DVSR">1.客户端(浏览器)发送请求,直接请求到 DispatcherServlet 。</p><p id="2DO5DVSS">2.DispatcherServlet 根据请求信息调⽤ HandlerMapping ,解析请求对应的 Handler 。</p><p id="2DO5DVST">3.解析到对应的 Handler (也就是 Controller 控制器)后,开始由HandlerAdapter 适配器处理。</p><p id="2DO5DVSU">4.HandlerAdapter 会根据 Handler 来调⽤真正的处理器开处理请求,并处理相应的业务逻辑。</p><p id="2DO5DVSV">5.处理器处理完业务后,会返回⼀个 ModelAndView 对象, Model 是返回的数据对象</p><p id="2DO5DVT0">6.ViewResolver 会根据逻辑 View 查找实际的 View 。</p><p id="2DO5DVT1">7.DispaterServlet 把返回的 Model 传给 View (视图渲染)。</p><p id="2DO5DVT2">8.把 View 返回给请求者(浏览器)</p><p id="2DO5DVT6"><strong>3 @SpringMybatis</strong></p><p><br/><br/></p><p id="2DO5DVT9"><strong>mybatis如何防止sql注入?</strong><br/></p><p id="2DO5DVTB">简单的说就是#{}是经过预编译的,是安全的,</p><p id="2DO5DVTD"> {}是未经过预编译的,仅仅是取变量的值,是非安全的,存在SQL注入。在编写mybatis的映射语句时,尽量采用“#{xxx}”这样的格式。如果需要实现动态传入表名、列名,还需要做如下修改:添加属性statementType="STATEMENT",同时sql里的属有变量取值都改成 {}是未经过预编译的,仅仅是取变量的值,是非安全的,存在SQL注入。在编写mybatis的映射语句时,尽量采用“#{xxx}”这样的格式。如果需要实现动态传入表名、列名,还需要做如下修改:添加属性statementType="STATEMENT",同时sql里的属有变量取值都改成 {xxxx}</p><p id="2DO5DVTF"><strong>Mybatis和Hibernate的区别</strong></p><p id="2DO5DVTH"><strong>Hibernate 框架:</strong></p><p id="2DO5DVTJ">Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,建立对象与数据库表的映射。是一个全自动的、完全面向对象的持久层框架。</p><p id="2DO5DVTL"><strong>Mybatis框架:</strong></p><p id="2DO5DVTN">Mybatis是一个开源对象关系映射框架,原名:ibatis,2010年由谷歌接管以后更名。是一个半自动化的持久层框架。</p><p id="2DO5DVTP"><strong>区别:</strong></p><p id="2DO5DVTR"><strong>开发方面</strong></p><p id="2DO5DVTT">在项目开发过程当中,就速度而言:</p><p id="2DO5DVTV">hibernate开发中,sql语句已经被封装,直接可以使用,加快系统开发;</p><p id="2DO5DVU1">Mybatis 属于半自动化,sql需要手工完成,稍微繁琐;</p><p id="2DO5DVU3">但是,凡事都不是绝对的,如果对于庞大复杂的系统项目来说,复杂语句较多,hibernate 就不是好方案。</p><p id="2DO5DVU5"><strong>sql优化方面</strong></p><p id="2DO5DVU7">Hibernate 自动生成sql,有些语句较为繁琐,会多消耗一些性能;</p><p id="2DO5DVU8">Mybatis 手动编写sql,可以避免不需要的查询,提高系统性能;</p><p id="2DO5DVUA"><strong>对象管理比对</strong></p><p id="2DO5DVUC">Hibernate 是完整的对象-关系映射的框架,开发工程中,无需过多关注底层实现,只要去管理对象即可;</p><p id="2DO5DVUE">Mybatis 需要自行管理映射关系;</p><p id="2DO5DVUG"><strong>4 @Transactional</strong></p><p><br/><br/></p><p id="2DO5DVUJ">注意事项:</p><p id="2DO5DVUL">①事务函数中不要处理耗时任务,会导致长期占有数据库连接。</p><p id="2DO5DVUM">②事务函数中不要处理无关业务,防止产生异常导致事务回滚。</p><p id="2DO5DVUO"><strong>事务传播属性</strong></p><p id="2DO5DVUQ">1.REQUIRED(默认属性) 如果存在一个事务,则支持当前事务。如果没有事务则开启一个新的事务。</p><p id="2DO5DVUR">2.MANDATORY 支持当前事务,如果当前没有事务,就抛出异常。<br/></p><p id="2DO5DVUU">3.NEVER 以非事务方式执行,如果当前存在事务,则抛出异常。</p><p id="2DO5DVV1">4.NOT_SUPPORTED 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。</p><p id="2DO5DVV2">5.REQUIRES_NEW 新建事务,如果当前存在事务,把当前事务挂起。</p><p id="2DO5DVV3">6.SUPPORTS 支持当前事务,如果当前没有事务,就以非事务方式执行。</p><p id="2DO5DVV6">7. NESTED (局部回滚) 支持当前事务,新增Savepoint点,与当前事务同步提交或回滚。 <strong>嵌套事务一个非常重要的概念就是内层事务依赖于外层事务。外层事务失败时,会回滚内层事务所做的动作。而内层事务操作失败并不会引起外层事务的回滚。</strong></p><p id="2DO5DVV8"><strong>Spring 源码阅读</strong></p><p id="2DO5DVVA"><strong>1 Spring中的设计模式</strong></p><p id="2DO5DVVC">单例设计模式 : Spring 中的 Bean 默认都是单例的。</p><p id="2DO5DVVE">⼯⼚设计模式 : Spring使⽤⼯⼚模式通过 BeanFactory 、 ApplicationContext 创建bean 对象。</p><p id="2DO5DVVG">代理设计模式 : Spring AOP 功能的实现。</p><p id="2DO5DVVI">观察者模式:Spring 事件驱动模型就是观察者模式很经典的⼀个应⽤。</p><p id="2DO5DVVK">适配器模式:Spring AOP 的增强或通知(Advice)使⽤到了适配器模式、spring MVC 中也是⽤到了适配器模式适配 Controller 。</p>
讯享网

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请联系我们,一经查实,本站将立刻删除。
如需转载请保留出处:https://51itzy.com/kjqy/156665.html