Java面试(基础篇)

Java基础

Java语言有哪些特点/什么是Java?

简单易学;
面向对象(封装,继承,多态);
平台无关性( Java 虚拟机实现平台无关性);
GC实现垃圾回收;
异常处理机制;
支持多线程;
支持网络编程并且很方便;
编译与解释并存;

面向对象与面向过程的区别

  • 面向过程 :面向过程性能比面向对象高。 因为类调用时需要实例化,开销比较大,比较消耗资源,所以当性能是最重要的考量因素的时候,比如单片机、嵌入式开发、Linux/Unix等一般采用面向过程开发。但是,面向过程没有面向对象易维护、易复用、易扩展
  • 面向对象面向对象易维护、易复用、易扩展。 因为面向对象有封装、继承、多态性的特性,所以可以设计出低耦合的系统,使系统更加灵活、更加易于维护。但是,面向对象性能比面向过程低

Java和C++的区别?

  • 都是面向对象的语言,都支持封装、继承和多态
  • Java 不提供指针来直接访问内存,程序内存更加安全
  • Java 的类是单继承的,C++ 支持多重继承;虽然 Java 的类不可以多继承,但是接口可以多继承
  • Java 有自动内存管理机制,不需要程序员手动释放无用内存
  • 在 C 语言中,字符串或字符数组最后都会有一个额外的字符‘\0’来表示结束。但是,Java 语言中没有结束符这一概念。

Java有几种基本数据类型

  • 整数 byte、short、int、long
  • 浮点 float、double
  • 字符 char
  • 布尔 boolean

Java基本类型和引用类型他们的区别?

  • java中的基本数据类型包括整数、浮点、字符、布尔
  • 引用数据类型变量包括:类、接口数组变量
  • 运用不同点
    1.赋值方法不同,基本类型直接赋值,引用类型通过 new 创建对象,然后再把对象赋予相应的变量。
    2.比较方面的不同,== 号的比较:引用类型比较的是引用地址,基本类型比较的是值
    3.在数据做为参数传递的时候,基本数据类型是值传递,而引用数据类型是引用传递(地址传递)。
  • 存储位置不同
    1.两者的区别主要在存储方式上:基本类型是放在栈空间的,引用类型是放在堆空间的、但是引用变量是放在栈空间的。

重载和重写的区别?

  • 重载
    发生在同一个类中,方法名必须相同,参数类型不同、个数不同、顺序不同,方法返回值和访问修饰符可以不同。
  • 重写
    重写是子类对父类的允许访问的方法的实现过程进行重新编写,发生在子类中,方法名、参数列表必须相同,返回值范围小于等于父类,抛出的异常范围小于等于父类,访问修饰符范围大于等于父类。另外,如果父类方法访问修饰符为 private 则子类就不能重写该方法。也就是说方法提供的行为改变,而方法的外貌并没有改变。

Java 面向对象编程三大特性: 封装 继承 多态?

  • 封装
    把描述一个对象的属性和行为的代码封装在一个类中,属性用变量定义,行为用方法进行定义,方法可以直接访问同一个对象中的属性。

  • 继承
    子类继承父类的特征和行为。子类可以有父类非私有的方法,属性。子类也可以对父类进行扩展,也可以重写父类的方法。缺点就是提高代码之间的耦合性。

  • 多态
    多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。因为在程序运行时才确定具体的类,这样,不用修改源程序代码,就可以让引用变量绑定到各种不同的类实现上,从而导致该引用调用的具体方法随之改变,即不修改程序代码就可以改变程序运行时所绑定的具体代码,让程序可以选择多个运行状态这就是多态性。

  • 但是在面试过程中有可能会问到Java面向对象的四大特征
    抽象是指强调实体的本质、内在的属性。在系统开发中,抽象指的是在决定如何实现对象之前的对象的意义和行为。使用抽象可以尽可能避免过早考虑一些细节。类实现了对象的数据(即状态)和行为的抽象。

final 在 java 中有什么作用?

final 修饰的类叫最终类,该类不能被继承。
final 修饰的方法不能被重写。
final 修饰的变量不可更改,其不可更改指的是其引用不可修改,对于引用类型值还是可能改变的,举个列子:String 内部对于 value 的定义;而对于基本类型来说就叫做常量了。

final、finally、finalize 有什么区别?

  • final可以修饰类、变量、方法,修饰类表示该类不能被继承、修饰方法表示该方法不能被重写、修饰变量表示该变量是一个常量不能被重新赋值。
  • finally一般作用在try-catch代码块中,在处理异常的时候,通常我们将一定要执行的代码方法finally代码块中,表示不管是否出现异常,该代码块都会执行,一般用来存放一些关闭资源的代码。
  • finalize是一个方法,属于Object类的一个方法,而Object类是所有类的父类,该方法一般由垃圾回收器来调用,当我们调用System的gc()方法的时候,由垃圾回收器调用finalize(),回收垃圾。

String 、StringBuffer 和 StringBuilder 的区别是什么?

  • 线程安全
    StringBuilder是线程不安全的,效率较高;而StringBuffer是线程安全的,效率较低。
  • 数据可变和不可变
    String的值不可变的。底层使用final修饰。
    StringBuffer和StringBuilder的值是可变的,底层使用的是可变字符数组:char[] value;
  • 使用场景
    如果需要操作少量的数据用 String
    单线程操作字符串缓冲区的情况下操作大量数据使用 StringBuilder
    多线程操作字符串缓冲区 下操作大量数据使用 StringBuffer

String 是如何实现不可变的?

  • 首先要明确:String 不可变的是字符串的值不变
  • String 类内部是用 char 数组来保存字符串的值而底层使用final修饰
  • 当我们实例化一个 String 对象并得到其引用后, 构造已经结束了, 即 value 的引用已经不能再变了 。 那么 value的值呢, 理论上是可以改变的, 只要我们拿到 value 的引用, 可以直接通过下标改变他的值 。然而,因为 String并没有提供接口来改变 value 的值,所以value 的值我们从 String 外部获取不到,也改变不了。这才是String才是不可变的真正原因,并不仅仅是使用 final 修饰了 value 数据。

补充:然而,并不是真正的完全不能获取,利用反射可以直接获取类内部属性。

String 为什么设置为不可变?

1.为了实现字符串常量池(只有当字符是不可变的,字符串池才有可能实现)
2.为了线程安全(字符串自己便是线程安全的)
3.为了保证同一个对象调用 hashCode() 都产生相同的值,String 设置为不可变可以对这个条件有很好的支持,这也是 Map 类的 key 使用 String 的原因。

Exception、Error、运行时异常与一般异常有何异同 /java 异常体系

所有的异常都是从Throwable继承而来的

  • Error是错误,对于所有的编译时期的错误以及系统错误都是通过Error抛出的。
  • Exception是异常,它规定的异常是程序本身可以处理的异常。

在这里插入图片描述
常见的RuntimeException(运行时异常):
IndexOutOfBoundsException(下标越界异常)
NullPointerException(空指针异常)
NumberFormatException (String转换为指定的数字类型异常)
ArithmeticException -(算术运算异常 如除数为0)
ArrayStoreException - (向数组中存放与声明类型不兼容对象异常)
SecurityException -(安全异常)
IOException(其他异常)
FileNotFoundException(文件未找到异常。)
IOException(操作输入流和输出流时可能出现的异常。)
EOFException (文件已结束异常)

接口和抽象类的区别?

  • 接口的方法默认是 public,所有方法在接口中不能有实现(Java 8 开始接口方法可以有默认实现),而抽象类可以有非抽象的方法。
  • 接口中的实例变量默认是 final 类型的,而抽象类中则不一定。
  • 一个类可以实现多个接口,但最多只能实现一个抽象类。
  • 一个类实现接口的话要实现接口的所有方法,而抽象类不一定。
  • 接口不能用 new实例化,但可以声明,但是必须引用一个实现该接口的对象。从设计层面来说,抽象是对类的抽象,是一种模板设计,而接口是对行为的抽象,是一种行为的规范。
  • 扩展:
    抽象类必须要有抽象方法吗?
    抽象类中不一定包含抽象方法,但是包含抽象方法的类一定要被声明为抽象类。
    抽象类能使用 final 修饰吗?
    抽象类不能用 final 来修饰。当用 final 修饰一个类时,表明这个类不能被继承。 final 类中的所有成员方法都会被隐式地指定为 final 方法,这明显违背了抽象类存在的意义了。

Object类有哪些常用的方法?

equals 方法,hashCode 方法,toString 方法,wait 和 notify 系列的几个, getclass

== 和 equals 的区别?

  • == 对于基本类型来说是值比较,对于引用类型来说是比较的是引用;而 equals 默认情况下是 ==比较,只是很多类重写了 equals 方法,比如 String、Integer 等把它变成了值比较,所以一般情况下 equals 比较的是值是否相等。

hashCode 与 equals (重要)

  • hashCode()介绍
    1.hashCode() 的作用是获取哈希码,也称为散列码;它实际上是返回一个 int 整数。这个哈希码的作用是确定该对象在哈希表中的索引位置。
    2.散列表存储的是键值对(key-value),它的特点是:能根据“键”快速的检索出对应的“值”。这其中就利用到了散列码!(可以快速找到所需要的对象)
  • 为什么要有 hashCode
    1.我们先以“HashSet 如何检查重复”为例子来说明为什么要有 hashCode: 当你把对象加入 HashSet 时,HashSet 会先计算对象的 hashcode 值来判断对象加入的位置,同时也会与其他已经加入的对象的 hashcode 值作比较,如果没有相符的hashcode,HashSet会假设对象没有重复出现。但是如果发现有相同 hashcode 值的对象,这时会调用 equals()方法来检查 hashcode 相等的对象是否真的相同。如果两者相同,HashSet 就不会让其加入操作成功。如果不同的话,就会重新散列到其他位置。这样我们就大大减少了 equals 的次数,相应就大大提高了执行速度。
    2.通过我们可以看出:hashCode() 的作用就是获取哈希码,也称为散列码;它实际上是返回一个int整数。这个哈希码的作用是快速确定该对象在哈希表中的索引位置。
  • hashCode()与equals()的相关规定
    1.如果两个对象相等,则hashcode一定也是相同的
    2.两个对象相等,对两个对象分别调用equals方法都返回true
    3.两个对象有相同的hashcode值,它们也不一定是相等的
    4.因此,equals 方法被覆盖过,则 hashCode 方法也必须被覆盖
    5.hashCode() 的默认行为是对堆上的对象产生独特值。如果没有重写 hashCode(),则该 class 的两个对象无论如何都不会相等(即使这两个对象指向相同的数据)
  • 哪些场景下,子类需要重写 equals 方法和 hashCode 方法?
    需要判断两个对象状态的相等性的时候
  • 为什么要重写 hashcode( ) 还要重写 equals( ) ?反之亦可问。
    重写equals方法是为了按我们自己的想法来比较两个对象是否相等。如果不重写hashCode方法,可能出现具有相同含义的不同对象(他们的hashCode不同)的情况。而如果只重写 hashCode 不重写 equals 方法,因为 equals其实就是 == ,只是判断两个对象是否是同一个对象,所以不能得到我们想要的结果。所以需要同时重写equals和hashCode方法,目的是为了准确定位到我们期望的key。
    通俗点的解释:
    1.hashcode就类似 门牌号,小区非常大但定位你住哪里告诉门牌号就可以,非常快速定位到(非常像组数下标)。
    2.equals就是找到门牌号后需要比较里面具体的房间,少一个都不可以。

Java序列化中如果有些字段不想进行序列化,怎么办?

  • 对于不想进行序列化的变量,使用transient关键字修饰。
  • transient关键字的作用是:阻止实例中那些用此关键字修饰的的变量序列化;当对象被反序列化时,被transient修饰的变量值不会被持久化和恢复。transient只能修饰变量,不能修饰类和方法。
  • ArrayList 中存储数据的数组 elementData 是用 transient 修饰的,因为这个数组是动态扩展的,并不是所有的空间都被使用,因此就不需要所有的内容都被序列化。通过重写序列化和反序列化方法,使得可以只序列化数组中有内容的那部分数据。

幂等性

  • 幂等(idempotent、idempotence)是一个数学与计算机学概念,常见于抽象代数中,简单的来说就是一个操作多次执行产生的结果与一次执行产生的结果一致。有些系统操作天生就具有幂等性例如数据库的select语句,但更多时候是需要程序员来做保证的,尤其是在分布式系统环境中,接口能不能做到保证幂等性对系统的影响可能是非常大的,例如很常见的支付下单等场景,由于分布式环境中网络的复杂性,用户误操作,网络抖动,消息重复,服务超时导致业务自动重试等等各种情况都可能会使线上数据产生了不一致,造成生产事故。

怎么防止前端重复提交?

1.提交按钮后屏蔽提交按钮(前端js控制)
2.前端生产唯一id, 后端在数据库设计时业务字段加唯一约束,防止数据库插入重复数据。
3.利用Session防止表单重复提交
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值