什么是引用类型和值类型
我们知道java中变量的类型从大的分可分为值类型和引用类型,那么问题来了,什么是引用类型什么是值类型?其实基本上值类型就是八种基本类型,注意我这里说的是基本上,至于为什么,后面会有解释,除了基本类型剩下的都是引用类型。
基本类型:
基本类型自然不用说了,它的值就是一个数字,一个字符或一个布尔值。java中有四类八种:
四类:1,整型 2,浮点型 3,字符型4,逻辑型
八种:byte,short,char,int,long,float,double,boolean
引用类型:
是一个对象类型,值是什么呢?它的值是指向内存空间的引用,就是地址,所指向的内存中保存着变量所表示的一个值或一组值。
在搞清楚值类型与引用类型之后,那么它们之间用起来有什么区别呢?这就引出了下一个话题就是值传递与引用传递。
值传递与引用传递
talk is cheap show, me your code
package com.lsj;/** * 讲解java中的引用传递和值传递 * @author vi@jolie * */public class TestValueTransmit { public static void main(String[] args) { // TODO Auto-generated method stub int i = 0; User user = new User(1);// String s ="Hello java"; testInt(i); System.out.println("main 方法中的int值"+i); testUser(user); System.out.println("main 方法中"+user);// testString(s);// System.out.println("main 方法中的string值"+s); } private static void testInt(int m) { m ++; }// private static void testString(String s) {// System.out.println("testString 方法中的初始值是:"+s);// s = "Hello world";// System.out.println("testString 方法中的结果是:"+s);// } private static void testUser(User user) { user.age ++; } }class User{ int age; public User(int age) { super(); this.age = age; } @Override public String toString() { return "用户的年龄是:" + age; } }复制代码
结果输出如下:
main 方法中的int值0main 方法中用户的年龄是:2复制代码
仔细观察看出它们的区别了么?
i在调用testInt()函数之后进行加1操作,可是实际上却并没有加1。因为int属于八种基本类型之一,是值类型,而值传递传递的是实实在在的变量值,是传递原参数的拷贝,值传递后,实参传递给形参的值,形参发生改变而不影响实参。也就是说testInt()这个方法吧main()函数中i的值0传递给了函数testInt(),在testInt()中吧0这个值付给了m,对m进行加1操作,当然不会影响到main()函数中i的值。而testUser()则不是这样,因为User不是八种基本类型值一,因此是引用类型,而引用传递传的是地址,就是将实参的地址传递给形参,也就是说两者指向的是同一块内存,所以当testUser对user进行操作时,内存里的值发生了变化,两者指向的又是同一块内存地址,取到的是同一个东西,因此都会发生改变,打个比方,篮子只有一个苹果,有两个人都看见了,第二个人拿起了咬一口,那么第一个人再去拿苹果的时候,那个苹果是已经被咬了一口的,以为只有一个苹果。到此本文可以说结束了,但是记得文章开始前,有个小问题,为什么说基本上值类型就是八种基本类型,难道还有特殊的?有!看如下代码
ublic class TestValueTransmit { public static void main(String[] args) { // TODO Auto-generated method stub String s ="Hello java"; String s1 ="Hello java"; String s2 = new String("Hello java"); String s3 = new String("Hello java"); System.out.println("s==s1"+(s==s1)); System.out.println("s2==s1"+(s2==s1)); System.out.println("s2==s3"+(s2==s3)); } }复制代码
输出结果如下:
s==s1trues2==s1falses2==s3false复制代码
是不是感到很奇怪,按道理如果String是引用类型,s == s1应该是false啊。别急,慢慢说道说道。首先首先 == 号比较的是引用的地址是否相当,而s 和 s1 都是String 类型的引用,被声明在栈里,"Hello java”是字符串常量存储在常量区内。但是编译器做了优化,当发现之前已经有"Hello java"后( String s ="Hello java"),便让s1直接指向"Hello java"不会重新创建一个"Hello java"的常量,即现在s和s1都是指向的同一个"Hello java"了,所以为true。而String s2 = new String("Hello java") 是在堆中创建一个新的String实体,并拷贝“Hello java”的内容,并返回新的String实体的地址,赋值给指针s2,s2==s1false,s2==s3false,到此结束。