Java 引用传递

1、Java 应用程序有且仅有的一种参数传递机制,即按值传递
2、按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本
3、按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址(副本),而不是值的副本

代码:

public class Main {
    public static void main(String[] args) {
        Employee worker=new Employee("张三");
        Employee manager=new Employee("李四");
        swap(worker,manager);
        System.out.println(worker.name+" "+manager.name);
    }
}
  class Employee {

    public String name=null;

    public Employee(String n){
        this.name=n;
    }
    //将两个Employee对象交换
    public static void swap(Employee e1,Employee e2){
        Employee temp=e1;
        e1=e2;
        e2=temp;
        System.out.println(e1.name+" "+e2.name);
    }
}

结果:

李四 张三
张三 李四

虽然形参对象e1,e2的内容交换了,但实参对象worker,manager并没有互换内容。这里面最重要的原因就在于形参e1,e2是实参worker,manager的地址拷贝。

如果大家对JVM内存管理中Java栈 的局部变量区 ,就很好理解上面这句话。在JVM运行上面的程序时,运行main方法和swap方法,会在Java栈中先后push两个叫做栈帧 的内存空间。

main栈帧中有一块叫局部变量区的内存用来存储实参对象worker和manager的引用。

而swap栈帧中的局部变量区则存储了形参对象e1和e2的引用。

虽然e1和e2的引用值分别与worker和manager相同,但是它们占用了不同的内存空间。

当e1和e2的引用发生交换时,下面的图很清晰的看出完全不会影响worker和manager的引用值。

String,Integer, Double等immutable类型因为没有提供自身修改的函数,每次操作都是新生成一个对象,所以要特殊对待。可以认为是传值。

Integer 和 String 一样。保存value的类变量是Final属性,无法被修改,只能被重新赋值/生成新的对象。 

当Integer 做为方法参数传递进方法内时,对其的赋值都会导致 原Integer 的引用被 指向了方法内的栈地址,失去了对原类变量地址的指向。对赋值后的Integer对象做得任何操作,都不会影响原来对象。

代码:

public class Main {
    public static void main(String[] args) {
        // 建立并构造两个对象
        Person p1 = new Person("Alexia", "female");
        Person p2 = new Person("Edward", "male");

        System.out.println("对象交换前:p1 = " + p1.toString());
        System.out.println("对象交换前:p2 = " + p2.toString());

        // 交换p1对象和p2对象
        Person.swapObject(p1, p2);
        // 从交换结果中看出,实际对象并未交换
        System.out.println("\n对象交换后:p1 = " + p1.toString());
        System.out.println("对象交换后:p2 = " + p2.toString());

        // 建立两个对象数组
        Person[] arraya = new Person[2];
        Person[] arrayb = new Person[2];

        // 分别构造数组对象
        arraya[0] = new Person("Alexia", "female");
        arraya[1] = new Person("Edward", "male");
        arrayb[0] = new Person("jmwang", "female");
        arrayb[1] = new Person("hwu", "male");

        System.out.println(‘\n‘ + "对象数组交换前:arraya[0] = "
                + arraya[0].toString() + ", arraya[1] = "
                + arraya[1].toString());
        System.out.println("对象数组交换前:arrayb[0] = "
                + arrayb[0].toString() + ", arrayb[1] = "
                + arrayb[1].toString());

        // 交换这两个对象数组
        Person.swapObjectArray(arraya, arrayb);
        System.out.println(‘\n‘ + "对象数组交换后:arraya[0] = "
                + arraya[0].toString() + ", arraya[1] = "
                + arraya[1].toString());
        System.out.println("对象数组交换后:arrayb[0] = "
                + arrayb[0].toString() + ", arrayb[1] = "
                + arrayb[1].toString());

        // 建立两个普通数组
        int[] a = new int[2];
        int[] b = new int[2];

        // 给数组个元素赋值
        for (int i = 0; i < a.length; i++) {
            a[i] = i;
            b[i] = i + 1;
        }

        System.out.println(‘\n‘ + "基本类型数组交换前:a[0] = " + a[0] + ", a[1] = " + a[1]);
        System.out.println("基本类型数组交换前:b[0] = " + b[0] + ", b[1] = " + b[1]);

        // 交换两个基本类型数组
        Person.swapIntArray(a, b);
        System.out.println(‘\n‘ + "基本类型数组交换后:a[0] = " + a[0] + ", a[1] = " + a[1]);
        System.out.println("基本类型数组交换后:b[0] = " + b[0] + ", b[1] = " + b[1]);

        // 改变对象数组的内容
        Person.changeObjectArray(arraya, arrayb);
        System.out.println(‘\n‘ + "对象数组内容交换并改变后:arraya[1] = " + arraya[1].toString());
        System.out.println("对象数组内容交换并改变后:arrayb[1] = " + arrayb[1].toString());

        // 改变基本类型数组的内容
        Person.changeIntArray(a, b);
        System.out.println(‘\n‘ + "基本类型数组内容交换并改变后:a[1] = " + a[1]);
        System.out.println("基本类型数组内容交换并改变后:b[1] = " + b[1]);
    }
}

class Person {

    private String name;

    private String sex;

    public Person(String x, String y) {
        this.name = x;
        this.sex = y;
    }

    // 重写toString()方法,方便输出
    public String toString() {

        return name + " " + sex;
    }

    // 交换对象引用
    public static void swapObject(Person p1, Person p2) {
        Person tmp = p1;
        p1 = p2;
        p2 = tmp;
    }

    // 交换基本类型
    public static void swapInt(int a, int b) {
        int tmp = a;
        a = b;
        b = tmp;
    }

    // 交换对象数组
    public static void swapObjectArray(Person[] p1, Person[] p2) {
        Person[] tmp = p1;
        p1 = p2;
        p2 = tmp;
    }

    // 交换基本类型数组
    public static void swapIntArray(int[] x, int[] y) {
        int[] tmp = x;
        x = y;
        y = tmp;
    }

    // 改变对象数组中的内容
    public static void changeObjectArray(Person[] p1, Person[] p2) {
        Person tmp = p1[1];
        p1[1] = p2[1];
        p2[1] = tmp;

        // 再将p1[1]修改
        Person p = new Person("wjl", "male");
        p1[1] = p;
    }

    // 改变基本类型数组中的内容
    public static void changeIntArray(int[] x, int[] y) {
        int tmp = x[1];
        x[1] = y[1];
        y[1] = tmp;

        x[1] = 5;
    }
}

结果:

对象交换前:p1 = Alexia female
对象交换前:p2 = Edward male

对象交换后:p1 = Alexia female
对象交换后:p2 = Edward male

对象数组交换前:arraya[0] = Alexia female, arraya[1] = Edward male
对象数组交换前:arrayb[0] = jmwang female, arrayb[1] = hwu male

对象数组交换后:arraya[0] = Alexia female, arraya[1] = Edward male
对象数组交换后:arrayb[0] = jmwang female, arrayb[1] = hwu male

基本类型数组交换前:a[0] = 0, a[1] = 1
基本类型数组交换前:b[0] = 1, b[1] = 2

基本类型数组交换后:a[0] = 0, a[1] = 1
基本类型数组交换后:b[0] = 1, b[1] = 2

对象数组内容交换并改变后:arraya[1] = wjl male
对象数组内容交换并改变后:arrayb[1] = Edward male

基本类型数组内容交换并改变后:a[1] = 5
基本类型数组内容交换并改变后:b[1] = 1

说明:不管是对象、基本类型还是对象数组、基本类型数组,在函数中都不能改变其实际地址但能改变其中的内容。

上面的代码,对数组对象Person[],int[]进行切换,不会发生变化,但是对p1[1],p2[1],x[1],y[1]进行置换,就会发现可以改变数组的内容。

http://www.jb51.net/article/80188.htm

http://blog.csdn.net/yangliuy/article/details/43230681

http://www.cnblogs.com/lanxuezaipiao/p/3371224.html

时间: 2024-08-29 21:12:09

Java 引用传递的相关文章

java 引用传递和值传递

1.为什么要分值传递和引用传递: 基本类型存在在栈中,复合类型(对象)存在堆中.操作栈的速度要快于堆,且对象的复制相比基本类型不仅浪费内存而且速度比较慢. 从这里就可以看出来:对象是按照引用传递(数据库事务封装Connection对象传递的时候最能体现这点):基本类型是按照值传递. 2.那为什么String类型传递后会表现的值传递的特性:这并不是由于值传递而导致的,而是由于String类的不可变性(只读特性)导致的.String类型按照引用传递,但是一旦改变了,就变成新的对象了,原来引用指向的对

java引用传递的基本应用

package com.cloud.day1; public class Demo1 { public static void main(String[] args) { // 引用传递的基本应用 Demo2 d2=new Demo2(); d2.temp=500; System.out.println("fun调用前:"+d2.temp); fun(d2); System.out.println("fun调用后:"+d2.temp); System.out.pri

java引用传递和值传递

关于Java传参时是引用传递还是值传递,一直是一个讨论比较多的话题,有论坛说Java中只有值传递,也有些地方说引用传递和值传递都存在,比较容易让人迷惑.关于值传递和引用传递其实需要分情况看待,今天学习和分析一下,着急可以先看最后的结论. 1.基本类型和引用类型在内存中的保存 Java中数据类型分为两大类,基本类型和对象类型.相应的,变量也有两种类型:基本类型和引用类型.基本类型的变量保存原始值,即它代表的值就是数值本身:而引用类型的变量保存引用值,"引用值"指向内存空间的地址,代表了某

java 引用传递及基本应用

java的引用传递

最近看着李兴华讲师的java视频教程学习java,关于java引用传递方面的知识的总结. 基础知识 java的常用内存空间 栈内存空间:保存所有的对象名称(更准确地说是保存了引用的堆内存空间的地址) 堆内存空间:保存具体对象的具体属性内容. 全局数据区:保存static类型的属性 全局代码区:保存所有的方法定义 实例分析 class Person { private String name; private int age; private static String city = "北京&qu

java值传递or引用传递解惑

java中的參数传递本质上仅仅有值传递,无论你传的是什么,传过去的都仅仅是一个副本而已,这个副本作为方法的局部变量保存在栈中. 1.假设參数类型为基本数据类型,改动这个值并不会影响作为參数传进来的那个变量,由于你改动的是方法的局部变量,是一个副本. 2.假设传的是一个对象的引用,也是一样的,也是一个副本,可是这个副本和作为參数传进来的那个引用指向的是内存中的同一个对象,所以你通过这个副本也能够操作那个对象.可是假设你改动这个引用本身,比方让他指向内存中的另外一个对象,原来作为參数传进来的那个引用

Java中的值传递和引用传递

解释 1.Java中有没有引用传递? 答:Java中只有按值传递,没有按引用传递! 2.当一个对象被当作参数传递到一个方法中后,在此方法中可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递? 答:是值传递. Java参数,不管是原始类型还是引用类型,传递的都是副本(有另外一种说法是传值,但是说传副本更好理解吧,传值通常是相对传址而言).如果参数类型是原始类型,那么传过来的就是这个参数的一个副本,也就是这个原始参数的值,这个跟之前所谈的传值是一样的,如果在函数中改变了副本

Java 为值传递而不是引用传递

——reference Java is Pass by Value and Not Pass by Reference 其实这个问题是一个非常初级的问题,相关的概念初学者早已掌握,但是时间长了还是容易混淆,特此总结一下 一.值传递和引用传递 首先这里我们先看下两者的异同: 值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参 数的值. 引用传递:也称为传地址.方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,在方法执行中

java中参数传递--值传递,引用传递

java中的参数传递--值传递.引用传递 参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递. 在 Java 应用程序中永远不会传递对象,而只传递对象引用.因此是按引用传递对象.Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数.参数可以是对象引用,而 Java 应用程序是按值传递对象引用的. Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型.当作为参数传递给一个方法时,处理这两种类型的方式是相同的.两