对于java引用的总结

真的是一切都是对象?

首先得明确java中的对象存储位置:

1)  寄存器:最快的存储区,在处理器内部,暂时忽略。

2)  堆栈:速度仅次于寄存器,但是得明确规定数据的大小和生命周期,对象的引用存储于其中(java对象不存在此)。

3)  堆:java对象的存储区,无需知道数据大小,生命周期。

4)  静态存储:static来标识的对象存储区。

5)  常量存储:常量值通常直接存放在程序代码内部,他们永远不会被改变。

6)  非RAM,忽略。

StringBuffer stringBuffer=new StringBuffer(“hello, world”),见惯不惯的一个代码段。实际java的操作机制:new StringBuffer(“hello, world”)创建了一个对象,该对象被存储到堆中,其实这个对象我们看不见摸不到。为了能够操作该对象,就有了StringBuffer stringBuffer这样一个引用,这个引用存储在堆栈中,”=”将两者联系在了一起。既然stringBuffer是指向new StringBuffer(“hello,world”)这个对象的,再来个StringBuffer stringBuffer1=stringBuffer,这里其实是引用的赋值而非对象本身的赋值,同理若有stringBuffer1==stringBuffer这样一个语句,也是引用的事。以上都好理解,直到看到Bruce Eckel说:java中的任何传递对象的场合,这里的传递实际上是引用,这不是c中的形参实参的关系么。于是立马敲下代码:

package com.cedric.thinkingInJava.test;

class Person{
    String name;
    int age;
}

public class ReferenceAndArgs {
    
    public static void main(String[] args) {
        Person person=new Person();
        Person personBeforeChangeAge;
        
        setPersonInfo(person);
        personBeforeChangeAge=person;
        
        System.out.println("Before the change:");
        printPersonInfo(personBeforeChangeAge);
        
        System.out.println("After the change:");
        changeTheAge(person);
        printPersonInfo(person);
        printPersonInfo(personBeforeChangeAge);
        //这里可以认为所有的引用都指向了最初的new Person()这个对象
    }
    
    static void setPersonInfo(Person person) {
        person.name="nerd";
        person.age=22;
    }
    
    static void printPersonInfo(Person person) {
        System.out.println(person.name+" "+person.age);
    }
    
    static void changeTheAge(Person person) {
        person.age++;
        printPersonInfo(person);
    }
}

运行结果:

Before the change:

nerd 22

After the change:

nerd 23

nerd 23

nerd 23

从结果上看,的确所有的引用都指向了一个对象。再试一次:

package com.cedric.thinkingInJava.test;
public class test {

    static void numberIncreaseByOne(String number){
        number=new String("2");
        printNumber(number);
    }
    
    static void printNumber(String number) {
        System.out.println(number);
    }
    
    public static void main(String[] args) {
        String integer=new String("1");
        
        numberIncreaseByOne(integer);
        printNumber(integer);        
    }
}

结果是:

2

1

问题出现:String是个对象,那定义的integer和number应该指向同一个对象才对,实际并非如此,它是值传递。这里就回到了最开始的问题:真的是一切都是对象?其实在java里有基本变量,也就是常见的int char,long ,float,double等,他们不用new来创建,他们的值存储在堆栈中,这样可以更高效的运行,说白了他们不是对象。既然不是对象,他们在函数的参数传递时就是值的传递,和c中的实参传递一样。再来看String,google后知道它是封装类,看看它的源码:

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence
{
    /** The value is used for character storage. */
    private final char value[];

    /** The offset is the first index of the storage that is used. */
    private final int offset;

    /** The count is the number of characters in the String. */
    private final int count;
……

里面的final char value[]就可以说明一切了:首先它其实是对char的一个封装而已,char就是基本类型,此外又是final类型,final类型网上看来是存储在常量存储区的。常量存储区的数据时永远不会变的,没有像其他对象一样的引用,所以就有了虽然是对象,但是还是值传递。那么Integer,Float等就可以解释为什么也是值传递了。至于StringBuffer,它的源代码继承了AbstractStringBuffer:

abstract class AbstractStringBuilder implements Appendable, CharSequence {
    /**
     * The value is used for character storage.
     */
    char value[];

    /** 
     * The count is the number of characters used.
     */
    int count;

    /** 
     * This no-arg constructor is necessary for serialization of subclasses.
     */
    AbstractStringBuilder() {
}

这里的char并非final类型,和String的实现方法并不相同。

虽然知道了类似String这种封装类,但是里面真正的实现机制还是模糊的,下次带着这个问题去看final。

总结:了解了引用,第一次特意去了解StringBuffer stringBuffer=new StringBuffer(“hello, world”)的运行机制,知道了封转类。

posted @ 2012-06-11 09:52  oh_nerd  阅读(114)  评论(1)    收藏  举报