Java 知识点总结笔记(篇1)

Java 知识点总结笔记(篇1)

一、基础语法

1、数值型

整数类型(byte,short,int,long)
浮点类型(float,double)
字符型(char)
布尔型(boolean)

在这里插入图片描述

2、移位运算

用最有效率的方法:计算 2 乘以 8;

2 << 3(左移 3 位相当于乘以 2 的 3 次方,右移 3 位相当于除以 2 的 3 次方);

左移“<<”和“>>”右移 :若为负数,则高位补1;例如:-2 << 3 = -16

“>>>”表示无符号右移,也叫逻辑右移,即若该数为正, 则高位补0; 而若该数为负数,则右移后高位同样补0;例如:-12 无符号右移2位

-12 >>> 2 = 1073741821;

步骤:
因为是整数,4个字节,32位,故:

原码:1000,0000 0000,0000 0000,0000 0000,1100

补码(符号为不变,其他位取反加1):1111,1111 1111,1111 1111,1111 1111,0100

符号位不变,右移2位:0011,1111 1111,1111 1111,1111 1111,1101 = 1073741821

3、强制转换

short s1 = 1; s1 = s1 + 1;有错吗?

short s1 = 1; s1 += 1;有错吗

对于 short s1 = 1; s1 = s1 + 1;由于 1 是 int 类型,因此 s1+1 运算结果也是 int型,需要强制转换
类型才能赋值给 short 型。
而 short s1 = 1; s1 += 1;可以正确编译,因为 s1+= 1;相当于 s1 = (short(s1 + 1);其中有隐含的强
制类型转换。

4、访问修饰符

定义:

Java中,可以使用访问修饰符来保护对类、变量、方法和构造方法的访问。Java 支持 4 种
不同的访问权限。

分类

private : 在同一类内可见。使用对象:变量、方法。 注意:不能修饰类(外部类)

default (即缺省,什么也不写,不使用任何关键字): 在同一包内可见,不使用任何修饰符。
使用对象:类、接口、变量、方法。

protected : 对同一包内的类和所有子类可见。使用对象:变量、方法。 注意:不能修饰类
(外部类)。

public : 对所有类可见。使用对象:类、接口、变量、方法

在这里插入图片描述

5、运算符

&和&&的区别

&运算符有两种用法:(1)按位与;(2)逻辑与。

&&运算符是短路与运算。

逻辑与跟短路与的差别是非常巨大的,虽然二者都要求运算符左右两端
的布尔值都是true 整个表达式的值才是 true。&&之所以称为短路运算,是因为如果&&左边的表
达式的值是 false,右边的表达式会被直接短路掉,不会进行运算

注意:逻辑或运算符(|)和短路或运算符(||)的差别也是如此。

6、关键字

final 有什么用?

用于修饰类、属性和方法;

  1. 被final修饰的类不可以被继承
  2. 被final修饰的方法不可以被重写
  3. 被final修饰的变量不可以被改变,被final修饰不可变的是变量的引用,而不是引用指向的内容,
  4. 引用指向的内容是可以改变的
final finally finalize区别
  1. final可以修饰类、变量、方法,修饰类表示该类不能被继承、修饰方法表示该方法不能被重写、
    修饰变量表示该变量是一个常量不能被重新赋值。
  2. finally一般作用在try-catch代码块中,在处理异常的时候,通常我们将一定要执行的代码方法
    finally代码块 中,表示不管是否出现异常,该代码块都会执行,一般用来存放一些关闭资源的代
    码。
  3. finalize是一个方法,属于Object类的一个方法,而Object类是所有类的父类,该方法一般由垃圾
    回收器来调 用,当我们调用System.gc() 方法的时候,由垃圾回收器调用finalize(),回收垃圾,一
    个对象是否可回收的最后判断。
this与super的区别
  1. super: 它引用当前对象的直接父类中的成员(用来访问直接父类中被隐藏的父类中成员数据或函
    数,基类与派生类中有相同成员定义时如:super.变量名 super.成员函数据名(实参)
  2. this:它代表当前对象名(在程序中易产生二义性之处,应使用this来指明当前对象;如果函数的
    形参与类中的成员数据同名,这时需用this来指明成员变量名)
  3. super()和this()类似,区别是,super()在子类中调用父类的构造方法,this()在本类内调用本类的其
    它构造方法。
  4. super()和this()均需放在构造方法内第一行。
  5. 尽管可以用this调用一个构造器,但却不能调用两个。
  6. this和super不能同时出现在一个构造函数里面,因为this必然会调用其它的构造函数,其它的构
    造函数必然也会有super语句的存在,所以在同一个构造函数里面有相同的语句,就失去了语句的
    意义,编译器也不会通过。
  7. this()和super()都指的是对象,所以,均不可以在static环境中使用。包括:static变量,static方
    法,static语句块。
  8. 从本质上讲,this是一个指向本对象的指针, 然而super是一个Java关键字。
static存在的主要意义
  • static的主要意义是在于创建独立于具体对象的域变量或者方法。以致于即使没有创建对象,也能
    使用属性和调用方法!
  • 怎么理解 “被类的实例对象所共享” 这句话呢?就是说,一个类的静态成员,它是属于大伙的【大
    伙指的是这个类的多个对象实例,我们都知道一个类可以创建多个实例!】,所有的类对象共享
    的,不像成员变量是自个的【自个指的是这个类的单个实例对象】。
  • static关键字还有一个比较关键的作用就是 用来形成静态代码块以优化程序性能。static块可以置
    于类中的任何地方,类中可以有多个static块。在类初次被加载的时候,会按照static块的顺序来执
    行每个static块,并且只会执行一次。
  • 为什么说static块可以用来优化程序性能,是因为它的特性:只会在类加载的时候执行一次。因此,
    很多时候会将一些只需要进行一次的初始化操作都放在static代码块中进行。
static的独特之处
  1. 被static修饰的变量或者方法是独立于该类的任何对象,也就是说,这些变量和方法不属于任
    何一个实例对象,而是被类的实例对象所共享。
  2. 在该类被第一次加载的时候,就会去加载被static修饰的部分,而且只在类第一次使用时加载
    并进行初始化,注意这是第一次用就要初始化,后面根据需要是可以再次赋值的。
  3. static变量值在类加载的时候分配空间,以后创建类对象的时候不会重新分配。赋值的话,是
    可以任意赋值的!
  4. 被static修饰的变量或者方法是优先于对象存在的,也就是说当一个类加载完毕之后,即便没
    有创建对象,也可以去访问。
static应用场景

因为static是被类的实例对象所共享,因此如果某个成员变量是被所有对象所共享的,那么这个成
员变量就应该定义为静态变量。

因此比较常见的static应用场景有:

  1. 修饰成员变量
  2. 修饰成员方法
  3. 静态代码块
  4. 修饰类【只能修饰内部类也就是静态内部类】
  5. 静态导包
static注意事项
  1. 静态只能访问静态。
  2. 非静态既可以访问非静态的,也可以访问静态的。
在 Java 中,如何跳出当前的多重嵌套循环

在Java中,要想跳出多重循环,可以在外面的循环语句前定义一个标号,然后在里层循环体的代码
中使用带有标号的break 语句,即可跳出外层循环。例如:

public static void main(String[] args) {
  ok:
  for (int i = 0; i < 10; i++) {
    for (int j = 0; j < 10; j++) {
      System.out.println("i=" + i + ",j=" + j);
      if (j == 5) {
        break ok;
      }
    }
  }
}

二、面向对象

1、面向对象概述

面向对象和面向过程的区别
面向过程:

优点:性能比面向对象高,因为类调用时需要实例化,开销比较大,比较消耗资源;比如单片机、
嵌入式开发、Linux/Unix等一般采用面向过程开发,性能是最重要的因素。
缺点:没有面向对象易维护、易复用、易扩展

面向对象:

优点:易维护、易复用、易扩展,由于面向对象有封装、继承、多态性的特性,可以设计出
低耦合的系统,使系统更加灵活、更加易于维护
缺点:性能比面向过程低

特点
  1. 面向过程是具体化的,流程化的,解决一个问题,你需要一步一步的分析,一步一步的实现。
  2. 面向对象是模型化的,你只需抽象出一个类,这是一个封闭的盒子,在这里你拥有数据也拥有解决问题的方法。需要什么功能直接使用就可以了,不必去一步一步的实现,至于这个功能是如何实现的,管我们什么事?我们会用就可以了。
  3. 面向对象的底层其实还是面向过程,把面向过程抽象成类,然后封装,方便我们使用的就是面向对象了。

2、面向对象三大特性

特点

抽象:抽象是将一类对象的共同特征总结出来构造类的过程,包括数据抽象和行为抽象两方面。抽
象只关注对象有哪些属性和行为,并不关注这些行为的细节是什么。

封装:把一个对象的属性私有化,同时提供一些可以被外界访问的属性的方法,如果属性不想被外界
访问,我们大可不必提供方法给外界访问。但是如果一个类没有提供给外界访问的方法,那么这个
类也没有什么意义了。

继承:是使用已存在的类的定义作为基础建立新类的技术,新类的定义可以增加新的数据或新的功
能,也可以用父类的功能,但不能选择性地继承父类。通过使用继承我们能够非常方便地复用以前
的代码。

多态:父类或接口定义的引用变量可以指向子类或具体实现类的实例对象。提高了程序的拓展性。
在Java中有两种形式可以实现多态:继承(多个子类对同一方法的重写)和接口(实现接口并覆盖
接口中同一方法)。

注意
  1. 子类拥有父类非 private 的属性和方法。
  2. 子类可以拥有自己属性和方法,即子类可以对父类进行扩展。
  3. 子类可以用自己的方式实现父类的方法。

3、什么是多态机制?Java语言是如何实现多态的?

  1. 多态:就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程
    时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用
    变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。因为在程序运
    行时才确定具体的类,这样,不用修改源程序代码,就可以让引用变量绑定到各种不同的类实现
    上,从而导致该引用调用的具体方法随之改变,即不修改程序代码就可以改变程序运行时所绑定的具体代码,让程序可以选择多个运行状态,这就是多态性。
  2. 多态分为编译时多态运行时多态。其中编辑时多态是静态的,主要是指方法的重载,它是根据参
    数列表的不同来区分不同的函数,通过编辑之后会变成两个不同的函数,在运行时谈不上多态。而
    运行时多态是动态的,它是通过动态绑定来实现的,也就是我们所说的多态性。
多态的实现

Java实现多态有三个必要条件:继承、重写、向上转型。

继承:在多态中必须存在有继承关系的子类和父类。

重写:子类对父类中某些方法进行重新定义,在调用这些方法时就会调用子类的方法。

向上转型:在多态中需要将子类的引用赋给父类对象,只有这样该引用才能够具备技能调用
父类的方法和子类的方法。

只有满足了上述三个条件,我们才能够在同一个继承结构中使用统一的逻辑实现代码处理不同的对象,从而达到执行不同的行为。

对于Java而言,它多态的实现机制遵循一个原则:当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。

4、面向对象五大基本原则是什么(可选)

  1. 单一职责原则SRP(Single Responsibility Principle):类的功能要单一,不能包罗万象,跟杂货铺似的。
  2. 开放封闭原则OCP(Open-Close Principle):一个模块对于拓展是开放的,对于修改是封闭的,想要增加功能热烈欢迎,想要修改,哼,一万个不乐意。
  3. 里式替换原则LSP(the Liskov Substitution Principle LSP):子类可以替换父类出现在父类能够出现的任何地方。比如你能代表你爸去你姥姥家干活。哈哈~~
  4. 依赖倒置原则DIP(the Dependency Inversion Principle DIP):高层次的模块不应该依赖于低层次的模块,他们都应该依赖于抽象。抽象不应该依赖于具体实现,具体实现应该依赖于抽象。就是你出国要说你是中国人,而不能说你是哪个村子的。比如说中国人是抽象的,下面有具体的xx省,xx市,xx县。你要依赖的抽象是中国人,而不是你是xx村的。
  5. 接口分离原则ISP(the Interface Segregation Principle ISP):设计时采用多个与特定客户类有关的接口比采用一个通用的接口要好。就比如一个手机拥有打电话,看视频,玩游戏等功能,把这几个功能拆分成不同的接口,比在一个接口里要好的多。

三、类与接口

1、抽象类和接口的对比

  • 抽象类是用来捕捉子类的通用特性的。接口是抽象方法的集合。
  • 从设计层面来说,抽象类是对类的抽象,是一种模板设计,接口是行为的抽象,是一种行为的规
    范。
相同点:
  1. 接口和抽象类都不能实例化
  2. 都位于继承的顶端,用于被其他实现或继承
  3. 都包含抽象方法,其子类都必须覆写这些抽象方法
不同点:
参数抽象类接口
声明抽象类使用abstract关键字声明接口使用interface关键字声明
实现子类使用extends关键字来继承抽象类。如果子类不是抽象类的话,它需要提供抽象类中所有声明的方法的实现子类使implements关键字来实现接口。它需要提供接口中所有声明的方法的实现
构造器抽象类可以有构造器接口不能有构造器
访问修饰符抽象类中的方法可以是任意访问修饰符接口方法默认修饰符是public。并且不允许定义为 private 或protected
多继承一个类最多只能继承一个抽象类一个类可以实现多个接口
字段声明抽象类的字段声明可以是任意的接口的字段默认都是 static 和 fina的

备注: Java8中接口中引入默认方法和静态方法,以此来减少抽象类和接口之间的差异。

现在,我们可以为接口提供默认实现的方法了,并且不用强制子类来实现它。

接口和抽象类各有优缺点,在接口和抽象类的选择上,必须遵守这样一个原则:

  1. 行为模型应该总是通过接口而不是抽象类定义,所以通常是优先选用接口,尽量少用抽象
    类。
  2. 选择抽象类的时候通常是如下情况:需要定义子类的行为,又要为子类提供通用的功能。

2、普通类和抽象类有哪些区别?

  • 普通类不能包含抽象方法,抽象类可以包含抽象方法。
  • 抽象类不能直接实例化,普通类可以直接实例化。

3、抽象类能使用 final 修饰吗?

不能,定义抽象类就是让其他类继承的,如果定义为 final 该类就不能被继承,这样彼此就会产生矛盾,所以 final 不能修饰抽象类。

4、创建一个对象用什么关键字?对象实例与对象引用有何不同?

new关键字,new创建对象实例(对象实例在堆内存中),对象引用指向对象实例(对象引用存放在栈内存中)。一个对象引用可以指向0个或1个对象(一根绳子可以不系气球,也可以系一个气球);一个对象可以有n个引用指向它(可以用n条绳子系住一个气球)。

四、变量与方法

1、成员变量与局部变量的区别有哪些

变量:在程序执行的过程中,在某个范围内其值可以发生改变的量。从本质上讲,变量其实是内存中的一小块区域.

成员变量:方法外部,类内部定义的变量
局部变量:类的方法中的变量。

成员变量和局部变量的区别
作用域

成员变量:针对整个类有效。
局部变量:只在某个范围内有效。(一般指的就是方法,语句体内)

存储位置

成员变量:随着对象的创建而存在,随着对象的消失而消失,存储在堆内存中。
局部变量:在方法被调用,或者语句被执行的时候存在,存储在栈内存中。当方法调用完,或者语句结束后,就自动释放。

生命周期

成员变量:随着对象的创建而存在,随着对象的消失而消失
局部变量:当方法调用完,或者语句结束后,就自动释放。

初始值

成员变量:有默认初始值。
局部变量:没有默认初始值,使用前必须赋值。

2、在Java中定义一个不做事且没有参数的构造方法的作用

Java程序在执行子类的构造方法之前,如果没有用super()来调用父类特定的构造方法,则会调用父类中“没有参数的构造方法”。因此,如果父类中只定义了有参数的构造方法,而在子类的构造方法中又没有用super()来调用父类中特定的构造方法,则编译时将发生错误,因为Java程序在父类中找不到没有参数的构造方法可供执行。解决办法是在父类里加上一个不做事且没有参数的构造方法。

3、在调用子类构造方法之前会先调用父类没有参数的构造方法,其目的是?

帮助子类做初始化工作。

4、一个类的构造方法的作用是什么?若一个类没有声明构造方法,改程序能正确执行吗?为什么?

主要作用是完成对类对象的初始化工作。可以执行。因为一个类即使没有声明构造方法也会有默认的不带参数的构造方法。

5、构造方法有哪些特性?

  1. 名字与类名相同;
  2. 没有返回值,但不能用void声明构造函数;
  3. 生成类的对象时自动执行,无需调用。

6、静态变量和实例变量区别

  • 静态变量: 静态变量由于不属于任何实例对象,属于类的,所以在内存中只会有一份,在类的加载过程中,JVM只为静态变量分配一次内存空间。
  • 实例变量: 每次创建对象,都会为每个对象分配成员变量内存空间,实例变量是属于实例对象的,在内存中,创建几次对象,就有几份成员变量。

7、静态变量与普通变量区别

  • static变量也称作静态变量,静态变量和非静态变量的区别是:静态变量被所有的对象所共享,在内存中只有一个副本,它当且仅当在类初次加载时会被初始化。而非静态变量是对象所拥有的,在创建对象的时候被初始化,存在多个副本,各个对象拥有的副本互不影响。
  • 还有一点就是static成员变量的初始化顺序按照定义的顺序进行初始化。

8、静态方法和实例方法有何不同?

静态方法和实例方法的区别主要体现在两个方面:

  • 在外部调用静态方法时,可以使用"类名.方法名"的方式,也可以使用"对象名.方法名"的方式。而实例方法只有后面这种方式。也就是说,调用静态方法可以无需创建对象。
  • 静态方法在访问本类的成员时,只允许访问静态成员(即静态成员变量和静态方法),而不允许访问实例成员变量和实例方法;实例方法则无此限制

9、在一个静态方法内调用一个非静态成员为什么是非法的?

由于静态方法可以不通过对象进行调用,因此在静态方法里,不能调用其他非静态变量,也不可以访问非静态变量成员。

五、内部类

1、 什么是内部类?

在Java中,可以将一个类的定义放在另外一个类的定义内部,这就是内部类。内部类本身就是类的一个属性,与其他属性定义方式一致。

2、内部类的分类有哪些

内部类可以分为四种:成员内部类、局部内部类、匿名内部类和静态内部类

静态内部类

定义在类内部的静态类,就是静态内部类:

public class Outer {
  private static int radius = 1;
  static class StaticInner {
    public void visit() {
      System.out.println("visit outer static variable:" + radius);
    }
  }
}

静态内部类可以访问外部类所有的静态变量,而不可访问外部类的非静态变量;静态内部类的创建方式, new 外部类.静态内部类() ,如下:

Outer.StaticInner inner = new Outer.StaticInner();
inner.visit();
成员内部类

定义在类内部,成员位置上的非静态类,就是成员内部类。

public class Outer {
  private static int radius = 1;
  private int count =2;
   class Inner {
    public void visit() {
      System.out.println("visit outer static variable:" + radius);
      System.out.println("visit outer  variable:" + count);
    }
  }
}

成员内部类可以访问外部类所有的变量和方法,包括静态和非静态,私有和公有。成员内部类依赖于外部类的实例,它的创建方式 外部类实例.new 内部类() ,如下:

Outer outer = new Outer();
Outer.Inner inner = outer.new Inner();
inner.visit();
局部内部类

定义在方法中的内部类,就是局部内部类。

public class Outer {
  private int out_a = 1;
  private static int STATIC_b = 2;
  public void testFunctionClass(){
    int inner_c =3;
    class Inner {
      private void fun(){
        System.out.println(out_a);
        System.out.println(STATIC_b);
        System.out.println(inner_c);
      }
    }
    Inner inner = new Inner();
    inner.fun();
  }
  public static void testStaticFunctionClass(){
    int d =3;
    class Inner {
      private void fun(){
// System.out.println(out_a); 编译错误,定义在静态方法中的局部类不可以访问外部类的实例变量
			 System.out.println(STATIC_b);
        System.out.println(d);
      }
    }
    Inner inner = new Inner();
    inner.fun();
  }
}

定义在实例方法中的局部类可以访问外部类的所有变量和方法,定义在静态方法中的局部类只能访问外部类的静态变量和方法。局部内部类的创建方式,在对应方法内, new 内部类() ,如下:

public static void testStaticFunctionClass(){
  class Inner {
  }
  Inner inner = new Inner();
 }
匿名内部类

匿名内部类就是没有名字的内部类,日常开发中使用的比较多:

public class Outer {
  private void test(final int i) {
    new Service() {
      public void method() {
        for (int j = 0; j < i; j++) {
          System.out.println("匿名内部类" );
        }
      }
    }.method();
  }
 }
 //匿名内部类必须继承或实现一个已有的接口
interface Service{
  void method();
}

除了没有名字,匿名内部类还有以下特点:

  1. 匿名内部类必须继承一个抽象类或者实现一个接口。
  2. 匿名内部类不能定义任何静态成员和静态方法。
  3. 当所在的方法的形参需要被匿名内部类使用时,必须声明为 final。
  4. 匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。

匿名内部类创建方式:

new/接口{
 //匿名内部类实现部分
}

3、内部类的优点

  1. 一个内部类对象可以访问创建它的外部类对象的内容,包括私有数据!
  2. 内部类不为同一包的其他类所见,具有很好的封装性;
  3. 内部类有效实现了“多重继承”,优化 java 单继承的缺陷。
  4. 匿名内部类可以很方便的定义回调。

4、内部类有哪些应用场景

  1. 一些多算法场合
  2. 解决一些非面向对象的语句块。
  3. 适当使用内部类,使得代码更加灵活和富有扩展性。
  4. 当某个类除了它的外部类,不再被其他的类使用时。

5、局部内部类和匿名内部类访问局部变量的时候,为什么变量必须要加上final?

局部内部类和匿名内部类访问局部变量的时候,为什么变量必须要加上final呢?它内部原理是什么呢?先看这段代码:

public class Outer {
  void outMethod(){
    final int a =10;
    class Inner {
      void innerMethod(){
        System.out.println(a);
      }
    }
  }
}

以上例子,为什么要加final呢?是因为生命周期不一致, 局部变量直接存储在栈中,当方法执行结束后,非final的局部变量就被销毁。而局部内部类对局部变量的引用依然存在,如果局部内部类要调用局部变量时,就会出错。加了final,可以确保局部内部类使用的变量与外层的局部变量区分开,解决了这个问题。

6、内部类相关,看程序说出运行结果

代码:
public class Outer {
 private int age = 12;
 class Inner {
   private int age = 13;
   public void print() {
     int age = 14;
     System.out.println("局部变量:" + age);
     System.out.println("内部类变量:" + this.age);
     System.out.println("外部类变量:" + Outer.this.age);
   }
 }
 public static void main(String[] args) {
   Outer.Inner in = new Outer().new Inner();
   in.print();
  }
}
运行结果:
局部变量:14
内部类变量:13
外部类变量:12

六、重写与重载

1、构造器(constructor)是否可被重写(override)

构造器不能被继承,因此不能被重写,但可以被重载。

2、重载(Overload)和重写(Override)的区别。重载的方法能否根据返回类型进行区分?
  1. 方法的重载和重写都是实现多态的方式,区别在于前者实现的是编译时的多态性,而后者实现的是运行时的多态性。
  2. 重载:发生在同一个类中,方法名相同参数列表不同(参数类型不同、个数不同、顺序不同),与方法返回值和访问修饰符无关,即重载的方法不能根据返回类型进行区分。
  3. 重写:发生在父子类中,方法名、参数列表必须相同,返回值小于等于父类,抛出的异常小于等于父类,访问修饰符大于等于父类(里氏代换原则);如果父类方法访问修饰符为private则子类中就不是重写。

七、对象相等判断

1、== 和 equals 的区别是什么

== : 它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象。(基本数据类型 == 比较的是值,引用数据类型 == 比较的是内存地址)。

equals() : 它的作用也是判断两个对象是否相等。但它一般有两种使用情况:

  • 情况1:类没有覆盖 equals() 方法。则通过 equals() 比较该类的两个对象时,等价于通过“==”比较这两个对象。

  • 情况2:类覆盖了 equals() 方法。一般,我们都覆盖 equals() 方法来两个对象的内容相等;若它们的内容相等,则返回 true (即,认为这两个对象相等)。

    举个例子: 说明:

    • String中的equals方法是被重写过的,因为object的equals方法是比较的对象的内存地址,而String的equals方法比较的是对象的值。

    • 当创建String类型的对象时,虚拟机会在常量池中查找有没有已经存在的值和要创建的值相同的对象,如果有就把它赋给当前引用。如果没有就在常量池中重新创建一个String对象。

2、hashCode 与 equals (重要)

  1. HashSet如何检查重复
  2. 两个对象的 hashCode() 相同,则 equals() 也一定为 true,对吗?
  3. hashCode和equals方法的关系
  4. 面试官可能会问你:“你重写过 hashcode 和 equals 么,为什么重写equals时必须重写hashCode方法?”
hashCode()介绍

我们以“HashSet 如何检查重复”为例子来说明为什么要有 hashCode:

当你把对象加入 HashSet 时,HashSet 会先计算对象的 hashcode 值来判断对象加入的位置,同时也会与其他已经加入的对象的 hashcode 值作比较,如果没有相符的hashcode,HashSet会假设对象没有重复出现。但是如果发现有相同 hashcode 值的对象,这时会调用 equals()方法来检查hashcode 相等的对象是否真的相同。如果两者相同,HashSet 就不会让其加入操作成功。如果不同的话,就会重新散列到其他位置。(摘自我的Java启蒙书《Head first java》第二版)。这样我们就大大减少了 equals 的次数,相应就大大提高了执行速度。

hashCode()与equals()的相关规定
  • 如果两个对象相等,则hashcode一定也是相同的
  • 两个对象相等,对两个对象分别调用equals方法都返回true
  • 两个对象有相同的hashcode值,它们也不一定是相等的

因此,equals 方法被覆盖过,则 hashCode 方法也必须被覆盖

hashCode() 的默认行为是对堆上的对象产生独特值。如果没有重写 hashCode(),则该 class 的两个对象无论如何都不会相等(即使这两个对象指向相同的数据 )。

3、对象的相等与指向他们的引用相等,两者有什么不同?

对象的相等:比的是内存中存放的内容是否相等而。

引用相等:比较的是他们指向的内存地址是否相等。

八、值传递

1、当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?

是值传递。Java 语言的方法调用只支持参数的值传递。当一个对象实例作为一个参数被传递到方法中时,参数的值就是对该对象的引用。对象的属性可以在被调用过程中被改变,但对对象引用的改变是不会影响到调用者的

2、 为什么 Java 中只有值传递

  • 首先回顾一下在程序设计语言中有关将参数传递给方法(或函数)的一些专业术语。按值调用(call by value)表示方法接收的是调用者提供的值,而按引用调用(call by reference)表示方法接收的是调用者提供的变量地址。一个方法可以修改传递引用所对应的变量值,而不能修改传递值调用所对应的变量值。 它用来描述各种程序设计语言(不只是Java)中方法参数传递方式。
  • Java程序设计语言总是采用按值调用。也就是说,方法得到的是所有参数值的一个拷贝,也就是说,方法不能修改传递给它的任何参数变量的内容。

总结:Java程序设计语言对对象采用的不是引用调用,实际上,对象引用是按值传递的。
下面再总结一下:Java中方法参数的使用情况:

  1. 一个方法不能修改一个基本数据类型的参数(即数值型或布尔型》
  2. 一个方法可以改变一个对象参数的状态。
  3. 一个方法不能让对象参数引用一个新的对象。

3、值传递和引用传递有什么区别

值传递:指的是在方法调用时,传递的参数是按值的拷贝传递,传递的是值的拷贝,也就是说传递后就互不相关了。

引用传递:指的是在方法调用时,传递的参数是按引用进行传递,其实传递的引用的地址,也就是变量所对应的内存空间的地址。传递的是值的引用,也就是说传递前和传递后都指向同一个引用(也就是同一个内存空间)。

九、Java包

1、java 中 IO 流分为几种

  • 按照流的流向分,可以分为输入流和输出流;
  • 按照操作单元划分,可以划分为字节流和字符流;
  • 按照流的角色划分为节点流和处理流。

Java Io 流共涉及40多个类,这些类看上去很杂乱,但实际上很有规则,而且彼此之间存在非常紧密的联系,
Java I0流的40多个类都是从如下4个抽象类基类中派生出来的。

  • InputStream/Reader: 所有的输入流的基类,前者是字节输入流,后者是字符输入流。
  • OutputStream/Writer: 所有输出流的基类,前者是字节输出流,后者是字符输出流。

按操作方式分类结构图:

在这里插入图片描述

按操作对象分类结构图:

在这里插入图片描述

2、BIO,NIO,AIO 有什么区别?

简答:

BIO:Block IO 同步阻塞式 IO,就是我们平常使用的传统 IO,它的特点是模式简单使用方
便,并发处理能力低。

NIO:Non IO 同步非阻塞 IO,是传统 IO 的升级,客户端和服务器端通过 Channel(通道)
通讯,实现了多路复用。

AIO:Asynchronous IO 是 NIO 的升级,也叫 NIO2,实现了异步非堵塞 IO ,异步 IO 的操
作基于事件和回调机制。

详细回答:

BIO (Blocking I/O): 同步阻塞I/O模式,数据的读取写入必须阻塞在一个线程内等待其完
成。在活动连接数不是特别高(小于单机1000)的情况下,这种模型是比较不错的,可以让
每一个连接专注于自己的 I/O 并且编程模型简单,也不用过多考虑系统的过载、限流等问
题。线程池本身就是一个天然的漏斗,可以缓冲一些系统处理不了的连接或请求。但是,当
面对十万甚至百万级连接的时候,传统的 BIO 模型是无能为力的。因此,我们需要一种更高
效的 I/O 处理模型来应对更高的并发量。

NIO (New I/O): NIO是一种同步非阻塞的I/O模型,在Java 1.4 中引入了NIO框架,对应
java.nio 包,提供了 Channel , Selector,Buffer等抽象。NIO中的N可以理解为Non-
blocking,不单纯是New。它支持面向缓冲的,基于通道的I/O操作方法。 NIO提供了与传统
BIO模型中的 Socket 和 ServerSocket 相对应的 SocketChannel 和ServerSocketChannel 两种不同的套接字通道实现,两种通道都支持阻塞和非阻塞两种模式。阻塞模式使用就像传统中的支持一样,比较简单,但是性能和可靠性都不好;非阻塞模式正好与之相反。对于低负载、低并发的应用程序,可以使用同步阻塞I/O来提升开发速率和更好的维护性;对于高负载、高并发的(网络)应用,应使用 NIO 的非阻塞模式来开发

AIO (Asynchronous I/O): AIO 也就是 NIO 2。在 Java 7 中引入了 NIO 的改进版 NIO 2,它
是异步非阻塞的IO模型。异步 IO 是基于事件和回调机制实现的,也就是应用操作之后会直接
返回,不会堵塞在那里,当后台处理完成,操作系统会通知相应的线程进行后续的操作。AIO是异步IO的缩写,虽然 NIO 在网络操作中,提供了非阻塞的方法,但是 NIO 的 IO 行为还是
同步的。对于 NIO 来说,我们的业务线程是在 IO 操作准备好时,得到通知,接着就由这个
线程自行进行 IO 操作,IO操作本身是同步的。查阅网上相关资料,我发现就目前来说 AIO
的应用还不是很广泛,Netty 之前也尝试使用过 AIO,不过又放弃了。

3、 Files的常用方法都有哪些?

  1. Files. exists():检测文件路径是否存在。
  2. Files. createFile():创建文件。
  3. Files. createDirectory():创建文件夹。
  4. Files. delete():删除一个文件或目录。
  5. Files. copy():复制文件。
  6. Files. move():移动文件。
  7. Files. size():查看文件个数。
  8. Files. read():读取文件。
  9. Files. write():写入文件。

十、反射

1、什么是反射机制?

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任
意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法
的功能称为java语言的反射机制。

静态编译和动态编译

静态编译:在编译时确定类型,绑定对象
动态编译:运行时确定类型,绑定对象

2、反射机制优缺点

优点: 运行期类型的判断,动态加载类,提高代码灵活度。
缺点: 性能瓶颈:反射相当于一系列解释操作,通知 JVM 要做的事情,性能比直接的java代码要
慢很多。

3、反射机制的应用场景有哪些?

反射是框架设计的灵魂:

在我们平时的项目开发过程中,基本上很少会直接使用到反射机制,但这不能说明反射机制没有用,实际上有很多设计、开发都与反射机制有关,例如模块化的开发,通过反射去调用对应的字节码;动态代理设计模式也采用了反射机制,还有我们日常使用的 Spring/Hibernate 等框架也大量使用到了反射机制。

举例:

①我们在使用JDBC连接数据库时使用Class.forName()通过反射加载数据库的驱动程序;

②Spring框架也用到很多反射机制,最经典的就是xml的配置模式。Spring 通过 XML 配置模式装
载 Bean 的过程:

  1. 将程序内所有 XML 或 Properties 配置文件加载入内存中;
  2. Java类里面解析xml或properties里面的内容,得到对应实体类的字节码字符串以及相关的属性信息;
  3. 使用反射机制,根据这个字符串获得某个类的Class实例;
  4. 动态配置实例的属性

4、Java获取反射的三种方法

  1. 通过new对象实现反射机制
  2. 通过路径实现反射机制
  3. 通过类名实现反射机制

例如:

public class Student {
 private int id;
 String name;
 protected boolean sex;
 public float score;
}
public class Get {
 //获取反射机制三种方式
public static void main(String[] args) throws ClassNotFoundException {
   //方式一(通过建立对象)
   Student stu = new Student();
   Class classobj1 = stu.getClass();
   System.out.println(classobj1.getName());
   //方式二(所在通过路径-相对路径)
Class classobj2 = Class.forName("fanshe.Student");
   System.out.println(classobj2.getName());
   //方式三(通过类名)
Class classobj3 = Student.class;
   System.out.println(classobj3.getName());
 }
}

十一、常用API

String相关

1、字符型常量和字符串常量的区别

形式上: 字符常量是单引号引起的一个字符;字符串常量是双引号引起的若干个字符

含义上: 字符常量相当于一个整形值(ASCII值),可以参加表达式运算;字符串常量代表一个地址值(该字符串在内存中存放位置)

占内存大小:字符常量只占一个字节;字符串常量占若干个字节(至少一个字符结束标志)

2、什么是字符串常量池?

字符串常量池位于堆内存中,专门用来存储字符串常量,可以提高内存的使用率,避免开辟多块空
间存储相同的字符串,在创建字符串时 JVM 会首先检查字符串常量池,如果该字符串已经存在池
中,则返回它的引用,如果不存在,则实例化一个字符串放到池中,并返回其引用。

3、String 是最基本的数据类型吗

不是。Java 中的基本数据类型只有 8 个 :byte、short、int、long、float、double、char、
boolean;除了基本类型(primitive type),剩下的都是引用类型(referencetype),Java 5 以后引入的枚举类型也算是一种比较特殊的引用类型。

扩展:

这是很基础的东西,但是很多初学者却容易忽视,Java 的 8 种基本数据类型中不包括 String,基本数据类型中用来描述文本数据的是 char,但是它只能表示单个字符,比如 ‘a’,‘好’ 之类的,如果要描述一段文本,就需要用多个 char 类型的变量,也就是一个 char 类型数组,比如“你好” 就是长度为2的数组 char[] chars = {‘你’,‘好’};但是使用数组过于麻烦,所以就有了 String,String 底层就是一个 char 类型的数组,只是使用的时候开发者不需要直接操作底层数组,用更加简便的方式即可完成对字符串的使用。

4、String有哪些特性

不变性:String 是只读字符串,是一个典型的 immutable 对象,对它进行任何操作,其实都是创建一个新的对象,再把引用指向该对象。不变模式的主要作用在于当一个对象需要被多线程共享并频繁访问时,可以保证数据的一致性。

常量池优化:String 对象创建之后,会在字符串常量池中进行缓存,如果下次创建同样的对象时,会直接返回缓存的引用。

final:使用 final 来定义 String 类,表示 String 类不能被继承,提高了系统的安全性。

5、String为什么是不可变的吗?

简单来说就是String类利用了final修饰的char类型数组存储字符,源码如下图所以:
/** The value is used for character storage. */ private final char value[];

String真的是不可变的吗?

我觉得如果别人问这个问题的话,回答不可变就可以了。 下面只是给大家看两个有代表性的例子:
1 String不可变但不代表引用不可以变:

String str = "Hello";
str = str + " World";
System.out.println("str=" + str);

结果:
str=Hello World

解析:
实际上,原来String的内容是不变的,只是str由原来指向"Hello"的内存地址转为指向"Hello
World"的内存地址而已,也就是说多开辟了一块内存区域给"Hello World"字符串。

通过反射是可以修改所谓的“不可变”对象
// 创建字符串"Hello World", 并赋给引用s
String s = "Hello World";
System.out.println("s = " + s); // Hello World
// 获取String类中的value字段
Field valueFieldOfString = String.class.getDeclaredField("value");
// 改变value属性的访问权限
valueFieldOfString.setAccessible(true);
// 获取s对象上的value属性的值
char[] value = (char[]) valueFieldOfString.get(s);
// 改变value所引用的数组中的第5个字符
value[5] = '_';
System.out.println("s = " + s); // Hello_World

结果:
s = Hello World s = Hello_World

解析:
用反射可以访问私有成员, 然后反射出String对象中的value属性, 进而改变通过获得的value引
用改变数组的结构。但是一般我们不会这么做,这里只是简单提一下有这个东西。

6、是否可以继承 String 类

String 类是 final 类,不可以被继承。

7、String str="i"与 String str=new String(“i”)一样吗?

不一样,因为内存的分配方式不一样。String str="i"的方式,java 虚拟机会将其分配到常量池中;而 String str=new String(“i”) 则会被分到堆内存中。

8、String s = new String(“xyz”);创建了几个字符串对象

两个对象,一个是静态区的"xyz",一个是用new创建在堆上的对象。

例子:

String str1 = "hello"; //str1指向静态区 
String str2 = new String("hello"); //str2指向堆上的对象

String str3 = "hello";
String str4 = new String("hello");
System.out.println(str1.equals(str2));//true 
System.out.println(str2.equals(str4));//true 
System.out.println(str1 == str3); //true
System.out.println(str1 == str2); //false 
System.out.println(str2 == str4); //false
System.out.println(str2 == "hello"); //false 
str2 = str1; System.out.println(str2 == "hello");//true
9、如何将字符串反转?

使用 StringBuilder 或者 stringBuffer 的 reverse() 方法。

示例代码:

// StringBuffer reverse 
StringBuffer stringBuffer = new StringBuffer(); 
stringBuffer.append("abcdefg");
System. out. println(stringBuffer. reverse()); // gfedcba 

// StringBuilder reverse 
StringBuilder stringBuilder = new StringBuilder();
stringBuilder. append("abcdefg");
System. out. println(stringBuilder. reverse()); // gfedcba
10、数组有没有 length()方法?String 有没有 length()方法
  • 数组没有 length()方法 ,有 length 的属性。
  • String 有 length()方法。
  • JavaScript中,获得字符串的长度是通过 length 属性得到的,这一点容易和 Java 混淆。
11、String 类的常用方法都有那些?
  1. indexOf():返回指定字符的索引。
  2. charAt():返回指定索引处的字符。
  3. replace():字符串替换。
  4. trim():去除字符串两端空白。
  5. split():分割字符串,返回一个分割后的字符串数组。
  6. getBytes():返回字符串的 byte 类型数组。
  7. length():返回字符串长度。
  8. toLowerCase():将字符串转成小写字母。
  9. toUpperCase():将字符串转成大写字符。
  10. substring():截取字符串。
  11. equals():字符串比较。
12、在使用 HashMap 的时候,用 String 做 key 有什么好处?

HashMap 内部实现是通过 key 的 hashcode 来确定 value 的存储位置,因为字符串是不可变的,所以当创建字符串时,它的 hashcode 被缓存下来,不需要再次计算,所以相比于其他对象更快。

13、String和StringBuffer、StringBuilder的区别是什么?String为什么是不可变的
可变性

String类中使用字符数组保存字符串,private final char value[],所以string对象是不可变的。

StringBuilder与StringBuffer都继承自AbstractStringBuilder类,在AbstractStringBuilder中也是使用字符数组保存字符串,char[] value,这两种对象都是可变的。

线程安全性

String中的对象是不可变的,也就可以理解为常量,线程安全。AbstractStringBuilder是
StringBuilder与StringBuffer的公共父类,定义了一些字符串的基本操作,如expandCapacity、
append、insert、indexOf等公共方法。

  • StringBuffer对方法加了同步锁或者对调用的方法加了同步锁,所以是线程安全的。
  • StringBuilder并没有对方法进行加同步锁,所以是非线程安全的。
性能

每次对String 类型进行改变的时候,都会生成一个新的String对象,然后将指针指向新的String 对象。

StringBuffer每次都会对StringBuffer对象本身进行操作,而不是生成新的对象并改变对象引用,相同情况下使用StirngBuilder 相比使用StringBuffer 仅能获得10%~15% 左右的性能提升,但却要冒多线程不安全的风险。

对于三者使用的总结
  1. 如果要操作少量的数据用 ==》 String
  2. 单线程操作字符串缓冲区 下操作大量数据 ==》 StringBuilder
  3. 多线程操作字符串缓冲区 下操作大量数据 ==》 StringBuffer

包装类相关

1、自动装箱与拆箱

装箱:将基本类型用它们对应的引用类型包装起来;

拆箱:将包装类型转换为基本数据类型;

2、int 和 Integer 有什么区别

Java 是一个近乎纯洁的面向对象编程语言,但是为了编程的方便还是引入了基本数据类型,但是
为了能够将这些基本数据类型当成对象操作,Java 为每一个基本数据类型都引入了对应的包装类
型(wrapper class),int 的包装类就是 Integer,从 Java 5 开始引入了自动装箱/拆箱机制,使
得二者可以相互转换。

Java 为每个原始类型提供了包装类型:

原始类型: boolean,char,byte,short,int,long,float,double
包装类型:Boolean,Character,Byte,Short,Integer,Long,Float,Double

3、Integer a= 127 与 Integer b = 127相等吗

对于对象引用类型:“= =”比较的是对象的内存地址。

对于基本数据类型:“= =”比较的是值。

如果整型字面量的值在-128到127之间,那么自动装箱时不会new新的Integer对象,而是直接引用常量池中的Integer对象,超过范围 a1==b1的结果是false

例子:

public static void main(String[] args) {
 Integer a = new Integer(3);
 Integer b = 3; // 将3自动装箱成Integer类型
 int c = 3;
System.out.println(a == b); // false 两个引用没有引用同一对象
System.out.println(a == c); // true a自动拆箱成int类型再和c比较
System.out.println(b == c); // true
 Integer a1 = 128;
 Integer b1 = 128;
 System.out.println(a1 == b1); // false
 Integer a2 = 127;
 Integer b2 = 127;
 System.out.println(a2 == b2); // true
}

关注林哥,持续更新哦!!!★,°:.☆( ̄▽ ̄)/$:.°★ 。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值