【JAVA基础】this关键字与final关键字

2024-08-24 14:12
文章标签 java 基础 关键字 final

本文主要是介绍【JAVA基础】this关键字与final关键字,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • this关键字
  • final关键字
  • Java值传递


this关键字

this代表当前对象,有时候不太方便直接使用对象,可以用this

**用法1:**this.属性名 在类中调用全局变量

举个例子

public class Student1 {public String name;public String sno;public void run(String aaaa){this.name = aaaa;  // =是赋值操作,等号前面是变量,等号后面是值}
}public class Test1 {public static void main(String[] args) {Student1 student = new Student1();student.run("111");Student1 student1 = new Student1();student1.run("222");}
}

用法2:this.方法()

  • static修饰的代码被所有对象共享,所以this可以调用static修饰的方法
  • static修饰的代码当中不能出现关键字,因为类是构建对象的模板,有类不一定有对象

举个例子

    public void run(String aaaa){this.name = aaaa;  // =是赋值操作,等号前面是变量,等号后面是值}/*** static修饰的代码被所有对象共享,所以this可以调用static修饰的方法* static修饰的代码当中不能出现关键字,因为类是构建对象的模板,有类不一定有对象*/
//    public static void run(String aaaa){
//        this.name = aaaa; // 会报错
//    }public void fff(){this.run("333");}

用法3:this()

  • this()不可以在普通方法中使用,只能在构造方法中使用
  • this()在构造方法中使用必须是第一条语句
  • 在一个类下两个构造方法不能通过this()互相调用
  • 不能与super()同时使用

举个例子

    public Student1(){this("aaa");}public Student1(String a){
//        this(); // 不可以两个构造方法同时调用,会报错}

final关键字

用法1:修饰类

  • final修饰的类无法被继承

举个例子:

// final修饰的类无法被继承,如String等类
public final class A {}public class B extends A{ // 报错
}

以上代码在B类想要继承A类时会报错
在这里插入图片描述

用法2:修饰方法

  • final修饰的方法无法被重写
public class A {public final void fly(){System.out.println("我可以飞");}
}public class B extends A{public void fly(){ // 报错System.out.println("我可以飞");}
}

在这里插入图片描述

用法3:修饰变量

final修饰的变量为常量,常量是只能被赋值一次的变量;常量的变量名必须大写

  1. 修饰全局变量:final定义的全局变量必须赋初始值,因为全局变量有默认值,系统会对全局变量进行初始化,为其指定默认值

    引用数据类型的基本值:null

    整形:0

    浮点型:0.0

    布尔型:false

    字符型:“ ”

    //    public final String name; // 报错public final String name = "123"; // final修饰的全局变量需要再定义时赋值// 全局变量是有默认值的
    
  2. 修饰局部变量:系统不会自动为局部变量分配空间与初始化,因此final局部变量可以在后面使用的时候再赋值,但是也只能赋值一次。

    public void fff(){String name;}
    

    注意:当final修饰方法中的形参时,该参数不能在方法体中被赋值,因为系统会在调用该方法时,根据传入的参数来初始化

    public class Demo01 {public void test1(final int a) {//下方语句会报错a = 10;}
    }
    
  3. 修饰基本数据类型定义的变量:对于基本数据类型而言,final固定的是变量的值

        public void run(){a = 11; //报错,final修饰的基本数据类型值无法被改变}
    
  4. 修饰引用数据类型定义的变量:对于引用数据类型而言,final固定的是对象的地址,对象中的内容是可以改变的

    public void change(){arr = brr; // 一旦被final修饰则无法进行赋值操作// 这里的意思是将brr的地址赋值给arr
    }public void change1(){arr[10] = 32;
    }
    

​ 内存图如下:
在这里插入图片描述

Java值传递

例1

public class Demo {public static void main(String[] args) {String nameString = "张三";name(nameString);System.out.println("main--->"+nameString);}public static void name(String name) {name = "李四";System.out.println("name--->"+name);}
}

输出:在这里插入图片描述

  1. 将字符串常量池中的“张三”所在的地址赋给nameString
  2. name()方法入栈,将nameString指向的地址赋值给形参name,此时name也指向字符串常量池中的“张三”
  3. 在字符串常量池中添加“李四”,并将其地址赋值给变量name,并输出name—>李四
  4. 输出main—>张三

在这里插入图片描述

例2

public class Person {private String nameString;private int age;public Person(String nameString,int age){this.nameString = nameString;this.age = age;}@Overridepublic String toString(){return "Person [nameString" + nameString + ",age=" + age + "]";}public static void name(Person person){person.age = 20;person.nameString = "李四";person = null;}public static void main(String[] args) {Person person = new Person("张三",18);System.out.println(person.toString());name(person);System.out.println(person.toString());}
}

输出:在这里插入图片描述

  1. main()方法入栈,创建对象person,并将字符串常量池中“张三”的地址赋给nameString,18赋值给age,
  2. 打印person对象
  3. name()方法入栈,将person对象传入name()方法,使name()中的形参person对象也指向地址0x1,此时形参对象person可以调用0x1中的变量,将age修改为20,将nameString指向字符串常量池中“李四”所在的地址,最后让将形参person不再指向地址0x1,name()方法出栈
  4. 打印person对象

在这里插入图片描述

例3

public class Student1 {private String name;private int age;public Student1(String name,int age){this.name = name;this.age = age;}@Overridepublic String toString(){return "Student [name=" + name + ", age=" + age +"]";}public static void change(Student1 s1, Student1 s2){Student1 temp = new Student1("王五",18);temp = s1;s1 = s2;s2 = temp;}public static void main(String[] args) {Student1 zhangsan = new Student1("张三",18);Student1 lisi = new Student1("李四",20);Student1.change(zhangsan,lisi);System.out.println(zhangsan.toString());System.out.println(lisi.toString());}
}

输出:在这里插入图片描述

  1. 创建Student1类对象“zhangsan”、“lisi”,并开辟内存空间,将两个对象中的name变量分别指向字符串常量池中的“张三”、“李四”,给age分别赋值
  2. 通过Student1()类调用change方法,change()方法入栈,将对象“zhangsan”、“lisi”作为形参s1、s2的值传入,使s1指向地址0x1,s2指向地址0x2
  3. 创建Student1类对象temp,并开辟内存空间0x3,将其中的name指向字符串常量池中的“王五”。给age赋值
  4. 将s1地址值赋给temp,此时temp指向0x1
  5. 将s2地址赋值给s1,此时s1指向0x2
  6. 将temp地址值赋给s2,此时s2指向0x1,执行结束之后change()方法出栈
  7. 打印“zhangsan”、“lisi”

==注意:==这道题中交换的只是s1、s2、temp的地址值,而不是对象中的变量值作为交换,所以并不影响外部的对象“zhangsan”、“lisi”依旧指向自己原本的地址

在这里插入图片描述

例4

class Two {byte x;
}
public class Student2{public static void main(String[] args) {Student2 student = new Student2();student.start();}void start(){Two two = new Two();System.out.println(two.x+" ");Two two2 = fix(two);System.out.println(two.x+" "+ two2.x);}Two fix(Two tt){tt.x = 42;return tt;}
}

输出:在这里插入图片描述

  1. main()方法入栈创建Student2类对象student,并开辟内存空间,调用start()方法
  2. start()方法入栈,创建Two类型对象two,初始化全局变量x = 0,打印two.x值为0
  3. fix()方法入栈,将对象two传给形参tt,使tt指向内存0x2,修改其中x的值,此时x=42,返回tt的值给对象two2,fix()方法出栈
  4. 打印此时的two.x和two2.x因为两个对象指向的是同一片内存地址,所以输出结果一样,都是42

在这里插入图片描述

这篇关于【JAVA基础】this关键字与final关键字的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1102707

相关文章

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

Java中HashMap的用法详细介绍

《Java中HashMap的用法详细介绍》JavaHashMap是一种高效的数据结构,用于存储键值对,它是基于哈希表实现的,提供快速的插入、删除和查找操作,:本文主要介绍Java中HashMap... 目录一.HashMap1.基本概念2.底层数据结构:3.HashCode和equals方法为什么重写Has

Java 正则表达式的使用实战案例

《Java正则表达式的使用实战案例》本文详细介绍了Java正则表达式的使用方法,涵盖语法细节、核心类方法、高级特性及实战案例,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录一、正则表达式语法详解1. 基础字符匹配2. 字符类([]定义)3. 量词(控制匹配次数)4. 边

Java Scanner类解析与实战教程

《JavaScanner类解析与实战教程》JavaScanner类(java.util包)是文本输入解析工具,支持基本类型和字符串读取,基于Readable接口与正则分隔符实现,适用于控制台、文件输... 目录一、核心设计与工作原理1.底层依赖2.解析机制A.核心逻辑基于分隔符(delimiter)和模式匹

Java中的stream流分组示例详解

《Java中的stream流分组示例详解》Java8StreamAPI以函数式风格处理集合数据,支持分组、统计等操作,可按单/多字段分组,使用String、Map.Entry或Java16record... 目录什么是stream流1、根据某个字段分组2、按多个字段分组(组合分组)1、方法一:使用 Stri

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装

使用Spring Cache本地缓存示例代码

《使用SpringCache本地缓存示例代码》缓存是提高应用程序性能的重要手段,通过将频繁访问的数据存储在内存中,可以减少数据库访问次数,从而加速数据读取,:本文主要介绍使用SpringCac... 目录一、Spring Cache简介核心特点:二、基础配置1. 添加依赖2. 启用缓存3. 缓存配置方案方案

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

深度剖析SpringBoot日志性能提升的原因与解决

《深度剖析SpringBoot日志性能提升的原因与解决》日志记录本该是辅助工具,却为何成了性能瓶颈,SpringBoot如何用代码彻底破解日志导致的高延迟问题,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言第一章:日志性能陷阱的底层原理1.1 日志级别的“双刃剑”效应1.2 同步日志的“吞吐量杀手”

Spring创建Bean的八种主要方式详解

《Spring创建Bean的八种主要方式详解》Spring(尤其是SpringBoot)提供了多种方式来让容器创建和管理Bean,@Component、@Configuration+@Bean、@En... 目录引言一、Spring 创建 Bean 的 8 种主要方式1. @Component 及其衍生注解