顽石系列: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是同样的字符数组
NIO相关
NIO有三大核心部分
传统IO有什么缺点,为什么NIO是非阻塞的?
传统IO为每一个客户端使用一个线程,如果客户端出现延迟等异常,线程可能会被占用很长时间,因为数据的准备和读取都在这个线程中,此时,如果客户端数量众多,将会消耗⼤量的系统资源。
⼀个线程对应⼀一个selector,但是⼀个 selector可以对应多个客户端。 极少的线程可以监控⼤量的客户端,即使发生阻 塞,数量也是很少的。