您当前的位置:首页 > 互联网教程

java新手关于基本参数传参和引用变量传参的区别

发布时间:2025-05-24 14:29:56    发布人:远客网络

java新手关于基本参数传参和引用变量传参的区别

一、java新手关于基本参数传参和引用变量传参的区别

1、要解决这个问题,首先你必须得理解两个概念。

2、值传递:在具体的实参传递到方法中定义的形参的时候,传递的是值的拷贝,也就是复制一份来的值,这样就有个两个值,在方法内部对参数进行处理的时候,实际上是处理的那个拷贝后的值,而原来的值更本没有改变,因此就不会动。

3、引用传递:在具体的实参传递到方法定义的形参的时候,传递的是引用的地址的拷贝,也就是对象在堆中的地址,这样的话指针指向实际对象的值是同一个,也就是你传递了引用,其实也就改变了对象本身。

4、ps:也有一种说法是引用传递也是值传递,其实是不同的说法,本质上都是一样的。

5、数组都是引用数据类型,因此属于引用传递。也就是说他们传递的过程会对值本身产生变化,而int是基本数据类型,属于值传递,传递前后参数和原始值没有任何关系,所以值不会发生变化。

二、java方法参数怎么引用传递boolean

Java中的参数传递:分为值传递和引用传递

但本质上,Java中只有值传递。引用传递,其实可以理解为传的是类似指针的东西。

值传递就是把基本变量的值拷贝一份,传递这个拷贝。引用传递则是传递的引用的地址,也就是该变量在内存空间的地址。

只有基本数据类型采用值传递,特点是传递的是值的拷贝,传递完后两者就没有关系了。也就是说方法内和方法外的值互不相干

注:8种基本数据类型以外的数据类型都为引用类型。

指的是在方法调用时,传递的参数是按引用进行传递,其实传递的引用的地址,也就是变量所对应的内存空间的地址。

传递的是一个拷贝,即副本。也就是说,对于一个参数传递,存在两个地址指向同一个内存空间。这里我们可以用内存分配示意图来体现

先说结论,String类型传递与基本数据类型的传递效果相似。

String类对象一旦创建,其内容不可更改:

String类的所有方法都不会改变String类对象内容,要改变String类对象的值就必须创建一个新的String对象。

也就是说,当进行参数传递时,如果方法内对String类对象的值进行了修改,那么实际上是创建了一个新的String类对象,然后让原来的变量指向它而已。但是这个“原来的变量”是一份拷贝副本,只是一开始创建的时候与主方法中的传递的值相同而已,现在改变之后,两者就毫无关系了。

三、java 多态怎么回事,如何实现

出自:

多态性:发送消息给某个对象,让该对象自行决定响应何种行为。

通过将子类对象引用赋值给超类对象引用变量来实现动态方法调用。

java的这种机制遵循一个原则:当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。

1.如果a是类A的一个引用,那么,a可以指向类A的一个实例,或者说指向类A的一个子类。

2.如果a是接口A的一个引用,那么,a必须指向实现了接口A的一个类的实例。

SUN目前的JVM实现机制,类实例的引用就是指向一个句柄(handle)的指针,这个句柄是一对指针:

一个指针指向一张表格,实际上这个表格也有两个指针(一个指针指向一个包含了对象的方法表,另外一个指向类对象,表明该对象所属的类型);

另一个指针指向一块从java堆中为分配出来内存空间。

The Java Virtual Machine does not require any particular internal structure for objects. In Sun's current implementation of the Java Virtual Machine, a reference to a class instance is a pointer to a handle that is itself a pair of pointers: one to a table containing the methods of the object and a pointer to the Class object that represents the type of the object, and the other to the memory allocated from the Java heap for the object data.(jvm规范中关于对象内存布局的说明)

1、通过将子类对象引用赋值给超类对象引用变量来实现动态方法调用。

BaseClass a1= c2;//BaseClass基类,DerivedC是继承自BaseClass的子类

a1.play();//play()在BaseClass,DerivedC中均有定义,即子类覆写了该方法

*为什么子类的类型的对象实例可以覆给超类引用?

自动实现向上转型。通过该语句,编译器自动将子类实例向上移动,成为通用类型BaseClass;

* a.play()将执行子类还是父类定义的方法?

子类的。在运行时期,将根据a这个对象引用实际的类型来获取对应的方法。所以才有多态性。一个基类的对象引用,被赋予不同的子类对象引用,执行该方法时,将表现出不同的行为。

在a1=c2的时候,仍然是存在两个句柄,a1和c2,但是a1和c2拥有同一块数据内存块和不同的函数表。

2、不能把父类对象引用赋给子类对象引用变量

BaseClass a2=new BaseClass();

在java里面,向上转型是自动进行的,但是向下转型却不是,需要我们自己定义强制进行。

c1=(DerivedC)a2;进行强制转化,也就是向下转型.

3、记住一个很简单又很复杂的规则,一个类型引用只能引用引用类型自身含有的方法和变量。

你可能说这个规则不对的,因为父类引用指向子类对象的时候,最后执行的是子类的方法的。

其实这并不矛盾,那是因为采用了后期绑定,动态运行的时候又根据型别去调用了子类的方法。而假若子类的这个方法在父类中并没有定义,则会出错。

例如,DerivedC类在继承BaseClass中定义的函数外,还增加了几个函数(例如 myFun())

当你使用父类引用指向子类的时候,其实jvm已经使用了编译器产生的类型信息调整转换了。

这里你可以这样理解,相当于把不是父类中含有的函数从虚拟函数表中设置为不可见的。注意有可能虚拟函数表中有些函数地址由于在子类中已经被改写了,所以对象虚拟函数表中虚拟函数项目地址已经被设置为子类中完成的方法体的地址了。

配合这到体有助理解: