• java方法调用中的值传递和引用传递


    一. 首先java中的基础数据类型 byte,short,int,long,float,double,boolean,char

     1. 通过基本数据类型声明的局部都是存储java虚拟机栈当中

     2. 非基本数据类型,存储于java 堆内存当中

    二  基本类型 和 引用类型的不同之处

    int num = 10;
    String str = "hello";

    1. 如图所示,num是基本类型,值就直接保存在变量中。而str是引用类型,变量中保存的只是实际对象的地址。一般称这种变量为"引用",引用指向实际对象,实际对象中保存着内容。

     三  赋值运算符(=)的作用

    num = 20;
    str = "java";
    1. 对于基本类型 num ,赋值运算符会直接改变变量的值,原来的值被覆盖掉。
    2. 对于引用类型 str,赋值运算符会改变引用中所保存的地址,原来的地址被覆盖掉。但是原来的对象不会被改变(重要)。
    3. 如上图所示,"hello" 字符串对象没有被改变。(没有被任何引用所指向的对象是垃圾,会被垃圾回收器回收)

    四.  代码与结果

    public class ComboBox {
        public static void main(String[] args) throws Exception {
            int a = 100000;
            funA(a);// 传递的基本数据类型
            System.out.println("a = " + a);
            
            Integer b = new Integer(100000);
            funB(b);// 传递非基本数据类型(包装类)
            System.out.println("b = " + b);
            
            Integer c = new Integer(100000);
            funC(c);// 同B 
            System.out.println("c = " + c);
        }
        
        public static void  funA(int a) {
            a = 9999999;
        }
        public static void  funB(Integer b) {
            b = 9999999;
        }
        private static void funC(Integer c) throws Exception {
            //JDK未提供Integer修改vlaue值的方式,通过反射修改
            Field field = Integer.class.getDeclaredField("value");
            field.setAccessible(true);
            field.set(c, 999999);
        }
    }
    
    a = 100000
    b = 100000
    c = 999999

    FunA链路:

          

    FunB链路:

     

    FunC链路图

     

     五. 总计与分析

        局部变量和方法参数在jvm中的储存方法是相同的,都是在栈上开辟空间来储存的,随着进入方法开辟,退出方法回收。以32位JVM为例,boolean/byte/short/char/int/float以及引用都是分配4字节空间,long/double分配8字节空间。

        在JVM内存模型中有,stack和heap的存在,但是更准确的说,是每个线程都分配一个独享的stack,所有线程共享一个heap。对于每个方法的局部变量来说,是绝对无法被其他方法,甚至其他线程的同一方法所访问到的,更无法修改。

        当我们在方法中声明一个 int i = 0,或者 Object obj = null 时,仅仅涉及stack,不影响到heap,当我们 new Object() 时,会在heap中开辟一段内存并初始化Object对象。当我们将这个对象赋予obj变量时,仅仅是stack中代表obj的那4个字节变更为这个对象的地址。

       上面中的传递?FunA传递了栈内存具体的值 num = 10000(FunA stack中存一份副本,形参), FunB传递了栈内num对象的地址 num = 0x000002(FunB stack中存一份副本,形参), FunC和FunB 一样中传递了栈内num对象的地址 num = 0x000002(但是对目标对象堆内存中的值做了修改)

        如果是数据类型,会直接当栈中存储的基本值传递给了形式参数复制一份。如果是非基本数据类型,是把实际参数的引用的地址复制了一份,传递给了形式参数,简单说就是把实参对象引用的地址当做值传递给了形式参数。

  • 相关阅读:
    寻找完全数(C++)
    莱布尼兹三角形(C++)
    简单的素数问题(C++)
    ubuntu17.04下安装LNMP
    ubuntu下连接mysql出现Access denied for user 'rose'@'localhost' (using password: NO)的解决方法
    快速理解面向对象的PHP编程--基础篇
    百度电面总结
    操作系统基础知识
    快速理解C语言指针
    新手学习MongoDB的基本命令
  • 原文地址:https://www.cnblogs.com/chihirotan/p/13970720.html
Copyright © 2020-2023  润新知