一、final关键字代表最终、不可改变的。
常见四种用法:
- 可以用来修饰一个类
- 可以用来修饰一个方法
- 还可以用来修饰一个局部变量
- 还可以用来修饰一个成员变量
一旦使用final用来修饰局部变量,那么这个变量就不能进行更改。
“一次赋值,终生不变”
对于基本类型来说,不可变说的是变量当中的数据不可改变
对于引用类型来说,不可变说的是变量当中的地址值不可改变
例子:
package cn.itcast.day11.demo01;
public class Student {
private String name;
public Student() {
}
public Student(String name) {
this.name = name;
}
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
main函数:
package cn.itcast.day11.demo01;
public class Demo01Final {
public static void main(String[] args) {
int num1 = 10;
System.out.println(num1);
num1 = 20;
System.out.println(num1);
//一旦使用final用来修饰局部变量,那么这个变量就不能进行更改
// "一次赋值,终生不变"
final int num2 = 200;
System.out.println(num2);
// num2 = 250 // 错误写法,不能改变
// 正确写法!只要保证唯一一次赋值即可
final int num3;
num3 = 30;
// 对于基本类型来说,不可变指的是变量当中的数据不可变
// 对于引用类型来说,不可变说的是变量当中的地址值不可变
Student stu1 = new Student("赵丽颖");
System.out.println(stu1);
System.out.println(stu1.getName());
stu1 = new Student("霍建华");
System.out.println(stu1);
System.out.println(stu1.getName());
System.out.println("==============");
final Student stu2 = new Student("高圆圆");
// 错误写法!final的引用类型变量,其中的地址不可变
//stu2 = new Student("赵又廷");
System.out.println(stu2.getName());
stu2.setName("高圆圆圆圆");
System.out.println(stu2.getName());
}
}
package cn.itcast.day11.demo01;
public class Demo01Final {
public static void main(String[] args) {
int num1 = 10;
System.out.println(num1);
num1 = 20;
System.out.println(num1);
//一旦使用final用来修饰局部变量,那么这个变量就不能进行更改
// "一次赋值,终生不变"
final int num2 = 200;
System.out.println(num2);
// num2 = 250 // 错误写法,不能改变
// 正确写法!只要保证唯一一次赋值即可
final int num3;
num3 = 30;
// 对于基本类型来说,不可变指的是变量当中的数据不可变
// 对于引用类型来说,不可变说的是变量当中的地址值不可变
Student stu1 = new Student("赵丽颖");
System.out.println(stu1);
System.out.println(stu1.getName());
stu1 = new Student("霍建华");
System.out.println(stu1);
System.out.println(stu1.getName());
System.out.println("==============");
final Student stu2 = new Student("高圆圆");
// 错误写法!final的引用类型变量,其中的地址不可变
//stu2 = new Student("赵又廷");
System.out.println(stu2.getName());
stu2.setName("高圆圆圆圆");
System.out.println(stu2.getName());
}
}
运行结果:
当final关键字用来修饰一个方法的时候,这个方法就是最终方法,也就是不能被覆盖重写。
格式:
修饰符 final 返回值类型 方法名称(参数列表) {
// 方法体
}
注意事项:
对于类、方法来说,abstract关键字和final关键字不能同时使用,因为矛盾。
例子:
父类方法
package cn.itcast.day11.demo01;
public class Student {
private String name;
public Student() {
}
public Student(String name) {
this.name = name;
}
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
子类方法
package cn.itcast.day11.demo01;
public class Zi extends Fu{
@Override
public void methodAbs() {
}
// 错误写法!不能覆盖重写父类当中final的方法
// @Override
// public void method() {
// System.out.println("子类覆盖重写父类的方法!");
// }
}
对于成员变量来说,如果使用final关键字修饰,那么这个变量也照样是不可变。
- 由于成员变量具有默认值,所以用了final之后必须手动赋值,不会再给默认值了。
- 对于final的成员变量,要么使用直接赋值,要么通过构造方法赋值。二者选其一。
- 必须保证类当中所有重载的构造方法,都最终会对final的成员变量进行赋值。
例子:
package cn.itcast.day11.demo01;
public class Person {
private final String name /*= "鹿晗"*/;
public Person(){
name = "关晓彤";
}
public Person(String name){
this.name = name;
}
public String getName(){
return name;
}
}
当final关键字用来修饰一个类的时候,格式:
public final class 类名称 {
// …
}
含义:当前这个类不能有任何的子类。(太监类)
注意:一个类如果是final的,那么其中所有的成员方法都无法进行覆盖重写(因为没儿子。)
例子:
package cn.itcast.day11.demo01;
public final class MyClass {
public void method(){
System.out.println("方法执行!");
}
}
二、Java中有四种权限修饰符:
public > protected > (default) > private
同一个类(我自己) YES YES YES YES
同一个包(我邻居) YES YES YES NO
不同包子类(我儿子) YES YES NO NO
不同包非子类(陌生人) YES NO NO NO
注意事项:(default)并不是关键字“default”,而是根本不写。
不同包的子类或者陌生人,public修饰的成员变量也可以访问
public修饰符的用法:
同一个包中的public成员变量可以相互访问
MyClass类
package cn.itcast.day11.demo02;
public class MyClass {
public int num = 10;
public void methhod(){
System.out.println(num);
}
}
MyAnother类
package cn.itcast.day11.demo02;
public class MyAnother {
public void anotherMethod() {
// System.out.println(new MyClass().num);
}
}
内部类
如果一个事物的内部包含另一个事物,那么这就是一个类内部包含另一个类。
例如:身体和心脏的关系。又如:汽车和发动机的关系。
分类:
- 成员内部类
- 局部内部类(包含匿名内部类)
成员内部类的定义格式:
修饰符 class 外部类名称 {
修饰符 class 内部类名称 {
// …
}
// …
}
注意:内用外,随意访问;外用内,需要内部类对象。
==========================
如何使用成员内部类?有两种方式:
- 间接方式:在外部类的方法当中,使用内部类;然后main只是调用外部类的方法。
- 直接方式,公式:
类名称 对象名 = new 类名称();
【外部类名称.内部类名称 对象名 = new 外部类名称().new 内部类名称();】
例子:
package cn.itcast.day11.demo03;
public class Body { // 外部类
public class Heart{// 成员内部类
// 内部类的方法
public void beat(){
System.out.println("心脏跳动:蹦蹦蹦");
System.out.println("我叫:" + name);
}
}
// 外部类的成员变量
public String name;
// 外部类的方法
public void methodBody(){
System.out.println("外部类的方法");
new Heart().beat();
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
main函数:
package cn.itcast.day11.demo03;
public class Demo01InnerClass {
public static void main(String[] args) {
Body body = new Body();
// 通过外部类的对象,调用外部类的方法,里面间接在使用内部类Heart
body.methodBody();
System.out.println("===========");
// 按照公式写
Body.Heart heart = new Body().new Heart();
heart.beat();
}
}
结果:
如变量出现了重名现象,那么格式是:外部类名称.this.外部类成员变量名
package cn.itcast.day11.demo03;
public class Outer {
int num = 10;// 外部类的成员变量
public class Inner{
int num = 20;
public void methodInner(){
int num = 30;// 内部类方法的局部变量
System.out.println(num); // 局部变量,就近原则
System.out.println(this.num); // 内部类的成员变量
System.out.println(Outer.this.num); // 外部类的成员变量
}
}
}
main函数
package cn.itcast.day11.demo03;
public class Demo02InnerClass {
public static void main(String[] args) {
// 外部类名称.内部类名称 对象名 = new 外部类名称().new 内部类名称();
Outer.Inner obj = new Outer().new Inner();
obj.methodInner();
}
}
结果