顽石系列:Java技术面试(一)

顽石系列: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");  
  1. 会分配一个11长度的char数组,并在常量池分配一个由这个char数组组成的字符串,然后由m去引用这个字符串

  2. 用n去引用常量池里边的字符串,所以和n引用的是同一个对象

  3. 生成一个新的字符串,但内部的字符数组引用着m内部的字符数组

  4. 同样会生成一个新的字符串,但内部的字符数组引用常量池里边的字符串内部的字符数组,意思是和u是同样的字符数组

  

NIO相关

NIO有三大核心部分

传统IO有什么缺点,为什么NIO是非阻塞的?

传统IO为每一个客户端使用一个线程,如果客户端出现延迟等异常,线程可能会被占用很长时间,因为数据的准备和读取都在这个线程中,此时,如果客户端数量众多,将会消耗⼤量的系统资源

 ⼀个线程对应⼀一个selector,但是⼀个 selector可以对应多个客户端极少的线程可以监控⼤量的客户端,即使发生阻 塞,数量也是很少的。 

猜你喜欢

转载自www.cnblogs.com/MrSaver/p/9329422.html