如何快速更新自己的技术积累?
- 在现有的项目里,深挖技术,比如用到netty可以把相关底层代码和要点都看起来。
- 如果不知道目前的努力方向,就看自己的领导或公司里技术强的人在学什么。
- 知道努力方向后不知道该怎么学,就到处去找相关资料然后练习。
- 学习以后不知道有没有学成,则可以通过面试去检验。
我个人觉得面试也像是一场全新的征程,失败和胜利都是平常之事。所以,劝各位不要因为面试失败而灰心、丧失斗志。也不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!
以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目
import org.springframework.util.StringValueResolver;
import org.springframework.web.context.ServletContextAware;
import javax.servlet.ServletContext;
@Component
public class AllAwareInterface implements BeanNameAware, BeanClassLoaderAware,
BeanFactoryAware, EnvironmentAware, EmbeddedValueResolverAware,
ResourceLoaderAware, ApplicationEventPublisherAware, MessageSourceAware,
ApplicationContextAware, ServletContextAware, LoadTimeWeaverAware, ImportAware {
@Override
public void setBeanName(String name) {
// BeanNameAware作用:让Bean对Name有知觉
//这个方法只是简单的返回我们当前的beanName,听官方的意思是这个接口更多的使用在spring的框架代码中,实际开发环境应该不建议使用
System.out.println(“1 我是 BeanNameAware 的 setBeanName 方法 —参数:name,内容:”+ name);
}
@Override
public void setBeanClassLoader(ClassLoader classLoader) {
System.out.println(“2 我是 BeanClassLoaderAware 的 setBeanClassLoader 方法”);
}
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
// 注意: 如果使用 @Configuration 注解的话,setBeanFactory方法会执行2次,
System.out.println(“3 我是 BeanFactoryAware 的 setBeanFactory 方法”);
}
@Override
public void setEnvironment(Environment environment) {
System.out.println(“4 我是 EnvironmentAware 的 setEnvironment 方法”);
}
@Override
public void setEmbeddedValueResolver(StringValueResolver stringValueResolver) {
System.out.println(“5 我是 EmbeddedValueResolverAware 的 setEmbeddedValueResolver 方法”);
}
@Override
public void setResourceLoader(ResourceLoader resourceLoader) {
System.out.println(“6 我是 ResourceLoaderAware 的 setResourceLoader 方法”);
}
@Override
public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {
System.out.println(“7 我是 ApplicationEventPublisherAware 的 setApplicationEventPublisher 方法”);
}
@Override
public void setMessageSource(MessageSource messageSource) {
System.out.println(“8 我是 MessageSourceAware 的 setMessageSource 方法”);
}
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
System.out.println(“9 我是 ApplicationContextAware 的 setApplicationContext 方法”);
}
@Override
public void setServletContext(ServletContext servletContext) {
System.out.println(“10 我是 ServletContextAware 的 setServletContext 方法”);
}
@Override
public void setLoadTimeWeaver(LoadTimeWeaver loadTimeWeaver) {
//LoadTimeWeaver 简称LTW,LTW是AOP的一种实现方式,此方法是为了获取Aop织入的对象,使用的织入方式是:类加载期织入,
// 一般的aop都是运行期织入,就是在运行的时候才进行织入切面方法,但是LTW是在类加载前就被织入了,也就是class文件在jvm加载之前进行织入切面方法
// 只有在使用 @EnableLoadTimeWeaving 或者存在 LoadTimeWeaver 实现的 Bean 时才会调用,顺序也很靠后
System.out.println(“11 我是 LoadTimeWeaverAware 的 setLoadTimeWeaver 方法”);
}
@Override
public void setImportMetadata(AnnotationMetadata annotationMetadata) {
//只有被其他配置类 @Import(XX.class) 时才会调用,这个调用对 XX.class 中的所有 @Bean 来说顺序是第 1 的。
System.out.println(“12 我是 ImportAware 的 setImportMetadata 方法”);
}
}
复制代码
启动spring后的控制台打印的部分结果如下:
可以看到它们的输出结果按照顺序依次排列打印出来了,这就是它的标准顺序了;接下来我们了解下它们的具体作用
6.1 BeanNameAware.setBeanName()
这个方法只是简单的返回我们当前的beanName,听官方的意思是这个接口更多的使用在spring的框架代码中,实际开发环境应该不建议使用
6.2 BeanClassLoaderAware.setBeanClassLoader()
获取Bean的类装载器,
6.3 BeanFactoryAware.setBeanFactory()
获取bean工厂,beanFactory让你可以不依赖注入方式,随意的读取IOC容器里面的对象,不过beanFactory本身还是要注入的。
需要注意的是,一般情况下我们都用 @Component 注解,如果使用 @Configuration 注解的话,setBeanFactory方法会执行2次;
6.4 EnvironmentAware.setEnvironment()
实现了EnvironmentAware接口重写setEnvironment方法后,在工程启动时可以获得application.properties 、xml、yml 的配置文件配置的属性值。
6.5 EmbeddedValueResolverAware.setEmbeddedValueResolver()
通常我们使用@Value注解来获取properties 和 yml 文件中的值,每个类中都要使用@Value也很繁琐,实现EmbeddedValueResolverAware接口后就方便多了。用法也跟@Value一样,需要用${}包裹住;
@Component
public class PropertiesUtil implements EmbeddedValueResolverAware {
@Override
public void setEmbeddedValueResolver(StringValueResolver stringValueResolver) {
System.out.println(stringValueResolver.resolveStringValue(“${logging.file}”));
}
}
复制代码
6.6 ResourceLoaderAware.setResourceLoader()
Spring ResourceLoader为我们提供了一个统一的getResource()方法来通过资源路径检索外部资源。从而将资源或文件(例如文本文件、XML文件、属性文件或图像文件)加载到Spring应用程序上下文中的不同实现 ,其实说白了,就是用来加载外部资源的;方法中有个参数:ResourceLoader ,这个参数其实就是ApplicationContext(spring 的上下文对象);可直接强转;
package org.crazyit.app.service;
import org.springframework.context.ResourceLoaderAware;
import org.springframework.core.io.ResourceLoader;
public class TestBean implements ResourceLoaderAware{
public void setResourceLoader(ResourceLoader resourceLoader) {
// 可直接强转为 ApplicationContext
ApplicationContext context = (ApplicationContext) resourceLoader;
System.out.println(“6 我是 ResourceLoaderAware 的 setResourceLoader 方法”);
}
}
复制代码
并且我们可以指定不同的前缀来创建路径以从不同位置加载资源
6.7 ApplicationEventPublisherAware.setApplicationEventPublisher();
ApplicationEventPublisherAware是一个事件发布器的接口,使用这个接口,我们自己的 Service 就拥有了发布事件的能力。用户注册后,不再是显示调用其他的业务 Service,而是发布一个用户注册事件。那么在这里是发布事件,那就肯定有监听事件的接口,这个接口叫做 ApplicationListener ,只要实现 ApplicationListener 接口就可以接受发布的事件了,接下来我们写一个示例来模拟发布事件和监听事件;
先创建一个实体类,用来存储发布的事件内容 StringEvent.java
package com.Spring.Boot.init.listener.eventModel;
import org.springframework.context.ApplicationEvent;
//事件监听对象
public class StringEvent extends ApplicationEvent {
private String str;
// 构造函数
public StringEvent(Object source) {
super(source);
str = source.toString();
}
// 获取字符串
public String getStr(){
return str;
}
}
复制代码
创建一个发布事件的类: ExtApplicationEventPublisherAware.java ,实现 ApplicationEventPublisherAware 接口增加发布事件的功能;
package com.Spring.Boot.init.aware;
import com.Spring.Boot.init.listener.eventModel.StringEvent;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.context.ApplicationEventPublisherAware;
import org.springframework.stereotype.Component;
/**
- 发布事件
*/
@Component
public class ExtApplicationEventPublisherAware implements ApplicationEventPublisherAware {
@Override
public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {
System.out.println(“发布事件,事件对象为 StringEvent ,内容为 :1234”);
StringEvent stringEvent = new StringEvent(“1234”);
// 发布事件 ,发布后会在 ApplicationListener.onApplicationEvent()方法进行捕获;
applicationEventPublisher.publishEvent(stringEvent); // 发布事件
}
}
复制代码
在创建一个事件监听器: EventListener.java ,用来监听所有发布的事件;
package com.Spring.Boot.init.listener;
import com.Spring.Boot.init.listener.eventModel.StringEvent;
import org.springframework.context.ApplicationEvent;
import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;
//事件监听器
@Component
public class EventListener implements ApplicationListener {
@Override
public void onApplicationEvent(StringEvent o) {
System.out.println(“监听到事件,内容:”+o.getStr());
}
}
复制代码
接下来,运行spring项目,看看打印的结果如下,到这里,事件的发布和监听就完成了;
6.8 MessageSourceAware.setMessageSource()
国际化消息通知操作
6.9 ApplicationContextAware.setApplicationContext()
ApplicationContextAware 主要用来全局获取 ApplicationContext 上下文,ApplicationContext其实就是容器,为此我们可以实现 ApplicationContextAware 接口来获取ApplicationContext容器对象;我们可以把它做成一个公共的静态类,这样可以在任意地方想拿就拿了,
package com.Spring.Boot.init.aware;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.annotation.Configuration;
import org.springframework.stereotype.Component;
@Component
public class ExtApplicationContextAware implements ApplicationContextAware {
/**
-
Spring容器会在加载完Spring容器后调用ApplicationContextAware.setApplicationContext方法
-
ApplicationContextAware 主要用来全局获取 ApplicationContext 上下文,
*/
private static ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
if (ExtApplicationContextAware.applicationContext == null) {
ExtApplicationContextAware.applicationContext = applicationContext;
}
System.out.println(“ApplicationContext配置成功”);
System.out.println(“在普通类可以通过调用SpringBootBeanUtil.getApplicationContext()获取applicationContext对象”);
System.out.println(“applicationContext=“+ ExtApplicationContextAware.applicationContext +””);
}
/**
-
获取applicationContext
-
@return
*/
public static ApplicationContext getApplicationContext() {
return applicationContext;
}
/**
-
通过name获取 Bean.
-
@param name
-
@return
*/
public static Object getBean(String name) {
return getApplicationContext().getBean(name);
}
/**
-
通过class获取Bean.
-
@param clazz
-
@return
*/
public static T getBean(Class clazz) {
return getApplicationContext().getBean(clazz);
}
/**
-
通过name,以及Clazz返回指定的Bean
-
@param name
-
@param clazz
-
@return
*/
public static T getBean(String name, Class clazz) {
return getApplicationContext().getBean(name, clazz);
}
}
复制代码
当然,也可以直接注入,就像这样:
@Autowired
private ApplicationContext applicationContext;
复制代码
6.10 ServletContextAware.setServletContext()
通过实现ServletContextAware接口可获取servletContext,也就是servlet的上下文;
什么是ServletContext
: WEB容器在启动时,它会为每个WEB应用程序都创建一个对应的ServletContext对象,它代表当前web应用。ServletConfig对象中维护了ServletContext对象的引用,开发人员在编写servlet时,可以通过ServletConfig.getServletContext方法获得ServletContext对象。
由于一个WEB应用中的所有Servlet共享同一个ServletContext对象,因此Servlet对象之间可以通过ServletContext对象来实现通讯。ServletContext对象通常也被称之为context域对象。
6.11 LoadTimeWeaverAware.setLoadTimeWeaver()
其实在调试的时候还有2个没打印出来,第11个就是 LoadTimeWeaver, 简称LTW,LTW是AOP的一种实现方式,此方法是为了获取Aop织入的对象,使用的织入方式是:类加载期织入,一般的aop都是运行期织入,就是在运行的时候才进行织入切面方法,但是LTW是在类加载前就被织入了,也就是class文件在jvm加载之前进行织入切面方法只有在使用 @EnableLoadTimeWeaving 或者存在 LoadTimeWeaver 实现的 Bean 时才会调用,顺序也很靠后;
6.12 ImportAware.setImportMetadata()
还有一个没打印的就是ImportAware接口,这个接口的方法只有被其他配置类 @Import(XX.class) 时才会调用,这个调用对 XX.class 中的所有 @Bean 来说顺序是第 1 的。
7、初始化前置
方法名称: BeanPostProcessor.postProcessBeforeInitialization()
在每一个 Bean 初始化之前执行的方法(有多少 Bean 调用多少次)
注意 : 启用该方法后,标注了@PostConstruct注解的方法会失效
8、初始化后置
方法名称: BeanPostProcessor.postProcessAfterInitialization()
在每一个 Bean 初始化之后执行的方法(有多少 Bean 调用多少次)
初始化前置和初始化后置的实现代码如下
package com.Spring.Boot.init;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.stereotype.Component;
@Component
public class ExtBeanPostProcessor implements BeanPostProcessor {
@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
// 在每一个 Bean 初始化之前执行的方法(有多少 Bean 调用多少次)
// 注意 : 启用该方法后,标注了@PostConstruct注解的方法会失效
System.out.println(“初始化前置方法”);
return null;
}
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
在每一个 Bean 初始化之后执行的方法(有多少 Bean 调用多少次)
System.out.println(“初始化后置方法”);
return null;
}
}
复制代码
9、执行初始化方法
初始化方法有三个,分别是 添加了@PostConstruct 注解的方法、实现InitializingBean接口、在@bean注解上添加 initMethod属性;我们一个个讲
10、初始化方法一:@PostConstruct
在bean对象内添加@PostConstruct 注解后即可实现初始化的功能,被@PostConstruct修饰的方法会在构造函数之后,init()方法之前运行。 有多个则会执行多次;
注意: 如果spring 实现了 BeanPostProcessor接口的postProcessBeforeInitialization() 方法,也就是12的初始后置方法,那么@PostConstruct注解会失效;
代码示例
package com.Spring.Boot.init;
import org.springframework.stereotype.Component;
import javax.annotation.PostConstruct;
// @PostConstruct注解
@Component
public class ExtPostConstruct {
/**
-
被@PostConstruct修饰的方法会在构造函数之后,init()方法之前运行。如果有多个则会执行多次
-
注意: 如果spring 实现了 BeanPostProcessor接口的postProcessBeforeInitialization方法,该@PostConstruct注解会失效
*/
@PostConstruct
public void init() {
System.out.println(“第一个init…”);
}
// 有多个会执行多次
@PostConstruct
public void init1() {
System.out.println(“第二个init1…”);
}
}
复制代码
11、InitializingBean.afterPropertiesSet()
spring 初始化方法之一,作用是在BeanFactory完成属性设置之后,执行自定义的初始化行为。
执行顺序:在initMethod之前执行,在@PostConstruct之后执行
代码示例
package com.Spring.Boot.init;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.annotation.Configuration;
import org.springframework.stereotype.Component;
@Component
public class ExtInitializingBean implements InitializingBean {
@Override
public void afterPropertiesSet() throws Exception {
// 一个 InitializingBean 执行一次
// spring 初始化方法,作用是在BeanFactory完成属性设置之后,执行自定义的 初始化行为.
// 执行顺序:在initMethod之前执行,在@PostConstruct之后执行
System.out.println(“InitializingBean”);
}
}
复制代码
12、init-method
bean 配置文件属性 init-method 用于在bean初始化时指定执行方法,用来替代继承 InitializingBean接口,
注意的一点是只有一个类完整的实例被创建出来后,才能走初始化方法。
示例代码,先定义一个类: BeanTest.java ,在类中定义一个初始化方法 initMethod_1()
package com.Spring.Boot.init.bean;
public class BeanTest {
// 将要执行的初始化方法
public void initMethod_1(){
System.out.println(“我是beanTest的init方法”);
}
}
复制代码
xml 配置方式
<bean id="beanTest" class="com.BeanTest" init-method="init"></bean>
注解配置方式
package com.Spring.Boot.init;
import com.Spring.Boot.init.bean.BeanTest;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
@Component()
public class InitMethod {
// 在@Bean注解上添加initMethod属性,指向类中的 initMethod_1 执行初始化方法
@Bean(initMethod = “initMethod_1”)
public BeanTest getBeanTest(){
return new BeanTest();
总结
我个人认为,如果你想靠着背面试题来获得心仪的offer,用癞蛤蟆想吃天鹅肉形容完全不过分。想必大家能感受到面试越来越难,想找到心仪的工作也是越来越难,高薪工作羡慕不来,却又对自己目前的薪资不太满意,工作几年甚至连一个应届生的薪资都比不上,终究是错付了,错付了自己没有去提升技术。
这些面试题分享给大家的目的,其实是希望大家通过大厂面试题分析自己的技术栈,给自己梳理一个更加明确的学习方向,当你准备好去面试大厂,你心里有底,大概知道面试官会问多广,多深,避免面试的时候一问三不知。
大家可以把Java基础,JVM,并发编程,MySQL,Redis,Spring,Spring cloud等等做一个知识总结以及延伸,再去进行操作,不然光记是学不会的,这里我也提供一些脑图分享给大家:
希望你看完这篇文章后,不要犹豫,抓紧学习,复习知识,准备在明年的金三银四拿到心仪的offer,加油,打工人!
ent;
@Component()
public class InitMethod {
// 在@Bean注解上添加initMethod属性,指向类中的 initMethod_1 执行初始化方法
@Bean(initMethod = “initMethod_1”)
public BeanTest getBeanTest(){
return new BeanTest();
总结
我个人认为,如果你想靠着背面试题来获得心仪的offer,用癞蛤蟆想吃天鹅肉形容完全不过分。想必大家能感受到面试越来越难,想找到心仪的工作也是越来越难,高薪工作羡慕不来,却又对自己目前的薪资不太满意,工作几年甚至连一个应届生的薪资都比不上,终究是错付了,错付了自己没有去提升技术。
这些面试题分享给大家的目的,其实是希望大家通过大厂面试题分析自己的技术栈,给自己梳理一个更加明确的学习方向,当你准备好去面试大厂,你心里有底,大概知道面试官会问多广,多深,避免面试的时候一问三不知。
大家可以把Java基础,JVM,并发编程,MySQL,Redis,Spring,Spring cloud等等做一个知识总结以及延伸,再去进行操作,不然光记是学不会的,这里我也提供一些脑图分享给大家:
[外链图片转存中…(img-xqzt0sCp-1715091363319)]
[外链图片转存中…(img-ZtX4xpMM-1715091363319)]
[外链图片转存中…(img-ZESdRfJQ-1715091363319)]
希望你看完这篇文章后,不要犹豫,抓紧学习,复习知识,准备在明年的金三银四拿到心仪的offer,加油,打工人!