Java Review (十二、面向对象----final 修饰符)

本文详细解释了Java中final关键字如何影响局部变量的默认值、赋值限制,以及final修饰的方法、类和引用类型的特性。讨论了不可变类的设计原则和final方法的性能优化作用。
  • 如果 final 修饰的局部变量在定义时没有指定默认值,则可以在后面代码中对该 final 变量赋初始值,但只能一次,不能重复赋值 ;

  • 如果 final 修饰的局部变量在定义时己经指定默认值,则后面代码中不能再对该变量赋值 。

final修饰局部变量、形参实例

pub1ic c1ass Fina1Loca1Variab1eTest{

pub1ic void test( final int a){

// 不能对 fina1 修饰的形参赋值,下面语句非法

// a = 5;

}

pub1ic static void main(String[] args){

//定义 fina1 局部变量时指定默认值,则 str 变量无法重新赋值

final String str = “hello”;

//下面赋值语句非法

// str = " Java";

// 定义 fina1 局部变量时没有指定默认值,则 d 变量可被赋值一 次

final double d;

// 第一次赋初始值,成功

d = 5.6;

// 对 fina1 变量重复赋值 , 下面语句非法

// d = 3.4;

}

}

final 修饰基本类型变量和引用类型变量的区别


  • 当使用 final 修饰基本类型变量时,不能对基本类型变量重新赋值,因此基本类型变量不能被改变 。

  • 但对于引用类型变量而言 ,它保存的仅仅是一个引用, final 只保证这个引用类型变量所引用的地址不会改变,即 一直引用同一个对象,但这个对象完全可以发生改变 。

final 修饰基本类型变量和引用类型变量的区别实例

c1ass Person{

private int age;

pub1ic Person() {

}

// 有参数的构造器

pub1ic Person(int age){

this.age = age;

//省略 age 的 setter 和 getter 方法

//age 的 setter 和 getter 方法

}

}

pub1ic c1ass Fina1ReferenceTest{

pub1ic static void main(String[] args){

//fina1 修饰数组变量, iArr 是一个引用变量

fina1 int[] iArr = (5 , 6, 12, 9) ;

System.out.print1n(Arrays.toString(iArr));

//对数组元素进行排序,合法

Arrays.sort(iArr);

System.out.println(Arrays.toString(iArr));

//对数组元素赋值,合法

iArr[2] = - 8 ;

System.out.println(Arrays.toString (iArr) );

// 下面语句对 iArr 重新赋值,非法

// iArr = null;

// final 修饰 Person 变量 , p 是一个引用变量

final Persoηp =new Person(45) ;

// 改变 Person 对象的 age 实例变量 ,合法

p.setAge(23) ;

System.out.println(p.getAge()) ;

//下面语句对 p 重新赋值 ,非法

// p = null;

}

}

final方法

=========================================================================

之所以要使用final 方法,可能是出于对两方面理由的考虑。

  • 第一个是为方法“上锁”,防止任何子类改变它的本来含义。用final修饰的方法的行为在继承期间保持不变,而且不可被重写。

  • 采用final 方法的第二个理由是程序执行的效率——将一个方法设成 final 后,编译器就可以把对那个方法的所有调用都置入“嵌入”调用里。只要编译器发现一个final 方法调用,就会(编译器判断)忽略为执行方法调用机制而采取的常规代码插入方法(将变量压入堆栈;跳至方法代码并执行它;跳回来; 清除堆栈变量; 最后对返回值进行处理)。相反,它会用方法主体内实际代码的一个副本来替换方法调用。这样做可避免方法调用时的系统开销。

public class Test{

public final void changeName(){

// 方法体

}

}

对于一个 private 方法 , 因为它仅在当前类中可见, 其子类无法访问该方法 , 所以子类无法重写该方法一一如果子类中定义一个与父类 private 方法有相同方法名、相同形参列表、相同返回值类型 的方法 ,也不是方法重写,只是重新定义了 一个新方法。因此, 即使使用 final 修饰一个 private 访问权限的方法,依然可 以在其子类中定义与该方法具有相同方法名 、 相同形参列表、相同返回值类型的方法。

public class PrivateFinalMethodTest{

private final void test(){}

}

class Sub extends PrivateFinalMethodTest{

//下面的方法定义不会出现 问题

public void test(){}

}

final 修饰的方法不能被重写 , 但是可以被重载 。

final类

========================================================================

final 修饰的类不可被继承。

public final class Test {

// 类体

}

不可变类


不可变( immutable ) 类的意思是创建该类的实例后,该实例 的实例变量是不可改变的。 Java 提供的 8 个包装类和 java.lang.String 类都是不可变类 , 当创建它们的实例后 , 其实例的实例变量不可改变。

Double d = new Double(6.5) ;

String str =new String( “Hello”);

如果需要创建自定义的不可变类,可遵守如下规则 。

  • 使用 private 和 final 修饰符来修饰该类的成员变量。

  • 提供带参数构造器,用于根据传入参数来初始化类里的成员变量 。

  • 仅为该类的成员变量提供 getter 方法,不要为该类的成员变量提供 setter 方法 ,因为普通方法无法修改 final 修饰的成员变量。

  • 如果有必要,重写 Object 类的 hashCode()和 equals()方法, equals()方法根据关键成员变量来作为两个对象是否相等的标准,除此之外,还应该保证两个用 equals()方法判断为相等的对象的 hashCode()也相等。

定义一个不可变的 Address 类,程序把 Address 类的 detail 和 postCode 成员变量都使用 private隐藏起来,并使用 final 修饰这两个成员变量 , 不允许其他方法修改这两个成员变量的值。

public class Address{

private final String detail ;

private final String postCode ;

// 在构造器里初始化两个实例变量

public Address(){

this .detail = “”;

this.postCode = “”;

}

pub1ic Address(String detai1 , String postCode){

this.deta = deta ;

this.postCode = postCode ;

}

//仅为两个实例变量提供 getter 方法

pub1ic String getDetai1(){

return this.detai1 ;

}

pub1ic String getPostCode(){

return this.postCode;

}

//重写 equa1s ()方法,判断两个对象是否相等

pub1ic boo1ean equa1s(Object obj){

if (this == obj ){

return true ;

}

if(obj != nu11 && obj.getC1ass() == Address.c1ass{

Address ad = (Address)obj;

//当 detai1 和 postCode 相等时 , 可认为两个 Address 对象相等

if(this . getDeta 工 1() .equa1s (ad . getDetai1())&& this.getPostCode (} . equa1s(ad.getPostCode())){

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

最后

这份《“java高分面试指南”-25分类227页1000+题50w+字解析》同样可分享给有需要的朋友,感兴趣的伙伴们可挑战一下自我,在不看答案解析的情况,测试测试自己的解题水平,这样也能达到事半功倍的效果!(好东西要大家一起看才香)

image

image

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!
g.cn/images/e5c14a7895254671a72faed303032d36.jpg" alt=“img” style=“zoom: 33%;” />

最后

这份《“java高分面试指南”-25分类227页1000+题50w+字解析》同样可分享给有需要的朋友,感兴趣的伙伴们可挑战一下自我,在不看答案解析的情况,测试测试自己的解题水平,这样也能达到事半功倍的效果!(好东西要大家一起看才香)

[外链图片转存中…(img-hZ0J5hW5-1712108351829)]

[外链图片转存中…(img-fA02ZoX4-1712108351829)]

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值