zoukankan      html  css  js  c++  java
  • Java函数传递方式值传递

    在 C++  中函数调用的时候,

    1. 值传递: 一般是将源数据复制一份然后 重新操作,变量内存地址都不一样。
    2. 引用传递:这点比较特殊 传递的还是原来变量,只是变量的别名而已,函数内部修改会直接导致外部改变。
    3. 指针传递: 获取到函数外部对象的指针(也就是地址),然后函数里面会根据地址去操作地址下对象的值。

    Java中也类似结论如下(具体讲解):

    1. 在Java里面,当调用方法时,如果传入的数值为基本数据类型(包含String类型),形式参数的改变对实际参数不影响,就是值传递。
    2. 在Java里面,当调用方法时,如果传入的数值为非基本数据类型比如对象(String类型除外),传入的是该对象的地址,然后将该对 象的地址复制给形参,然后操作形参的时候会影响实参,但是切记也是值传递 不过传递的是对象地址

    1. 按值传递(深拷贝 一个一样的值)

    import java.util.Scanner;
    public class learnJava
    {
    	public static void main(String[] args)
    	{
    		int a = 12;
    		fun(a);
    		System.out.println("方法外部对象的A值:" + a);
    	}
    
    	private static void fun(int a)
    	{
    		a += 1;
    		System.out.println("函数内部对象的A值:" + a);
    	}
    }
    ================
    函数内部对象的A值:13
    方法外部对象的A值:12
    
    package mytest;
    
    public class Client
    {
    	public static void main(String[] args)
    	{
    		String a = "12";
    		fun(a);
    		System.out.println("方法外部对象的A值:" + a);
    	}
    
    	private static void fun(String a)
    	{
    		// a = new String("34");
    		a  = a + 34;
    		System.out.println("函数内部对象的A值:" + a);
    	}
    }
    /*
    * String  虽然是对象 但是 可以将其看成基本类型传递 是深拷贝  直接弄了一个value 一样但是地址不相干的对象。
    * 函数内部对象的A值:1234
    * 方法外部对象的A值:12
    * */

    注意:String类型也是引用类型为什么函数调用没有改变外部的原型的值呢?

       这主要是由于String内部保存值方式的原因 String是通过一个final char[] 数组进行保存的  由于final类型的数组值不能够改变,所以在外部调用函数时将String地址的引用进行传递给函数的参数,函数内部再修改这个引用时,由于final char[] 不可修改,所以再给变量赋值新值时都会重新创建一个String实例(这个新创建的String实例所指向的地址和外部string变量所指向的地址是不同的所以改变不了外部的变量)另外一个点说明因为final  String 对象是不可变的,所以可以共享 也就是说 String  op="abc" 和 String op1="abc" 指向的是同一个引用对象(字符串是常量;它们的值在创建之后不能更改 ---jdk Api)

    package mytest;
    import java.util.Scanner;
    public class learnJava
    {
    	public static void main(String[] args)
    	{
    		A a = new A(5);
    		fun(a);
    		System.out.println("方法外部对象的A值:" + a.a);
    	}
    
    	private static void fun(A a)
    	{
    		a.a += 1;
            // a = new A(); 即使这样 原来 fun(a) 调用的地方仍然不会改变。因为是值传递下的 地址传递而已。不是引用传递!!!
    		System.out.println("函数内部对象的A值:" + a.a);
    	}
    
    	static class A
    	{
    		public int a;
    
    		public A(int a)
    		{
    			this.a = a;
    		}
    	}
    }
    
    ===================================
    函数内部对象的A值:6
    方法外部对象的A值:6
    /**
     * 测试参数传值机制
     * @author ljj
     *
     */
    public class User4 {
        int id;        //id
        String name;   //账户名
        String pwd;   //密码
           
        public User4(int id, String name) {
            this.id = id;
            this.name = name;
        }
          
        public   void   testParameterTransfer01(User4  u){
            u.name="高小八";
        }
         
        public   void   testParameterTransfer02(User4  u){
            u  =  new  User4(200,"高三"); 理解在这的操作不会 影响到数据的原始层
        }
          
        public static void main(String[] args) {
            User4   u1  =  new User4(100, "高小七");
             
            u1.testParameterTransfer01(u1); 
            System.out.println(u1.name);
     
            u1.testParameterTransfer02(u1);
            System.out.println(u1.name);
        }
    }

    关于字符串的重点说明:

       java中字符串存在常量池中。 java的内存分为堆,栈,方法区(包括常量池)。 java中字符串存在常量池中。  方法区中主要存在类结构,静态变量。方法区又包含常量池,常量池保存字符串常量。
    String类的概念

       String类是用于字符串相关操作的一个类。

       类包括成员变量和方法。

       (1)String类有一个特殊的成员变量 final char[],保存着常量池中某个字符串的内存地址,也可以理解为一个指针。

       (2)String类有一些方法,如indexOf(),charAt()。String类没有对字符串进行修改的方法

           虽然String类没有修改字符串的方法,但保留字符串地址的成员变量是可以修改的,也就是说String类的对象可以指向另外的字符串。

    String 两种实例化方式解析

    String str= "abc" 创建方式

              创建对象的过程

              1 首先在常量池中查找是否存在内容为"abc"字符串对象

              2 如果不存在则在常量池中创建"abc",并让str引用该对象

              3 如果存在则直接让str引用该对象

    String str= new String("abc")创建方式
              创建对象的过程

             1 首先在堆中(不是常量池)创建一个指定的对象,并让str引用指向该对象。

             2 在字符串常量池中查看,是否存在内容为"abc"字符串对象

             3 若存在,则将new出来的字符串对象与字符串常量池中的对象联系起来(即让那个特殊的成员变量value的指针指向它)

             4 若不存在,则在字符串常量池中创建一个内容为"abc"的字符串对象,并将堆中的对象与之联系起来。(有可能此时常量池中的"abc"已经被回收,所以要先创建一个内容为"abc"的字符串对象)

    关注公众号 海量干货等你
  • 相关阅读:
    Java对象的生命周期与作用域的讨论(转)
    [置顶] Oracle学习路线与方法
    Java实现 蓝桥杯 算法训练 未名湖边的烦恼
    Java实现 蓝桥杯 算法训练 未名湖边的烦恼
    Java实现 蓝桥杯 算法训练 未名湖边的烦恼
    Java实现 蓝桥杯 算法训练 最大的算式
    Java实现 蓝桥杯 算法训练 最大的算式
    Java实现 蓝桥杯 算法训练 最大的算式
    Java实现 蓝桥杯 算法训练 最大的算式
    Java实现 蓝桥杯 算法训练 最大的算式
  • 原文地址:https://www.cnblogs.com/sowhat1412/p/12734154.html
Copyright © 2011-2022 走看看