顽石系列:Java技术面试
JDBC相关
1.Statement与PreparedStatement的区 别,什什么是SQL注⼊入,如何防⽌止SQL注⼊?
PreparedStatement支持动态设置参数、支持预编译、防止SQL注入,而statement不支持。
SQL注入:通过拼接SQL语句达到无参数查询数据库数据的目的的方法。
字符串相关
1.String、StringBuffer、StringBuilder?
String是不可变的,StringBuffer和StringBuilder都是可变的。后两者的字符内容可变,前者创建后内容不可变。
StringBuffer是线程安全的,但是会带来额外的系统开销。StringBuilder效率比较高。
2.==与.equals()的区别
使用==,当比较的是原生数据类型时,比较的是值。比较引用数据类型是,比较的是地址(也就是说是否同一个对象)。
equals()⽅法: 该⽅法定义在Object类中,因此java中的每个 类都具有该方法,对于Object类的equal()⽅方法来说,它是判断 调⽤用equals()⽅方法的引⽤用与传进来的引⽤用是否⼀一致,即这两个 引⽤用是否指向同⼀一个对象。
说明一点:
String重写了equals方法,对于String类的equal()⽅方法来说,它是判断当前字符串与传进 来的字符串的内容是否⼀致。
3.字符串常量池
字符串常量池的设计思想
字符串的分配,和其他的对象分配一样,耗费高昂的时间与空间代价,作为最基础的数据类型,大量频繁的创建字符串,极大程度地影响程序的性能
-
JVM为了提高性能和减少内存开销,在实例化字符串常量的时候进行了一些优化
为字符串开辟一个字符串常量池,类似于缓存区
创建字符串常量时,首先坚持字符串常量池是否存在该字符串
存在该字符串,返回引用实例,不存在,实例化该字符串并放入池中
- 实现基础
- 字符串不可变,不用担心数据冲突
- 常量池的字符串不会被垃圾回收器回收。
运行时数据区
堆:
存放的是对象,每个对象都包含一个与之对应的class
JVM只有一个堆区被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身。
对象由垃圾回收器复杂回收,因此大小和生命周期不需要确定。
栈:
每个线程包含一个栈区,栈中只保存基础数据类型和对象引用,不是对象。
每个栈中的数据(原始类型和对象引用)都是私有的
栈中分三个部分:基本类型变量区、执行环境上下文、操作指令区(存放操作指令)
数据大小和生命周期时可以确定的,当没有引用执行数据时,这个数据就会消失
方法区:
静态区和堆一样,被所有线程共享。
方法区存在的都是在整个程序中永远唯一的,如class、static等。
字符串常量池存储于方法区(说明:方法区是堆的一个逻辑组成部分):
面试题:栈内存与堆内存
当一个方法执行时,每个方法都会建立自己的内存栈,在这个方法内定义的变量将会逐个放入逐个栈内存中,随着方法的执行结束,这个方法的内存栈也会随之销毁。因此,所有方法的局部变量都是放在栈内存中的;在程序中,创建一个对象时,这个对象被保存到运行时数据区中,以便反复利用(因为对象的创想成本较大),这个运行时数据区就是堆内存。堆内存的变量不会随着方法的执行而销毁。只有当一个对象没有被任何引用变量引用时,系统的垃圾回收期才会在合适的时候自动销毁它。
面试题:String str4 = new String(“abc”) 创建多少个对象?
1.首先在字符串常量池中查找是否有"abc"对象,有则返回对应的引用实例、没有则创建对应的引用实例
2.在堆中new一个String(abc)对象
3.将对象地址赋值给str4,创建一个引用
常量池中没有“abc”字面量则创建两个对象,否则创建一个对象,以及创建一个引用。
说明:
通过new操作符创建的字符串对象不指向字符串池中的任何对象,但是可以通过使用字符串的intern()方法来指向其中的某一个。java.lang.String.intern()返回一个保留池字符串,就是一个在全局字符串池中有了一个入口。如果以前没有在全局字符串池中,那么它就会被添加到里面。
字面量和常量池初探
String m = "hello,world";
String n = "hello,world";
String u = new String(m);
String v = new String("hello,world");
会分配一个11长度的char数组,并在常量池分配一个由这个char数组组成的字符串,然后由m去引用这个字符串
用n去引用常量池里边的字符串,所以和n引用的是同一个对象
生成一个新的字符串,但内部的字符数组引用着m内部的字符数组
同样会生成一个新的字符串,但内部的字符数组引用常量池里边的字符串内部的字符数组,意思是和u是同样的字符数组
IO/NIO相关
NIO有三大核心部分
传统IO有什么缺点,为什么NIO是非阻塞的?
传统IO为每一个客户端使用一个线程,如果客户端出现延迟等异常,线程可能会被占用很长时间,因为数据的准备和读取都在这个线程中,此时,如果客户端数量众多,将会消耗⼤量的系统资源。
⼀个线程对应⼀一个selector,但是⼀个 selector可以对应多个客户端。 极少的线程可以监控⼤量的客户端,即使发生阻 塞,数量也是很少的。
Scanner类
next() 与 nextLine() 区别
next:
一定要读到有效字符后才可以结束输入。
对输入有效字符之前的空白,next可以自动去掉。
只有输入有效字符后才将其后面输入的空白作为分隔符或者结束符。
next不能得到带有空格的字符串。
nextline:
以Enter为结束符,也就是说返回的是输入回车之前的所有字符。
可以获得空白。
Java基础
Java语言关键字
以下是Java编程语言中的关键字列表。您不能在程序中使用以下任何标识符作为标识符。
说明
const和goto是保留关键字,即使现在仍未被使用。
true,false和null可能看起来像关键字,但它们实际上是文字;您不能在程序中将它们用作标识符。
abstract |
continue |
for |
new |
switch |
assert ***
|
default |
goto *
|
package |
synchronized |
boolean |
do |
if |
private |
this |
break |
double |
implements |
protected |
throw |
byte |
else |
import |
public |
throws |
case |
enum ****
|
instanceof |
return |
transient |
catch |
extends |
int |
short |
try |
char |
final |
interface |
static |
void |
class |
finally |
long |
strictfp **
|
volatile |
const *
|
float |
native |
super |
while |
访问修饰符及范围
内部类及注意事项