Java 参数的值传递和引用传递

在Java中,方法的参数的传递分为值传递(基本数据)和引用传递(引用数据:对象、字符串),这是最容易接受的。如果你能知道有这两种情况存在,那么,在遇到调用方法时,你可以避免很多问题的产生。但是,仔细查阅资料发现,Java中只有值传递。那么你会问,那为什么还叫引用传递呢?这让我想弄明白到底,方法执行过程中,参数是怎么使用的。

Java只有值传递

  这里我准备了一个事例用来分析,如下:

    @Test
    public void Test2() {
        A a = new A("apple");
        int b = 2;

        System.out.println();
        System.out.println("方法执行前:");
        System.out.println(a.toString());
        System.out.println(b);

        testParam(a, b);

        System.out.println();
        System.out.println("方法执行后:");
        System.out.println(a.toString());
        System.out.println(b);
    }

    private void testParam(A a1, int b1) {
        a1.setName("banana");
        b1 = 3;

        System.out.println();
        System.out.println("方法中:");
        System.out.println(a1.toString());
        System.out.println(b1);
    }

    class A {
        private String name;

        public A(String name) {
            this.name = name;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        @Override
        public String toString() {
            return "A{" +
                    "name='" + name + '\'' +
                    '}';
        }
    }

  输入结果:

  1:首先,所有的方法在调用时,参数都会进行拷贝

  想理解这句话,我觉得很有必要先弄懂 栈内存 和 堆内存。首先,java中的基本数据类型 和 指针(指针是指向一块内存地址的内存数据,也就是说指针本身是一个占用4字节内存的int(32 位系统内))都是存放在 栈内存 中的。所以,当方法被调用时,基本数据类型 和 指针 都进行了拷贝。

  

  也就是说,引用传递实际上也是对栈内存中的指针进行拷贝

  所谓传引用的说法是为了更好的区别基本数据类型,理解其调用方式。基于上面对指针的理解,我们不难看出,指针其实也是一个int值,所谓传引用,我们是复制了指针的int值进行传递。为了便于理解,我们可以姑且把指针看作一种数据类型,透明化指针的int特性,从而提出传引用的概念。

   在Java中,很多人说没有指针,事实上,在Java更深层次里,到处都是大师封装好的精美绝伦的指针。为了更容易的讲解Java中关于类和类型的调用,Java中出现了值与引用的说法。浅显的来说,我们可以认为Java中的引用与C中的指针等效。

  引用传递:当使用引用变量作为形参时,它将变为实参列表中相应变量的别名,对形参进行的任何更改都将真正更改正在调用它的函数中的变量。当以这种方式将数据传递给形参时,该实参被称为按引用传递。

下面我们通过一个反例来说明:「Java不是引用传递」

class B {
    private String name;
    private Integer age;

    public B(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "B{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }

    public static void exchange(B b3, B b4) {
        B b5 = b3;
        b3 = b4;
        b4 = b5;
    }

    public static void main(String[] args) {
        B b1 = new B("张三", 24);
        B b2 = new B("李四", 13);
        
        System.out.println("调用前");
        System.out.println(b1.toString());
        System.out.println(b2.toString());
        exchange(b1, b2);
        System.out.println("调用后");
        System.out.println(b1.toString());
        System.out.println(b2.toString());
    }
}

  打印结果是:

  我们发现b1和b2的值没有变化,我们只是将拷贝的b3和b4的值进行了交换,b1和b2引用仍然指向之前的对象。

加餐

  提问:如果传一个String字符串作为参数,并在方法中对字符串参数进行修改。站在 String字符串 是引用数据类型的前提下,那么你认为,在方法调用前和调用后,值会是一致的吗?

    @Test
    public void Test2() {
        String str = "1111";

        System.out.println("方法执行前:");
        System.out.println(str);

        testString(str);

        System.out.println();
        System.out.println("方法执行后:");
        System.out.println(str);
    }

    private void testString(String str1) {
        str1 += "2222";

        System.out.println();
        System.out.println("方法中:");
        System.out.println(str1);
    }

  如果你回答不一致,那么只能说,上面的你确实听懂了。不过,字符串这块的知识你需要再更深入的学习了。╰( ̄▽ ̄)╮╰( ̄▽ ̄)╮╰( ̄▽ ̄)╮

  输入结果:

  咦,怎么字符串str没有因为参数字符串str1的改变而改变呢?难道,参数str1没有修改str的堆内存内容吗,往这方面思考是对的,它确实没有修改。

  这里需要讲一个我们听过最多的,但是在用的时候又想不起来的知识点:字符串不能修改

  通过反编译字节码文件,可以看到,字符串在进行 + 运算符时,新建了一个 StringBuild对象,通过 该对象的  append()  方法进行字符串的拼接,然后通过  toString()  方法返回一个新的堆内存地址。所以str1 指向的是新的堆内存地址。

  

 

终于,对于Java 参数的传递问题,应该都明白了。( ^_^ )/~~拜拜

欢迎大家评论与交流,加油!!

 

【参考】

  1. https://www.cnblogs.com/laipDIDI/articles/2524309.html

 

posted @ 2019-08-01 13:04  坚持的力量々  阅读(3082)  评论(0编辑  收藏  举报