Java面向对象进阶5——包和final(含源码阅读)

简介: 包在操作系统中其实就是一个文件夹。包是用来分门别类的管理技术,不同的技术类放在不同的包下,方便管理和维护

一、前言


从本文开始,将会讲述Java面向对象中的一些小知识点,本文主要是讲述有关包和final的知识


二、包


1️⃣定义


包在操作系统中其实就是一个文件夹。包是用来分门别类的管理技术,不同的技术类放在不同的包下,方便管理和维护


2️⃣命名


  • 包名一般是公司域名倒写 + 包的作用
  • 全部英语小写
  • 路径名.路径名.xxx.xxx
  • 包名要用 “.” 连接
  • 包名的每个路径名必须是一个合法的标识符,而且不能是Java的关键字


3️⃣使用其他类


🟡使用同一个包中的类时,不需要导包

b4eb4aed7d0f463c85792fe65cd73fea.png

由于Test类和Teacher类同在一个polymorphism包中,使用在调用Teacher类时并不会报错,也不需要导包


🟡使用java.lang包中类时,不需要导包


37576bd0d80347bda39c507a40da26d6.png


当我们选中String后再使用 Ctrl+B 跟进,查看源码,发现String类是在 java.lang 包下


acd22e07b37c42849bcd6523299229e0.png


🟡如果同时使用两个包中同类名,需要用全类名


当不同的包中有相同的类名时,我们需要看清楚要调用的类名是在哪个包中的,并且在导入别的包中的类名时,需要用全类名


99a42a3a6f754fd1b3973821d43c19af.png

1ee121ddb7774e0493b52892efb19982.png


final关键字


1️⃣定义


不可改变,最终的含义,可以用于修饰类、方法和变量。

类:被修饰的类,不能被继承

方法:被修饰的方法,不能被重写

变量:被修饰的变量,有且仅能被赋值一次(常量)


2️⃣final修饰类


被修饰的方法,不能被继承


🟡代码测试


final class  Fu{
        public  void  show(){
            System.out.println("这是父类");
        }
    }
    class Zi extends Fu {
        @Override
        public void show(){
            System.out.println("这是子类");
        }
    }


上述代码在IDEA中会爆红


4ba2d90ea07d4038a5aab3f1487be8c1.png


🟡源码阅读


选中String类后使用 Ctrl+B 跟进,查看源码


76ede7851e614473927d18af9d05402f.png


此处的String类是被final修饰的,即不可被继承,所以String类不能被任何一个类当作是父类


3️⃣final修饰方法


被修饰的方法,不能被重写

🟡代码测试


class  Fu{
        public final void  show(){
            System.out.println("这是父类");
        }
    }
    class Zi extends Fu {
        @Override
        public void show(){
            System.out.println("这是子类");
        }
    }


上述代码在IDEA中会爆红,所以在编写代码时一定要注意该问题


c5ccc81135c14a2eaead24071fbb88e1.png


🟡源码阅读


Ctrl+N 查看类,搜索object类,并选择Java.lang


b85148f4d70a495b8f530f1f63e808c0.png

99866f1fd3784f24a533a07a0ebcde3b.png


native


这个方法体是调用本地其他语言(C语言、汇编语言等)来写的,在这里是看不到它的方法体


get class()


获取编译以后的字节码文件


方法的作用


获取到字节码文件的对象,并与本地的操作系统产生交互


可以看到这个类是被final类修饰的,即该方法不可改变


如果再往下阅读源码,会发现还有public final native void notify() 、public final native void notifyAll() 和 public final native void wait(long timeout) throws InterruptedException 三个被final修饰的方法,同样是不想让方法被改变


4️⃣final修饰基本数据类型


被修饰的变量,有且仅能被赋值一次(变为常量)


🟡代码实现


public static void main(String[] args) {
        final int a = 20;
        a = 30;
        System.out.println(a);
    }


此时代码会爆红


e32bd52f32ef4454b2a5ad11ceccf175.png


🟡源码阅读


Ctrl+N 查看类,搜索Math


70f3acc1441d4d3596345df4aeedbc60.png

dffd79c9067841a3a73c8e5ff370e591.png


可以看到,在Math类里面定义了两个常量 PIE


🟡常量


在实际开发中,常量一般作为系统的配置信息,方便维护,提高可读性


命名规范


单个单词:全部大写

多个单词:全部大写,单词间用下划线展开


5️⃣ final修饰引用数据类型


变量存储地址值不变,对象内部属性值可改变


🟡代码实现


public static void main(String[] args) {
        final int[] arr = {1,2,3,4,5};
        arr[0] = 0;
        arr[1] = 10;
        for (int i = 0; i < arr.length; i++) {
            System.out.print(arr[i] + " ");
        }
    }


输出结果:0 10 3 4 5


6ea94e99b2f545319c9c7f73de656b78.png


上述代码改变的就是对象内部属性值,所以是可行的,但是如果我们改变一下存储地址值(类型改变)的话会怎么样


public static void main(String[] args) {
        final int[] arr = {1,2,3,4,5};
        arr = new int[20];
        arr[0] = 0;
        arr[1] = 10;
        for (int i = 0; i < arr.length; i++) {
            System.out.print(arr[i] + " ");
        }

791f4a8942204d36b6d8668436cf51a3.png


代码报错,原因是这样会改变其地址值


🟡源码阅读


Ctrl+N 查看类,搜索String类,并选择 Java.long


11c7c38cda504be2bc495ca90d481148.png

3c2cdb2c7fcc4c0ebf121021174d3940.png


可以看到底层是有一个数组来存储字符串的内容,因为是用final来修饰的数组,所以其地址值不能改变

byte

存储数据类型

value[]

用来存储字符串的数组

private

私有,即外界无法获取到 value 记录的地址值


我们可以在这个源码中通过 Ctrl+F12 来搜索一下,发现没有value的get和set方法,使得我们没办法从外界获取其记录的地址值


47c90e20a2dd47ea9afd3217c956f66f.png


总结一下,上面这些都是字符串无法改变的原因


四、结语


通过阅读源码,我们可以更加深刻的了解到其原理所在,在之后的文章中,我也会多加分析源码,尽可能的将原理讲清楚,如果有任何疑问以及不足之处,欢迎指正

相关文章
|
3月前
|
Java
Java基础语法与面向对象
重载(Overload)指同一类中方法名相同、参数列表不同,与返回值无关;重写(Override)指子类重新实现父类方法,方法名和参数列表必须相同,返回类型兼容。重载发生在同类,重写发生在继承关系中。
150 1
|
3月前
|
存储 Java 关系型数据库
Java 项目实战基于面向对象思想的汽车租赁系统开发实例 汽车租赁系统 Java 面向对象项目实战
本文介绍基于Java面向对象编程的汽车租赁系统技术方案与应用实例,涵盖系统功能需求分析、类设计、数据库设计及具体代码实现,帮助开发者掌握Java在实际项目中的应用。
127 0
|
4月前
|
安全 Java 编译器
Java面向对象
本文深入讲解了Java面向对象编程(OOP)的四大特性:封装、继承、多态与抽象,以及方法的设计与使用。通过示例展示了如何用类和对象组织代码,提升程序的可维护性与扩展性。
|
5月前
|
存储 Java 测试技术
Java基础 - 面向对象
面向对象编程是Java的核心,包含封装、继承、多态三大特性。封装隐藏实现细节,提升代码可维护性与安全性;继承实现类间IS-A关系,支持代码复用;多态通过继承、重写与向上转型,实现运行时方法动态绑定,提升系统扩展性与灵活性。
|
7月前
|
人工智能 Java 编译器
Java:面向对象
本文介绍了Java编程中的核心概念,包括包的命名规范与自动导入机制、构造方法的特点与使用、`this`和`super`关键字的作用、继承的基本规则、访问权限的设置、封装的意义、多态的实现原理以及`static`关键字的用法。通过详细解析每个知识点,并结合代码示例,帮助读者深入理解Java面向对象编程的核心思想与实践技巧。内容适合初学者及进阶开发者学习参考。
170 0
|
Java
Java并发编程笔记之FutureTask源码分析
FutureTask可用于异步获取执行结果或取消执行任务的场景。通过传入Runnable或者Callable的任务给FutureTask,直接调用其run方法或者放入线程池执行,之后可以在外部通过FutureTask的get方法异步获取执行结果,因此,FutureTask非常适合用于耗时的计算,主线程可以在完成自己的任务后,再去获取结果。
4393 0
|
Java 调度 API
Java并发编程笔记之Timer源码分析
timer在JDK里面,是很早的一个API了。具有延时的,并具有周期性的任务,在newScheduledThreadPool出来之前我们一般会用Timer和TimerTask来做,但是Timer存在一些缺陷,为什么这么说呢?   Timer只创建唯一的线程来执行所有Timer任务。
3158 0
|
Java
Java并发编程笔记之Semaphore信号量源码分析
JUC 中 Semaphore 的使用与原理分析,Semaphore 也是 Java 中的一个同步器,与 CountDownLatch 和 CycleBarrier 不同在于它内部的计数器是递增的,那么,Semaphore 的内部实现是怎样的呢?   Semaphore 信号量也是Java 中一个同步容器,与CountDownLatch 和 CyclicBarrier 不同之处在于它内部的计数器是递增的。
4369 0
|
Java
Java并发编程笔记之CyclicBarrier源码分析
JUC 中 回环屏障 CyclicBarrier 的使用与分析,它也可以实现像 CountDownLatch 一样让一组线程全部到达一个状态后再全部同时执行,但是 CyclicBarrier 可以被复用。
2335 0
|
Java
Java并发编程笔记之 CountDownLatch闭锁的源码分析
JUC 中倒数计数器 CountDownLatch 的使用与原理分析,当需要等待多个线程执行完毕后在做一件事情时候 CountDownLatch 是比调用线程的 join 方法更好的选择,CountDownLatch 与 线程的 join 方法区别是什么? 日常开发中经常会遇到需要在主线程中开启多线程去并行执行任务,并且主线程需要等待所有子线程执行完毕后再进行汇总的场景,它的内部提供了一个计数器,在构造闭锁时必须指定计数器的初始值,且计数器的初始值必须大于0。
6499 0