面试知识点总结

final的四个作用

1.final修饰的常量不能被第二次赋值,并且final修饰的常量要用大写字符表示

2.final修饰的类不能被继承

3.final修饰的方法不能被覆盖,但是可以重载
重载是可以在方法名相同的情况下,参数不一样
覆盖是在方法名相同的情况下,参数一样

4.final防止指令重排序,保证多线程下的并发安全

5.final不用来修饰复杂数据,例如数组类型数据,在被final修饰的时候,其内部的数据不会起到被修饰的作用,只能用来修饰基本类型和不可变类型

String,StringBuffer,StringBuilder的区别(面试重点)

StringBuilder的拼接速度远远快于String

StringBuffer和StringBuilder的内部机制是一样的,但是StringBuffer内部增加了一个锁,起到了保障线程安全的作用,但是牺牲了微量的时间造成会比StringBuilder的速度慢一些

在内存中,如果存储空间足够的话会尽可能的讲数据挨着存储,这种存储机制就形成了操作系统中的逻辑页存储的原理
在内存中的物理存储单元都是以1字节为单位的,这样在读取数据的时候会浪费大量的时间,所以操作系统声明了一种逻辑存储单元的机制,这种逻辑存储单元的机制是将4096个字节(4KB)的内容作为一个单位,这样在cpu调用大内存的数据时,可以减少大量的调用次数,我们的8G内存是有2百万个4096字节的逻辑单元,而且操作系统为每一个逻辑单元设置了一个逻辑地址

逻辑单元的地址:整个逻辑单元的地址是第一个实际物理单元的地址

操作系统在为逻辑单元分配地址的时候,是以变量为单位进行分配的,所以数组的优势就体现出来了,数组内可以包含多个同类型的数据,这样的话操作系统就会为其分配一个页来进行存储

我们的字符串的底层是一个字符数组,这样的情况下我们的内存是给每一个字符串变量分配一个页的,分配页的同时又默认的给字符数组按照字符串实际长度进行定义的,这样的话每一个字符串变量都会有一个固定长度的字符数组,又因为如果我们要改变固定长度的数组的长度时需要重新分配页,所以在使用String类型的变量的字符串拼接时就会占用大量的内存空间

StringBuilder和StringBuffer则是为一个字符串变量分配一个足够大的数组长度,例如分配我们的内存中的逻辑页的大小4KB,甚至可以更大,这样的话在后续的字符串拼接的时候就不用像String那样的形式重新占用新的页,这样的话就可以大大的节省内存空间和计算机cpu的调用效率

Buffer字段的数据类型的内部必定有一个富裕的空间的数组作为缓冲,在增加数据的时候能够避免消耗新内存

char

char类型的位数根据编码决定,如果是ASCII就是8位,Unicode就是16位,UTF-8(变长字符集)就是16到32位,UTF-32就是32位

switch

switch的参数可以选择的类型为,char,byte,short,int,enum,String,而且在参数选择完类型之后,case后面的数据类型要和参数类型一致

switch击穿,如果我们在某一个case语句结束的时候没有加上default的话,就会继续向下执行下一个case,这样的话就会形成switch击穿的情况

static

static修饰的变量 面试重点
放到静态常量池中,我们所修饰的变量是属于类的,也就是被所有对象公用,都可以对其进行修改
static修饰的方法
属于类方法,可以调用类方法,但是不能调用非类方法,非类方法可以调用类方法
底层逻辑
类方法是和类一体的,但是非类方法是需要先定义了对象才能存在的,所以我们不能在不确定对象是否声明的情况下就直接使用类方法调用非类方法
this关键字不能在static修饰的方法中存在
底层逻辑
因为this代表的是当前对象,而我们的静态方法存在的时候,我们的对象不一定是存在的
static修饰的代码块
如果存在static修饰的代码块的话,我们的程序会由上到下的先执行静态代码块,然后再执行构造方法,再执行普通代码块
底层逻辑

我们的程序在执行之前是需要先加载的,在加载的时候就可以把静态常量池的数据进行显示,之后我们在执行的时候需要创建对象,对象的构造方法这个时候就会被执行,然后再执行普通代码块

访问器和修改器

面试重点
通过公有的方法对私有变量进行访问和修改,在多线程下可以保障线程安全
但是如果对访问器和修改器不做任何代码处理的情况下,我们的线程是不能保障安全的,例如以下代码
public class Person
{
private int age;
public String name;
public int height;
private int[] arr = {1,2,3,4,51};
public int[] getArr({
return arr;
}
}
public class Test{
public static void main(String[] aaaa){
Person a1 = new Person();
int]b = a1.getArr();
b[0] = 100;
int[c = a1.getArr();
System.out.printIn( c[0] );
}
这种访问器就会修改原数据的值,所以我们需要在访问器内部写入深拷贝的代码,这样可以避免这种情况产生
public int[] getArr(){
int[] arrnew = new int[ arr.length ];
for(int i = o; i< arr.length; i++){
arrnew[i] = arr[i];
return arrnew;
}
}
我们需要在访问器内部写入深拷贝的代码,让原数据的地址隐藏起来,这样才能保证线程安全

深拷贝,浅拷贝

引用类型的=,是指将两个数据指向了同一个地址
浅拷贝:引用数据类型的=是浅拷贝,是我们引用类型的数据=之后,两个数据指向了同一个地址,这个时候改变其中一个数据都会对地址中的数据进行改变
深拷贝:数组元素使用new()方法进行初始化,再对元素进行逐一的赋值,这样的话在改变其中的一个数组元素时,不会对另一个数组有影响

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值