zoukankan      html  css  js  c++  java
  • JAVA中的指针

      不同于CPP,JAVA中不需要程序员对指针进行操作。不过,这不代表JAVA没有指针,事实上,JAVA的指针操作都被底层代码封装了。笔者在初学Java时,虽然就了解了形参,实参,StringBuffer这些概念,但一直只流于表面,对此没有一个深度的认识。直到最近开始学习JVM虚拟机,才真正认识到了JAVA的精妙之处。

      首先,先说结论,Java中所有的基本数据类型的传递,都是按值传递,即传递的都是形参。除此以外的其他任何传递都是按地址传递,传递过去的都是实参,即cpp概念中的引用传递。最初笔者学习时也是就看到这,虽然知道了结论,但由于概念过于抽象,始终无法理解其本源(或者说内存模型中到底发生了什么)。Talk is cheap,show me the code。下面,我们来看一段代码。

    package point;
    
    public class Formal {
        private void formal(int p){
            p=p*5;
            System.out.println(p);
        }
    
        public static void main(String[]args){
            int i=5;
            Formal fromal=new Formal();
            fromal.formal(i);
            System.out.println(i);
        }
    }

      这段代码在控制台输出的结果是25和5,很显然,按照正常的逻辑来说的话应该是25,25才对。为什么会出现这种结果?事实上,Java中对基本数据类型的传递都是值传递(不清楚为什么这样,笔者的理解是基本数据类型在内存中的组合是有限的,这样内存可以知道你需要传递的是什么,换而言之,它可以真正理解你要传递的东西,因此可以复制一份进行传递,这样虽然会增大程序员的学习成本,但可以节省内存或者是增进效率。而非基本数据类型因为变化太多了,所以无法这么操作。不知道本人的理解有没有误差,才疏学浅,希望大神们多多指正)。通俗一点说,就是复制操作,在内存中,JVM只是把你操作的数据复制了一份给你传递到的对象,而不是把它本身传递了过去。

      这里要说一个特例,String类,贴一段代码。

    package point;
    
    public class ActualString {
        private void actual(String color){
            color=color+" is red";
            System.out.println(color);
        }
    
        public static void main(String[]args){
            String car="myCar";
            ActualString actualString =new ActualString();
            actualString.actual(car);
            System.out.println(car);
        }
    }

      这里控制台输出结果是myCar is red,myCar,这不代表String传递的也是形参,或者是String是基本数据类型,事实上因为String类都是被final关键字修饰的,所以它的值不会变化,所以,字符串拼接之类的操作事实上只是在内存中开辟了一个新的空间,而原有的空间其实还存在,所以这就是为什么字符串的操作要使用StringBuffer类。

      下面我们看一下实参的代码。

    package point;
    
    public class Actual {
        private void actual(int arr[]){
            arr[0]=arr[0]*5;
            System.out.println(arr[0]);
        }
    
        public static void main(String[]args){
            int a[]=new int[]{5};
            Actual actual=new Actual();
            actual.actual(a);
            System.out.println(a[0]);
        }
    }

      毫无疑问,这次控制台输出的结果是25,25。这里传递的就是对象本身的地址值,即实参,可以理解为剪切操作,将自身传递过去。

      

    package point;
    
    class Variable{
        public static Variable VAR=null;
    
        public int i;
    
        protected void a(int p){
            p=p*5;
            System.out.println(p);
        }
    
        public Variable(){
            i=5;
            Variable.VAR=this;
        }
    
    
    }
    
    public class FormalInActual {
        public static void main(String[] args){
            Variable variable=new Variable();
            System.out.println(variable.i);
            variable.a(variable.VAR.i);
            System.out.println(variable.VAR.i);
        }
    
    }

      输出结果是5,25,5。即使是像这样把基本数据类型写进方法里,通过方法传递,其传递的依旧是形参。

  • 相关阅读:
    2020系统综合实践 第3次实践作业
    2020系统综合实践 第2次实践作业
    2020系统综合实践 第1次实践作业
    WireShark组 2019 SDN大作业
    2019 SDN阅读作业
    第07组 Beta版本演示
    OO第四单元总结
    OO第三单元总结
    OO第二单元总结
    OO第一单元总结
  • 原文地址:https://www.cnblogs.com/CNty/p/10913844.html
Copyright © 2011-2022 走看看