java中为什么用final修饰了引用类型,但还是能修改对象,请大神解答下哈

class Person{
String name;
int age;
Person(){}
Person(String name,int age){
this.name=name;
this.age=age;
}
public void ShowInfo(){
System.out.println("姓名:"+this.name+"年龄:"+this.age);
}
}
public class finala {
//建立的同时进行初始化
final int NUM;
//定义的同时对常亮进行初始化
final int sd=120;
//修饰一个对象引用,将不能改变对象本身,但可以改变对象的值
final Person p=new Person("无名氏",20);
finala(){
this.NUM=100;
}
//可以用final来修饰形参,本例中形参是一个对象的引用
public void change(final Person p){
p.name="职工那样";
//对象不能被改变
//p=new Person();

    }
    public static void main(String[] args){
        finala f=new finala();
        //比如这个就可以,他只是改变了对象的值,因此是合法的
        f.p.name="张三丰";
        f.p.ShowInfo();
        //比如这个就是错误的,因为不能指向一个新的对象
        //f.p=new Person();,但是下面这个地方还是能创建一个指向新对象还是P,但却不报错,请指点下哈
        Person p=new Person("张飞",30);
        f.change(p);
        p.ShowInfo();
    }

}

前面两个说的都是正确的,第三个Person p = new person() ,f,change(p), 这里,其实并不是让f的p指向一个新对象,
change里面只是拿到那个新p的值,所以,change只是改变值,并没有指向新对象,所以是对的

final 修饰的是对象的引用,即对象地址吧。修改对象只要不修改对象地址即不改变Person p的指向,但是可以修改p指向的对象的内部属性。

final 修饰的基本类型是如你所理解的,不可更改。但当final修饰引用类型时,是规定改引用类型变量不可更改,引用类型变量所指向的值是不受影响的。
或者可以理解为,final修饰的是存储在栈中的对象时,不可变的。