引用传递与基本应用
一. 引用传递
所谓的引用传递就是指将堆内存的使用权交给多个栈内存空间。
1. 引用实例一
【对象引用传递】

class Demo
{
    int temp=30;
}
public class RefDemo01
{
    public static void main(String[] args){
        Demo d1=new Demo();
        d1.temp=50;
        System.out.println("fun()方法调用之前:"+d1.temp);
        fun(d1);
        System.out.println("fun()方法调用之后:"+d1.temp);
    }
    public static void fun(Demo d2){
        d2.temp=1000;
    }
}

运行结果为:

fun()方法调用之前:50
fun()方法调用之后:1000

从运行结果可知,在fun方法中接收了Demo类对象d1,并将temp属性的内容进行修改,因为是引用传递,所以最终temp的值是1000,此程序可以通过下图表示:

Java传递int类型引用 java引用传递的例子_栈内存


Java传递int类型引用 java引用传递的例子_内存_02


2. 引用实例二

public class RefDemo02
{
    public static void main(String[] args){
        String str1="hello";
        System.out.println("fun方法调用之前"+str1);
        fun(str1);
        System.out.println("fun方法");
    }
    public static void fun(String str2){
        str2="ZZU";
    }
}

程序运行结果为:

fun方法调用之前:hello
fun方法调用之后:hello

从程序的运行结果发现,虽然此时传递的是一个string类型的对象,但是结果并没有和之前一样发生变化,因为字符串的内容一旦声明是不可改变的,改变的只是其内存的地址的指向
3. 引用实例三

class Demo{
    String temp="hello";

}
public class RefDemo03
{
    public static void main(String[] args){
        Demo d1=new Demo();
        d1.temp="world";
        System.out.println("fun()方法调用之前:"+d1.temp);
        fun(d1);
        System.out.println("fun()方法调用之后:"+d1.temp);
    }
    public static void fun(Demo d2){
        d2.temp="ZZU";
    }
}

程序运行结果:

fun()方法调用之前:world
fun()方法调用之后:ZZU

本程序运行之后,发现fun()方法中将属性的内容修改了,本程序与第一个范例相同,因为String是作为Demo类的属性存在的,而在操作时更改的只是Demo类中的属性内容。
二. 接受本类的引用
以上为引用传递的基本形式,实际上在对象引用传递上也存在一个类中接收自己本类对象的实例,而且接收完之后,可以方便地通过此对象直接进行本类中封装属性的访问。
【接收自己的本类】

class Demo
{
    private int temp=30;
    public void fun(Demo d2){
        d2.temp=50;
    }
    public int getTemp(){
        return temp;
    }
    public void setTemp(int t){
        temp=t;
    }
}
public class RefDemo04
{
    public static void main(String[] args){
        Demo d1=new Demo();
        d1.setTemp(50);
        d1.fun(d1);
        System.out.println("temp="+d1.getTemp());
    }
}

运行结果:

temp=50

此种方式的传递在关于对象比较操作时才会使用,其他时候基本上很少使用。
三. 范例-——一对一的关系
一个人有一本书,一本书有一本书。人应该是一个具体的类,书也应该是一个具体的类,在人类中应该存在一个属性表示书,在书的类中应该也有一个属性表示人。

class Person{           // 定义Person类
    private String name ;   // 姓名
    private int age ;       // 年龄
    private Book book ;     // 一个人有一本书
    public Person(String name,int age){
        this.setName(name) ;
        this.setAge(age) ;
    }
    public void setName(String n){
        name = n ;
    }
    public void setAge(int a){
        age = a ;
    }
    public String getName(){
        return name ;
    }
    public int getAge(){
        return age ;
    }
    public void setBook(Book b){
        book = b ;
    }
    public Book getBook(){
        return book ;
    }
};
class Book{             // 定义Book类
    private String title ;  // 标题
    private float price ;   // 价格
    private Person person ; // 一本书属于一个人
    public Book(String title,float price){
        this.setTitle(title) ;
        this.setPrice(price) ;
    }
    public void setTitle(String t){
        title = t ;
    }
    public void setPrice(float p){
        price = p ;
    }
    public String getTitle(){
        return title ;
    }
    public float getPrice(){
        return price ;
    }
    public void setPerson(Person p){
        person = p ;
    }
    public Person getPerson(){
        return person ;
    }
};
public class RefDemo05{
    public static void main(String arg[]){
        Person per = new Person("张三",30) ;
        Book bk = new Book("JAVA SE核心开发",90.0f) ;
        per.setBook(bk) ;       // 设置两个对象间的关系,一个人有一本书
        bk.setPerson(per) ;     // 设置两个对象间的关系,一本书属于一个人
        System.out.println("从人找到书 --> 姓名:" + per.getName()+";年龄:" 
            + per.getAge() +";书名:" + per.getBook().getTitle() + ";价格:" 
            + per.getBook().getPrice()) ;   // 可以通过人找到书
        System.out.println("从书找到人 --> 书名:" + bk.getTitle() + ";价格:"
            + bk.getPrice() + ";姓名:" + bk.getPerson().getName() + ";年龄:"
            + bk.getPerson().getAge()) ;    // 也可以通过书找到其所有人
    }
};

运行结果:

从人找到书 --> 姓名:张三;年龄:30;书名:JAVA SE核心开发;价格:90.0
从书找到人 --> 书名:JAVA SE核心开发;价格:90.0;姓名:张三;年龄:30

以上代码完成了一个基本关系,但是当人有一个孩子,每个孩子有一本书,怎么办?这时不许要新建一个孩子类,因为孩子也是一个人,所以此时,只需修改Person类,在类中增加一个自己的引用即可:

class Person{           // 定义Person类
    private String name ;   // 姓名
    private int age ;       // 年龄
    private Book book ;     // 一个人有一本书
    private Person child ;  // 一个人有一个孩子
    public Person(String name,int age){
        this.setName(name) ;
        this.setAge(age) ;
    }
    public void setName(String n){
        name = n ;
    }
    public void setAge(int a){
        age = a ;
    }
    public String getName(){
        return name ;
    }
    public int getAge(){
        return age ;
    }
    public void setBook(Book b){
        book = b ;
    }
    public Book getBook(){
        return book ;
    }
    public void setChild(Person c){
        child = c ;
    }
    public Person getChild(){
        return child ;
    }
};
class Book{             // 定义Book类
    private String title ;  // 标题
    private float price ;   // 价格
    private Person person ; // 一本书属于一个人
    public Book(String title,float price){
        this.setTitle(title) ;
        this.setPrice(price) ;
    }
    public void setTitle(String t){
        title = t ;
    }
    public void setPrice(float p){
        price = p ;
    }
    public String getTitle(){
        return title ;
    }
    public float getPrice(){
        return price ;
    }
    public void setPerson(Person p){
        person = p ;
    }
    public Person getPerson(){
        return person ;
    }
};
public class RefDemo06{
    public static void main(String arg[]){
        Person per = new Person("张三",30) ;
        Person cld = new Person("张草",10) ;  // 定义一个孩子
        Book bk = new Book("JAVA SE核心开发",90.0f) ;
        Book b = new Book("一千零一夜",30.3f) ;
        per.setBook(bk) ;       // 设置两个对象间的关系,一个人有一本书
        bk.setPerson(per) ;     // 设置两个对象间的关系,一本书属于一个人
        cld.setBook(b) ;        // 设置对象间的关系,一个孩子有一本书
        b.setPerson(cld) ;      // 设置对象间的关系,一本书属于一个孩子
        per.setChild(cld) ;     // 设置对象间的关系,一个人有一个孩子
        System.out.println("从人找到书 --> 姓名:" + per.getName()+";年龄:" 
            + per.getAge() +";书名:" + per.getBook().getTitle() + ";价格:" 
            + per.getBook().getPrice()) ;   // 可以通过人找到书
        System.out.println("从书找到人 --> 书名:" + bk.getTitle() + ";价格:"
            + bk.getPrice() + ";姓名:" + bk.getPerson().getName() + ";年龄:"
            + bk.getPerson().getAge()) ;    // 也可以通过书找到其所有人
        // 通过人找到孩子,并找到孩子所拥有的书
        System.out.println(per.getName() + "的孩子 --> 姓名:"
            + per.getChild().getName() + ";年龄:" + per.getChild().getAge()
            + ";书名:" + per.getChild().getBook().getTitle() + ";价格:"
            + per.getChild().getBook().getPrice()) ;
    }
};

运行结果:

从人找到书 --> 姓名:张三;年龄:30;书名:JAVA SE核心开发;价格:90.0
从书找到人 --> 书名:JAVA SE核心开发;价格:90.0;姓名:张三;年龄:30
张三的孩子 --> 姓名:张草;年龄:10;书名:一千零一夜;价格:30.3