第二次面试总结 - 宏汉科技 - Java后端开发

简介: 本文是作者对宏汉科技Java后端开发岗位的第二次面试总结,面试结果不理想,主要原因是Java基础知识掌握不牢固,文章详细列出了面试中被问到的技术问题及答案,包括字符串相关函数、抽象类与接口的区别、Java创建线程池的方式、回调函数、函数式接口、反射以及Java中的集合等。

总结 (非详细)

  • 面试结果:非常不好
  • 面试内容:Java基础八股
  • 原因:Java基础相关八股文太差

面试内容(提问内容) - 带答案

  1. 字符串相关的函数
  2. 抽象类和接口的区别
  3. Java有几种创建线程池的方式,分别是什么,有什么不同
  4. 什么是回调函数
  5. 什么是函数式接口,函数式接口与普通接口的区别
  6. 什么是反射
  7. Java中的集合

1、字符串相关的函数

  • 创建字符串

String str = "hello" // 直接赋值创建字符串

String str = new String("Hello") // 使用构造函数创建字符串对象

  • 获取字符串长度

int length = str.length() // 获取字符串长度

  • 获取指定位置的字符

char ch = str.charAt(index) // 获取指定位置的字符

  • 查找子字符串的位置

int index = str.indexOf("hello world") // 查找子字符串的位置

  • 判断是否包含子字符串

boolean contains = str.contains("substring") // 判断是否包含子字符串

  • 替换字符串中的内容

String replaced = str.replace("old", "new") // 替换字符串中的内容

  • 截取子字符串

Strin sub = str.substring(startIndex, endIndex) // 截取子字符串

  • 字符串拼接

String concat = str.concat(" World") // 字符串拼接

  • 使用指定分隔符连接多个字符串

String joined = String.join("-", "Hello", "World") // 使用指定分隔符连接多个字符串

  • 根据指定分隔符拆分字符串为数组

String[] parts = str.split("delimiter") // 根据指定分隔符拆分字符串为数组

  • 去除字符串首尾空白字符

String trimmed = str.trim() // 去除字符串首尾空白字符

  • 字符串大小写转换

String upper = str.toUpperCase() // 转换为大写

String lower = str.toLowerCase() // 转换为小写

  • 判断字符串是否为空

boolean isEmpty = str.isEmpty() // 判断字符串是否为空(更常用)

boolean isBlank = str.isBlank() // 判断字符串是否为空(Java11 新增)

  • 将其他类型转换为字符串

基本上的万能方法: String.valueOf(需要转换的变量名)

以及 toString 方法,只不过什么.toString , .toString的前缀各有不同

int num = 10;
String str = String.valueOf(num); // 将整数类型(int、long)转换为字符串
// 或者
String str = Integer.toString(num); // 将整数类型(int、long)转换为字符串

double num = 10.5;
String str = String.valueOf(num); // 将浮点数类型(float、double)转换为字符串
// 或者
String str = Double.toString(num); // 将浮点数类型(float、double)转换为字符串

boolean flag = true;
String str = String.valueOf(flag); // 将Boolean类型转换为字符串

char ch = 'A';
String str = String.valueOf(ch); // 将字符类型转换为字符串

// 除基本数据类型外,其他类型可以使用对象的toString()方法

Object obj = new Object();
String str = obj.toString(); // 将其他类型转换为字符串

  • 将字符串转换为其他类型

String str = "123";
int intValue = Integer.parseInt(str); // 字符串转换为整数

String str = "1234567890";
long longValue = Long.parseLong(str); // 字符串转换为长整数

字符串转换为浮点数

String str = "3.14";
float floatValue = Float.parseFloat(str); // 字符串转换为 float
double doubleValue = Double.parseDouble(str); // 字符串转换为 double

String str = "true";
boolean boolValue = Boolean.parseBoolean(str); // 字符串转换为布尔值

字符串转换为自定义对象

引入Jackson 或 FastJson依赖,使用这其中的内置函数,需要注意的是这个被转换的字符串本身就要是JSON格式的字符串,要和对象中的属性对应,具体操作此处不再展示,用时查询即可

2、抽象类和接口的区别

抽象类

特点:

  • 可以包含抽象方法和非抽象方法
  • 可以有成员变量
  • 可以有构造方法
  • 通过 “extends” 关键字被子类继承
  • 不能被实例化,只能作为其他类的父类,子类必须实现(覆盖)其抽象方法

用法:

  • 适合用于需要某些默认实现的情况
  • 用于定义类的层次结构,提供一些通用行为

接口:

特点:

  • 只能包含抽象方法(Java8+可以有默认方法和静态方法)
  • 不能有成员变量(Java8+可以有静态常量)
  • 不能有构造方法
  • 通过 “implements” 关键字被类发现
  • 可以多继承,一个类可以实现多个接口

用法:

  • 用于定义一组相关的方法,而不包含实现
  • 用于实现多继承,使类具有多种类型特征
  • 用于实现不同类之间的通用协议
  • 可以多继承,一个类可以实现多个接口

如何选择使用抽象类还是接口

如果你需要提供一些默认实现,或者需要在未来扩展更多的方法,可以使用抽象类

如果你需要定义一组方法的契约,而不关心具体实现,并且需要支持多继承特性,可以使用接口

有时候也可以将抽象类和接口结合使用,使用接口定义契约,使用抽象类提供默认实现

3、Java有几种创建线程池的方式,分别是什么,有什么不同

有好多种,甚至我感觉应该没有明确的数字来确定Java有几种创建线程池的方式,因为有些是JavaJDK提供的,有些是某些框架提供的

下面说一下我了解的常见的五种方式

3.1、“Executors” 工厂类(常用) - 精简

“newFixedThreadPool(int nThreads)” :创建一个固定大小的线程池,线程数量固定不变

代码举例:(private是权限修饰符,可更改)

private ExecutorService executorService = Executors.newFixedThreadPool(8);

“newCachedThreadPool()”:创建一个根据需要自动扩展的线程池,线程数量可以根据任务的多少来进行动态调整

“newSingleThreadExecutor()”:创建一个单线程的线程池,保证任务按顺序执行,这些方法返回“ExecutorService”接口的实例,可以用于提交任务、关闭线程池等操作

3.2、“ThreadPoolExecutor类”(常用) - 细节

“ThreadPoolExecutor” 类提供了更灵活的线程池创建方式,可以自定义线程池的配置

ThreadPoolExecutor executor = new ThreadPoolExecutor(
    corePoolSize, // 核心线程数
    maximumPoolSize, // 最大线程数
    keepAliveTime, // 线程空闲时间
    TimeUnit.MILLISECONDS, // 时间单位
    new LinkedBlockingQueue<Runnable>() // 任务队列
);

通过 “ThreadPoolExecutor” 类,可以灵活地配置核心线程数、最大线程数、线程空闲时间、任务队列等

3.3、Spring框架提供的线程池(光Spring创建线程的方式就有好几种,虽然都与TaskExecutor接口有关系)

Spring 框架提供了 TaskExecutor 接口来管理线程池,以及一些实现了这个接口的具体线程池实现,例如使用 “ThreadPoolTaskExecutor”进行线程池的管理和配置

示例代码如下:

import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

// 创建 ThreadPoolTaskExecutor 对象
ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();

// 设置线程池属性
taskExecutor.setCorePoolSize(5);
taskExecutor.setMaxPoolSize(10);
taskExecutor.setQueueCapacity(25);

// 设置其他属性...

// 初始化线程池
taskExecutor.initialize();

// 提交任务给线程池执行
taskExecutor.execute(() -> {
    // 执行的任务逻辑
});

3.4、“ScheduledExecutorService” 接口

用于执行延迟任务或定期任务

“newScheduledThreadPool(int corePoolSize)”:创建一个可调度的线程池,可以执行定时或延迟任务

3.5、ForkJoinPool

用于执行 Fork-Join 任务,通过拆分任务并行执行来提高性能

不同的线程池创建方式适用于不同的场景和需求,可以根据具体的应用情况选择合适的方式创建线程池

4、什么是回调函数

回调函数是一种常见的编程概念,它在事件驱动编程异步编程等场景下非常常见,指的是在特定条件满足或事件发生时执行的函数。在编程中,回调函数通常作为参数传递给其他函数,以便在需要时被调用执行

举个最简单的示例,在图形化界面编程和前端中,就经常用到回调函数,比如当用户点击按钮时,可以将处理点击事件的函数作为回调函数传递给按钮的事件监听器。当按钮被点击时,事件监听器会调用这个回调函数

又比如我之前写过一个mq的项目,里面我专门定义了一个函数式接口来存放回调函数,因为函数式接口与普通接口不同,其中只可以有一个抽象方法,虽然定义接口时不可以直接写方法体,但可以有参数,而且后续使用时可以用Lambda表达式等方式实现函数式接口。我当时直接把这个接口当作参数,new出来后直接大括号,在里面把回调函数实现了,然后把这整个当作参数传递了,还挺方便的

5、什么是函数式接口,函数式接口与普通接口的区别

函数式接口是Java中的一个特殊类型的接口,它仅包含一个抽象方法(除了默认方法、静态方法或来自 “Object” 类的 “public” 方法)。函数式接口可以被Lambda表达式实现(口语化,不需要记:当然,也可以后续new接口的时候,在小括号内加个大括号,大括号内放重写的方法)

函数式接口特点:

  • 单一抽象方法:函数式接口只包含一个抽象方法。他的特点是只能有一个抽象方法,但可以包含多个默认方法,静态方法或来自 “Object” 类的 “public” 方法
  • @FunctionalInterface注解:函数式接口通常都会使用 “FunctionalInterface” 注解标识。这个注解可以让编译器帮助检查这个接口是否符合函数式接口的定义
  • Lamabda表达式支持:函数式接口可以被Lambda表达式所实现。Lambda表达式提供了一种简洁的方式来创建函数式接口的实例

注:也可以通过其他方式实现

普通接口特点:

  • 多个抽象方法:普通接口可以包含多个抽象方法
  • 没有 @FunctionalInterface 注解:普通接口没有 @FunctionalInterface 注解的限制,它可以包含多个抽象方法,并且不要求只有一个抽象方法

函数式接口和普通接口在设计和使用上有明显的差异,函数式接口更注重于函数式编程,鼓励使用 Lambda 表达式和函数式思维。而普通接口则更通用,可以包含多个方法,用于更广泛的场景。其实相对于两者的格式,往往是两者的使用场景不同而造成我们关注的真正意义上的 “不同”

6、 什么是反射

    反射是Java语言的特征之一,它允许**运行中**的**Java程序对自身进行检查**,或者说 “**自审**”,并能**直接操作程序的内部属性和方法**(当然,反射并不是Java独有的,许多编程语言都提供了反射功能)

通俗地讲,一提到反射,我们就可以想到镜子。镜子可以明明白白地照出我是谁,还可以照出别人是谁。反映到程序中,反射就是用来让开发者知道这个类中有什么成员,以及别的类中有什么成员

其实反射的经典用处之一在于通过反射实现某些本不可能实现的功能,我们知道private修饰的属性是只能在类和对象内部访问的,但是有些时候我们需要获得这个“私有的”方法或属性,但是又不想破坏这个属性的修饰权限和范围,那我们就可以使用反射动态地获取到这个类的信息,并能够进行访问甚至修改

其实,我们常见的Spring托管对象的操作和JUnit框架的某些操作等本质也是利用了反射

可以说,反射无处不在,只是我们真正敲出明面上就是反射的代码少

注:反射是面试经常考的,而且相关的知识有特别多,真正想细致地去了解反射以及反射的用法等需要专门去搜寻

7、Java中的集合

Java集合分三种,分别是 List、Set、Map,这三种集合适用于不同的场景

  • List:适用于有序,可重复的集合
  • Set:适用于不可重复集合
  • Map:适用于键值对的存储

注:通常List与Map最为常用

  • List 常见的实现类:ArrayList(列表) 、 LinkedList(链表)
  • Set 常见的实现类:HashSet、LinkedHashSet和TreeSet
  • Map 常见的实现类:HashMap 、 TreeMap 与 HashTable

上面的实现类大多都是线程不安全的,因此,Java也提供了对应的线程安全版本:

  • 比如 List 中的 Victor、CopyOnWriteArrayList等
  • Set本身就用的少,所以线程安全的Set实体类用的更少,我确实没太记得住,个人感觉记这个不如到时候查询性价比高(面试可以这样说,一般不会让面试官感觉你不会)
  • 比如Map中的HashTable,当然这个太古老了,用的很少,但是另一个 ConcurrentHashMap 用的非常多,ConcurrentHashMap 也是线程安全的,当然,也有其他实现方式,看具体使用情况选择合适的实现类就好

详细总结(注意事项)

关于第一个问题 “字符串相关的函数” ,当时回答了几个经典的,比如 valueOf 等,下意识地忽略了简单的length、 contains等,平常使用的时候下意识就写出来了,有的写代码过程中没有想起来也直接查询后就使用了,所以答得并不好

关于第二个问题 “抽象类和接口的区别”,这个答得还可以,但是面试的时候,确实有些平常都知道的东西没想起来(这个问题其实真的属于纯八股了)

关于第三个问题 “Java有几种创建线程池的方式,分别是什么,有什么不同”,这个答得就非常不好了,真的平常光知道用了

关于第四个问题和第五个问题 “什么是回调函数”, “什么是函数式接口,函数式接口与普通接口的区别” 确实当场没想起来

关于第六个问题 “什么是反射”,我当时回答了反射怎么用,什么时候用,为什么要用,也都回答的可以,但面试官说这不是他想听到的,他想听到的只是反射的定义,好吧~ 这个怪我自己,平时确实没注意这些八股,只知道用了

关于第七个问题 “Java中的集合”,只能说回答的勉强凑合,忘记List、Set、Map都是集合了

面试官如果问Spring的八股,我还能答得不错;如果问我项目,我能答得非常优秀,但他直接就不问我项目,开场我自我介绍,就主要说了我项目能力比较好,然后他来了句 “好,那我既然看你一直在讲你的项目,那今天咱不提问项目,提问点Java基础”,然后 我就挂了~

个人情绪等无关重要的在此就不总结了,最后提一句话,即便简历不怎么样,也能搞几次大厂面试的机会,所以面试更重要的说白了就是八股,其次算法和项目,项目反而是最不重要的

加油背八股!

🧸正是时间好,过年发的此篇文章,在此祝大家新的一年 万事顺遂

目录
相关文章
|
5天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
18 2
|
10天前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
9天前
|
存储 SQL API
探索后端开发:构建高效API与数据库交互
【10月更文挑战第36天】在数字化时代,后端开发是连接用户界面和数据存储的桥梁。本文深入探讨如何设计高效的API以及如何实现API与数据库之间的无缝交互,确保数据的一致性和高性能。我们将从基础概念出发,逐步深入到实战技巧,为读者提供一个清晰的后端开发路线图。
|
8天前
|
JSON 前端开发 API
后端开发中的API设计与文档编写指南####
本文探讨了后端开发中API设计的重要性,并详细阐述了如何编写高效、可维护的API接口。通过实际案例分析,文章强调了清晰的API设计对于前后端分离项目的关键作用,以及良好的文档习惯如何促进团队协作和提升开发效率。 ####
|
10天前
|
存储 SQL 数据库
深入浅出后端开发之数据库优化实战
【10月更文挑战第35天】在软件开发的世界里,数据库性能直接关系到应用的响应速度和用户体验。本文将带你了解如何通过合理的索引设计、查询优化以及恰当的数据存储策略来提升数据库性能。我们将一起探索这些技巧背后的原理,并通过实际案例感受优化带来的显著效果。
28 4
|
9天前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
【10月更文挑战第36天】本文将引导您探索Node.js的世界,通过实际案例揭示其背后的原理和实践方法。从基础的安装到高级的异步处理,我们将一起构建一个简单的后端服务,并讨论如何优化性能。无论您是新手还是有经验的开发者,这篇文章都将为您提供新的视角和深入的理解。
|
11天前
|
SQL 安全 Java
安全问题已经成为软件开发中不可忽视的重要议题。对于使用Java语言开发的应用程序来说,安全性更是至关重要
在当今网络环境下,Java应用的安全性至关重要。本文深入探讨了Java安全编程的最佳实践,包括代码审查、输入验证、输出编码、访问控制和加密技术等,帮助开发者构建安全可靠的应用。通过掌握相关技术和工具,开发者可以有效防范安全威胁,确保应用的安全性。
24 4
|
10天前
|
监控 API 持续交付
后端开发中的微服务架构实践与挑战####
本文深入探讨了微服务架构在后端开发中的应用,分析了其优势、面临的挑战以及最佳实践策略。不同于传统的单体应用,微服务通过细粒度的服务划分促进了系统的可维护性、可扩展性和敏捷性。文章首先概述了微服务的核心概念及其与传统架构的区别,随后详细阐述了构建微服务时需考虑的关键技术要素,如服务发现、API网关、容器化部署及持续集成/持续部署(CI/CD)流程。此外,还讨论了微服务实施过程中常见的问题,如服务间通信复杂度增加、数据一致性保障等,并提供了相应的解决方案和优化建议。总之,本文旨在为开发者提供一份关于如何在现代后端系统中有效采用和优化微服务架构的实用指南。 ####
|
12天前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
37 4
|
12天前
|
消息中间件 设计模式 运维
后端开发中的微服务架构实践与挑战####
本文深入探讨了微服务架构在现代后端开发中的应用,通过实际案例分析,揭示了其在提升系统灵活性、可扩展性及促进技术创新方面的显著优势。同时,文章也未回避微服务实施过程中面临的挑战,如服务间通信复杂性、数据一致性保障及部署运维难度增加等问题,并基于实践经验提出了一系列应对策略,为开发者在构建高效、稳定的微服务平台时提供有价值的参考。 ####