`
ZangXT
  • 浏览: 118028 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

从“关于Java堆与栈的思考”一帖看错误信息的传播

阅读更多

        我对转贴的信息一直有敌意,原因如下:首先,除了制造更多的信息垃圾,转贴不会带来新的价值,想收藏的话一个链接足矣;其次,将错误信息以讹传讹,混淆视听。不妨选一个典型的例子说明一二。

        相信《关于Java堆与栈的思考》这个帖子大家并不陌生,转载量极大。但内容如何呢?我就试着分析一下。说明:以下内容,黑色字体为引用别人的帖子,红色字体是我的分析,蓝色字体是引用相关文献。

1. 栈(stack)与堆(heap)都是Java用来在Ram中存放数据的地方。与C++不同,Java自动管理栈和堆,程序员不能直接地设置栈或堆。

与C++不同,对于堆来说是有道理的。但C++的栈也是由编译器负责安排和布局的,这和java没什么区别。其实,java里提到的堆和栈应该是逻辑层面上的,是jvm划分的,并不同于C++的实际运行时内存,因此根本不是一个层面上的东西,最好不要进行比较。

2. 栈的优势是,存取速度比堆要快,仅次于直接位于CPU中的寄存器。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。另外,栈数据可以共享,详见第3点。堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。

 

        在逻辑上看,可以认为栈比堆快,但是请不要和寄存器比。上面已经提到,java所谓的堆和栈是逻辑上的,《The Java Virtual Machine Specification》中,将运行时数据区分为pc寄存器、java虚拟机栈、堆、方法区、运行时常量池、本地方法栈共6个部分,而CPU中的寄存器是物理实体。

        比如java执行一个加法操作,逻辑上看是读取栈顶的两个数据,执行加法,将结果写回栈顶,但实际的代码经过jvm jit编译之后很可能就是在寄存器中执行加法。因此逻辑意义上的比较根本没有什么价值。

        "栈数据可以共享",这个说法是该帖子"创造性"的提出来的。翻遍所有的java 资料,根本找不到这样的说法(现在看来,也不全是,国内已经有书籍将这个帖子完全照搬了,抄袭可耻!貌似有本《java程序员面试宝典》,看过的都知道,呵呵)。 

3. Java中的数据类型有两种。
  一种是基本类型(primitive types), 共有8种,即int, short, long, byte, float, double, boolean, char(注意,并没有string的基本类型)。这种类型的定义是通过诸如int a = 3; long b = 255L;的形式来定义的,称为自动变量。值得注意的是,自动变量存的是字面值,不是类的实例,即不是类的引用,这里并没有类的存在。如int a = 3; 这里的a是一个指向int类型的引用,指向3这个字面值。

    这里混淆了一个问题。首先要知道什么是字面值,所谓的字面值是形如1,3,2.0,2.0f,'c',"Hello"这样的形式出现的常量。基本类型变量里存的是什么?值。存到变量中就不存在"字面值"一说了!"如int a = 3; 这里的a是一个指向int类型的引用",这个例子就太不恰当了,我们要注意的是,int a = 3;我们得到的是一个空间,里面存的是3这个数。这里又"int类型的引用"云云,岂不缪哉?

  1. 这些字面值的数据,由于大小可知,生存期可知(这些字面值固定定义在某个程序块里面,程序块退出后,字段值就消失了),出于追求速度的原因,就存在于栈中。

    基本类型的变量存在栈里或者堆里不是由"大小可知,生存期可知"就能确定了。关键是上下文。

    比如

    void func(){

    int a = 3;

    }

    这自然是存在栈里的。

    class Test{
    int a = 3;

    }

    这就肯定是随对象放到堆里的。

    因此,不要孤立的看到基本类型就说放到栈里,看到引用类型就说放到堆里。区分引用变量和对象本身特别重要,这个下面再说。
      另外,栈有一个很重要的特殊性,就是存在栈中的数据可以共享。假设我们同时定义

      int a = 3; 
      int b = 3;

      编译器先处理int a = 3;首先它会在栈中创建一个变量为a的引用,然后查找有没有字面值为3的地址,没找到,就开辟一个存放3这个字面值的地址,然后将a指向3的地址。接着处理int b = 3;在创建完b的引用变量后,由于在栈中已经有3这个字面值,便将b直接指向3的地址。这样,就出现了a与b同时均指向3的情况。
      特别注意的是,这种字面值的引用与类对象的引用不同。假定两个类对象的引用同时指向一个对象,如果一个对象引用变量修改了这个对象的内部状态,那么另一个对象引用变量也即刻反映出这个变化。相反,通过字面值的引用来修改其值,不会导致另一个指向此字面值的引用的值也跟着改变的情况。如上例,我们定义完a与b的值后,再令a=4;那么,b不会等于4,还是等于3。在编译器内部,遇到a=4;时,它就会重新搜索栈中是否有4的字面值,如果没有,重新开辟地址存放4的值;如果已经有了,则直接将a指向这个地址。因此a值的改变不会影响到b的值。

    这段就特别离谱了。首先我们要知道基本类型的变量里存的是什么。参考《The Java Language Specification》:Primitive values do not share state with other primitive values. A variable whose type is a primitive type always holds a primitive value of that same type.也就是说基本类型的变量中存放的就是该基本类型的值。 

    下面我们再看那栈中变量是如何存储的。Java是将所有的局部变量按数组的方式编号存储的。其中,一个int,float占该数组中的1项,一个long,double变量占数组的两项,引用变量占1项。其余byte、char、short等得不到jvm承认的就按int处理了。

      看一个实际的例子:

    public class Main {

    public static void main(String[] args){

            int i = 3;

            int j = 3;

            int p = 10;

            int q = 10;

            Object o = new Object();

            char c = 'c';

            double d = 1;

            int m = 10;

        }

    }

     

    javap -c Main,反汇编处理:

    public static void main(java.lang.String[]);

    Code:

    0:    iconst_3

    1:    istore_1

    2:    iconst_3

    3:    istore_2

    4:    bipush    10

    6:    istore_3

    7:    bipush    10

    9:    istore    4

    11:    new    #2; //class java/lang/Object

    14:    dup

    15:    invokespecial    #1; //Method java/lang/Object."":()V

    18:    astore    5

    20:    bipush    99

    22:    istore    6

    24:    dconst_1

    25:    dstore    7

    27:    bipush    10

    29:    istore    9

    31:    return

     

    }

      其实反汇编之后可以看出,程序中的变量名已经不存在了,代码中是按变量在局部变量数组中的索引来处理的(学过编译的同学应该很容易明白)。这里main函数的栈帧(Frame)里局部变量的内存布局是:

    字节码中的istore后的参数指定存储的目标位置索引。   

    容易看出,即使两个变量存储的数据是相同的,它们也是各自存储自己的变量值,互不干涉。比如对p和q的操作是:

    4:    bipush    10  //将常数10放到栈顶

    6:    istore_3   //将栈定的10存入栈变量数组的第三个位置(即p)

    7:    bipush    10  //同上,同样存储的是10

    9:    istore    4

       当然还有最后的m,也是存储的10。并不是"由于在栈中已经有3这个字面值,便将b直接指向3的地址。这样,就出现了a与b同时均指向3的情况。"

       不过必须指出的是,的确存在数据共享的情形。当然,所谓的共享是指编译器在生成class文件时的数据共享。比如下面的例子:

    public class Main {

    public static void main(String[] args){

        int b1 = 64;

        int b2 = 64;

        int s1 = 128;

    int s2 = 128;

    int i1 = 65536;

    int i2 = 65536;

    double d1 = 15;

    double d2 = 15;        

        }

    }

    反汇编:

    public static void main(java.lang.String[]);

    Code:

    0:    bipush    64

    2:    istore_1

    3:    bipush    64

    5:    istore_2

    6:    sipush    128

    9:    istore_3

    10:    sipush    128

    13:    istore    4

    15:    ldc    #2; //int 65536

    17:    istore    5

    19:    ldc    #2; //int 65536

    21:    istore    6

    23:    ldc2_w    #3; //double 15.0d

    26:    dstore    7

    28:    ldc2_w    #3; //double 15.0d

    31:    dstore    9

    33:    return

      b1=64<128,我们用1个字节就可以存储这个数了,所以直接用bipush指令+操作数64,该指令一共占2个字节。64<s1=128<65536,常数128两个字节可以表示,因此这里生成的指令是sipush+操作数128,指令占3个字节。而到了65536的时候是如何处理的呢?编译器将它放到了常量池表里,这样编译器生成的class文件中,读取65536这个常数时,只需要1条读取指令和1个1字节的索引,共2个字节。如果直接存这个数的话,我们需要的是加载指令+四个字节操作数。当这个数多次出现时,通过索引进行操作的方式就可以较好的节省操作数所占的字节数(空间)。这样最大的好处就是能有效的减少class文件的大小。

      同理,当处理double类型数据的时候,常数15.0d也是放入常量池里的,这样读取常数进行赋值时,读取指令只用了3个字节(ldc2_w    #3;操作符1个字节,操作数2个字节,如果直接将15.0这个double常数放到指令中的话,单操作数就占去8个字节)。意义上面已经说明。

      注意:不要把我说的共享和原贴所谓的共享混起来。我说的共享是class文件中字面量常数的共享,而变量是独立的、互不影响的,重复一遍,就是double d1 = 15.0;double d2 = 15.0,d1和d2都存的都是15.0,互不影响,但用来初始化它们的参数在class文件中只保存了一份。

      
    4. String是一个特殊的包装类数据。即可以用String str = new String("abc");的形式来创建,也可以用String str = "abc";的形式来创建(作为对比,在JDK 5.0之前,你从未见过Integer i = 3;的表达式,因为类与字面值是不能通用的,除了String。而在JDK 5.0中,这种表达式是可以的!因为编译器在后台进行Integer i = new Integer(3)的转换)。

      对于Integer i = 3;编译器并不是进行了Integer i = new Integer(3)这样的转换,而是这样处理:

    Integer i = Integer.valueOf(3);

      为什么要这样处理呢?因为Java的设计者们有以下考虑(《The Java Language Specification》):

    Ideally, boxing a given primitive value p, would always yield an identical reference.[理想的情况下,将一个基本类型值p装箱,总是得到一个同样的引用] In practice, this may not be feasible using existing implementation techniques. The rules above are a pragmatic compromise. The final clause above requires that certain common values always be boxed into indistinguishable objects. The implementation may cache these, lazily or eagerly.

      因技术手段所限,于是有了这样的规定:

    If the value p being boxed is true, false, a byte, a char in the range \u0000 to \u007f, or an int or short number between -128 and 127, then let r1 and r2 be the results of any two boxing conversions of p. It is always the case that r1 == r2.

      也就有了这样的现象:

    Integer i = 3;

    Integer j = 3;

    System.out.println(i==j);

      打印true。

      而

    Integer i = 128;

    Integer j = 128;

    System.out.println(i==j);

      打印false。

    至于实现,有兴趣的可以参考Integer 类中public static Integer valueOf(int i)方法。

    关于这一点,有些人也给出了好玩的结论,比如说我上面的这个例子,他们说当变量在-128~127时,比较的是对象的值,而此范围之外是比较的是对象的地址。麻烦不?


    5. 关于String str = "abc"的内部工作。Java内部将此语句转化为以下几个步骤:
      (1)先定义一个名为str的对String类的对象引用变量:String str;
      (2)在栈中查找有没有存放值为"abc"的地址,如果没有,则开辟一个存放字面值为"abc"的地址,接着创建一个新的String类的对象o,并将o的字符串值指向这个地址,而且在栈中这个地址旁边记下这个引用的对象o。如果已经有了值为"abc"的地址,则查找对象o,并返回o的地址。
      (3)将str指向对象o的地址。
      值得注意的是,一般String类中字符串值都是直接存值的。但像String str = "abc";这种场合下,其字符串值却是保存了一个指向存在栈中数据的引用!

      这段讲解问题是比较严重的。String具体是如何处理的呢?前面通过大于65535的int值和一般的double值已经说明了class文件中数据共享的原理,String的处理也是一样的。javac对.java文件进行处理,最后生成Class文件的字节码表示时,会同时生成一个常量池结构。在将数据放入常量池时,会对重复的数据进行过滤(其实是一个Map),保证值保存一份。这一点可以参考OpenJDK中javac的实现,常量池的代码在openjdk\langtools\src\share\classes\com\sun\tools\javac\jvm\Pool.java中,看一下put方法的实现就都清楚了。

      我们可以通过分析class文件证明这些常量的唯一性,例如:

    public class Main {

    public static void main(String[] args){

    double d1 = 15;

    double d2 = 15;        

    String s1 = "Hello World!";

    String s2 = "Hello World!";

    }

    }

    反汇编:

    public static void main(java.lang.String[]);

    Code:

    0:    ldc2_w    #2; //double 15.0d

    3:    dstore_1

    4:    ldc2_w    #2; //double 15.0d

    7:    dstore_3

    8:    ldc    #4; //String Hello World!

    10:    astore    5

    12:    ldc    #4; //String Hello World!

    14:    astore    6

    16:    return
      

    常数15.0在常量池的第二项,而常量String "Hello World!"在常量池的第4项。可以通过javap -verbose 类名,得到常量池的信息。

    需要注意的是class文件中的常量池是静态信息,类加载的时候jvm会根据这个常量池的信息在内存中真正建立起常量池结构。

    那常量池中的String对象"Hello World!"是何时创建的呢?我们再来看《Java Virtual Machine Specification 》:

     

  • Loading of a class or interface that contains a String literal may create a new String object (§2.4.8) to represent that literal. This may not occur if the a String object has already been created to represent a previous occurrence of that literal, or if the String.intern method has been invoked on a String object representing the same string as the literal.
  • To derive a string literal, the Java virtual machine examines the sequence of characters given by the CONSTANT_String_info structure.
    • If the method String.intern has previously been called on an instance of class String containing a sequence of Unicode characters identical to that given by the CONSTANT_String_info structure, then the result of string literal derivation is a reference to that same instance of class String.
    • Otherwise, a new instance of class String is created containing the sequence of Unicode characters given by the CONSTANT_String_info structure; that class instance is the result of string literal derivation. Finally, the intern method of the new String instance is invoked.

通过上面的说明,我们知道类加载之后,常量池中已经创建好了字符串常量对象。

在我们的例子中就是,加载Main类的时候,创建了"Hello World!"所表示的字符串对象。因此,执行到String s1 = "Hello World!"; String s2 = "Hello World!";不过把通过#2解析出的字符串常量对象的引用赋给s1和s2。 

 

    综上可知,字符串字面中的常量数据是保存在常量池中的。相同的内容也没有帖子中所谓的栈查找过程,当然就更没有所谓的"栈地址旁边记录引用" 之类的东东了。

原帖中又弄了一大堆例子试图说明"栈数据共享"这个论断,这里就不一一引用了。

    还有一点需要交代一下,常量池在内存中的Perm区,这里一般是不会被垃圾回收器打扰的。所以,不要认为对常量字符串的引用不存在了之后它就会被回收了。

6. 数据类型包装类的值不可修改。不仅仅是String类的值不可修改,所有的数据类型包装类都不能更改其内部的值。

7. 结论与建议:

  (1)我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,我们创建了String类的对象str。担心陷阱!对象可能并没有被创建!唯一可以肯定的是,指向String类的引用被创建了。至于这个引用到底是否指向了一个新的对象,必须根据上下文来考虑,除非你通过new()方法来显要地创建一个新的对象。因此,更为准确的说法是,我们创建了一个指向String类的对象的引用变量str,这个对象引用变量指向了某个值为"abc"的String类。清醒地认识到这一点对排除程序中难以发现的bug是很有帮助的。

这里总结的不错。说到这简单提一下,其实很多书上经常将对象和引用混在一起。比如说Object obj = new Object();说obj是个对象云云。我感觉这是个很坏的习惯。按照《The Java Language Specification》上的说明:

There are two kinds of types in the Java programming language: primitive types (§4.2) and reference types (§4.3). There is also a special null type, the type of the expression null, which has no name.

在初学时就自然的把变量分为两种(基本类型变量和引用类型变量)是再好不过的了。我们也不用费力的去分析==和equals()什么关系。因为很自然,==就是比较变量的"值",基本类型变量存的就是数据值,数据相等就可以确定"=="为 true,而引用类型存的是可以找到对象的信息(一般的jvm实现中都将引用作为一个指针,起码SUN和IBM的虚拟机都是这么处理的,可以参考相关资料,这种意义下就可以认为引用保存的是地址值),如果两个引用变量的"值"能让我们找到同一个对象,那它们的"=="也为true。

  (2)使用String str = "abc";的方式,可以在一定程度上提高程序的运行速度,因为JVM会自动根据栈中数据的实际情况来决定是否有必要创建新对象。而对于String str = new String("abc");的代码,则一概在堆中创建新对象,而不管其字符串值是否相等,是否有必要创建新对象,从而加重了程序的负担。这个思想应该是享元模式的思想,但JDK的内部在这里实现是否应用了这个模式,不得而知。
String str = new String("abc");千万不要这样写代码!
  (3)当比较包装类里面的数值是否相等时,用equals()方法;当测试两个包装类的引用是否指向同一个对象时,用==。
  (4)由于String类的immutable性质,当String变量需要经常变换其值时,应该考虑使用StringBuffer类,以提高程序效率。

        虽然分析基本错误,但原帖中的结论是比较有价值的。

我这里只是以《关于java栈与堆的思考》为例说明一个现象,就是错误的知识因转载和共享也会成为大多数人认定的真理。类似的还有很多很多这样形成的"真理",比如:

《Java关键字final、static使用总结》中有一句话,注意:父类的private成员方法是不能被子类方法覆盖的,因此private类型的方法默认是final类型的。

        想不明白这个因为所以怎么推出来的,本来不相关的两个概念。

有一个帖子《如何优化JAVA程序开发,提高JAVA性能》,对性能分析给出了很多好的建议,但使用String类型举例子,结果带来了很多错误的证据。

还有一个耳熟能详的说法,"如果自己不定义构造方法,编译器自动提供一个public的缺省构造方法",谁说一定是public的呢?

再比如,"一个引用变量不用了之后,使用obj = null;是个好习惯",了解GC的话,这并不是什么好习惯。可以参考http://developers.sun.com/learning/javaoneonline/2007/pdf/TS-2906.pdf。

 

50
3
分享到:
评论
34 楼 hu_xuefeng 2016-05-25  
hsbljyy 写道
看来真的有很多谬误啊!

当变量在-128~127时,比较的是对象的值,而此范围之外是比较的是对象的地址。

估计说这话的人没有经过实际测试了。

试过,楼主是正确的.
33 楼 hu_xuefeng 2016-05-25  
dugu666 写道
Type mismatch: cannot convert from int to Integer

这个就是执行下面程序的结果
Integer i = 128;

Integer j = 128;

System.out.println(i==j);
其他的我就不说了,你自己认真检讨去吧!
楼主就是那种外行误导内行的那种人,我很憎恨,因为就是这种人,使很多真正的人才被埋没被压抑。。。很多外行的人很容易就被这种人给蒙蔽了。。。真正的人才被踩死也只能怨自己命不好了。。。呜呼哀哉。。。
这种人一旦握住权利之刀,对真正的人才是相当危险的。而他们在错杀人才的时候,可以说是浑然不知。。。


这个结果是:false, 本人试验过.
32 楼 nxkid1412 2015-09-12  
java编程思想也是人编的,虽说是圣经,但难免有人自己的理解在里面,记得编程思想里还有说类的构造方法隐式是static的,但也被证明是不一定对的,iteye有位大牛RednaxelaFX的博客有过证明,你可以去看看
scu_cxh 写道
我在这里补充几点:

2)“父类的private成员方法是不能被子类方法覆盖的,因此private类型的方法默认是final类型的”。这句话是正确的,具体的楼主可以参考《Java编程思想》第4版第143页的内容“类中所有的private方法都隐士地指定为final的”。
31 楼 scu_cxh 2014-06-18  
我在这里补充几点:
1)“还有一点需要交代一下,常量池在内存中的Perm区,这里一般是不会被垃圾回收器打扰的。所以,不要认为对常量字符串的引用不存在了之后它就会被回收了。”楼主的这句话说的不是很全面,运行时常量池在方法区,而方法区不一定在Perm区,只是在HotSpot虚拟机的实现中,将方法区放在了Perm区。另外虽然Java虚拟机规范并没有要求对方法区进行垃圾回收,但是在HotSpot虚拟机中是进行了回收的,对方法区的回收主要回收废弃常量和无用的类。
2)“父类的private成员方法是不能被子类方法覆盖的,因此private类型的方法默认是final类型的”。这句话是正确的,具体的楼主可以参考《Java编程思想》第4版第143页的内容“类中所有的private方法都隐士地指定为final的”。
3)“一个引用变量不用了之后,使用obj = null;是个好习惯”这个在我个人看来确实是一个好习惯,这个更加有利于GC回收无用对象占用的空间,何乐而不为?
4)“String str = new String("abc");千万不要这样写代码!”请给我一个“千万”不要写这样的代码的理由。这个说的过于绝对,对于以下几种方式的使用要看情况,根据自己的需求选择,
String str = new String("abc");
String str ="abc";
StringBuilder str = new StringBuilder("abc");
StringBuffer str = new StringBuffer("abc");
这几种方式各有优缺点。
按照楼主你这么说的这么绝对,JDK可以直接把这个API去掉了。
其他的大部分还是比较赞同楼主的看法。望楼主考虑以下这几点说法是否合理。
30 楼 hekp 2010-11-26  

我是来学习的~~
29 楼 longdick 2009-09-28  
lz具有学人的怀疑精神和严谨治学精神,赞一个
28 楼 ZangXT 2009-09-09  
凤舞凰扬 写道
   刚发完就看到25楼关于默认构造函数的帖子了,呵呵,这里确实是一个误区。默认的类的构造函数的可见性其实是和类本身的可见性相同的。
   哈哈,这个题目用来做SCJP看来是个不错的题目哦,十有八九答错!

关于“栈比堆快",可以参考《Practical Java》,里面有具体的说明。

关于“父类的private成员方法是不能被子类方法覆盖的,因此private类型的方法默认是final类型的。”,参考我的另一个帖子:
http://zangxt.iteye.com/admin/blogs/461227
27 楼 凤舞凰扬 2009-09-09  
   刚发完就看到25楼关于默认构造函数的帖子了,呵呵,这里确实是一个误区。默认的类的构造函数的可见性其实是和类本身的可见性相同的。
   哈哈,这个题目用来做SCJP看来是个不错的题目哦,十有八九答错!
26 楼 凤舞凰扬 2009-09-09  
   通过连接过来看了一下,呵呵,有些理解还是有问题的,给指出一下。
引用
与C++不同,对于堆来说是有道理的。但C++的栈也是由编译器负责安排和布局的,这和java没什么区别。其实,java里提到的堆和栈应该是逻辑层面上的,是jvm划分的,并不同于C++的实际运行时内存,因此根本不是一个层面上的东西,最好不要进行比较。
这个理解是偏差的,堆和栈本身就是逻辑概念,对于编译语言C甚至汇编也好,它们所用到的栈也只是由寄存器实现的。虚拟机的一个方面就是模拟硬件机器,从语言角度来说,两者的堆栈是一致的。
  
引用
在逻辑上看,可以认为栈比堆快
这个概念是不存在的,堆和栈的存储目的是不同的,两者本身就不存在比的概念,如果只是比所谓的速度,一个很深的栈访问是肯定比一个很小的堆访问要慢的。
  
引用
这里总结的不错。说到这简单提一下,其实很多书上经常将对象和引用混在一起。比如说Object obj = new Object();说obj是个对象云云。我感觉这是个很坏的习惯。按照《The Java Language Specification》上的说明:
呵呵,这个说法有些绝对了。 我们可以说obj是一个一个Object类型对象的引用,同样也可以说obj是Object的一个实例,其实这只是一个约定俗成的表达,不必于太过抠字眼的
  
引用
《Java关键字final、static使用总结》中有一句话,注意:父类的private成员方法是不能被子类方法覆盖的,因此private类型的方法默认是final类型的。
        想不明白这个因为所以怎么推出来的,本来不相关的两个概念。
这个说法是没错的,楼主没道理批判的说。
   
引用
还有一个耳熟能详的说法,"如果自己不定义构造方法,编译器自动提供一个public的缺省构造方法",谁说一定是public的呢?
这个有错么?倒是愿闻其详
25 楼 hsbljyy 2009-09-09  
ZangXT 写道
shenjianwangyi 写道

现在就给你一个理由 你说是 谁说一定是public的呢? 你来告诉我 为什么不一定是public的

http://zangxt.iteye.com/admin/blogs/421510


刚看了楼主http://zangxt.iteye.com/admin/blogs/421510这篇博文了,我刚刚测试了一下。的确默认构造器不一定是public的。

哦,我的测试环境是JDK 6.0

测试代码很简单:

public class Main {

    /**
     * @param args
     */
    public static void main(String[] args) throws Exception {
        Class<?> clazz = Test.class;
        Constructor<?>[] constructors = clazz.getDeclaredConstructors();
        System.out.println(constructors.length);
        for (Constructor<?> constructor : constructors) {
            System.out.println(constructor);
        }
    }

}

class Test{
}
24 楼 hsbljyy 2009-09-09  
看来真的有很多谬误啊!

当变量在-128~127时,比较的是对象的值,而此范围之外是比较的是对象的地址。

估计说这话的人没有经过实际测试了。
23 楼 ZangXT 2009-08-13  
patrickyao1988 写道
很佩服楼主啊!请问下《The Java Virtual Machine Specification》中文名是叫《深入java虚拟机》吗?能不能推荐下关于java虚拟机的好书?很有兴趣深入学习一下

这不是一本书。
《The Java Virtual Machine Specification》没有中文版,目前最新的是第三版。
《深入java虚拟机》是《Inside the Java Virtual Machine》,有第二版,有中文版。
关于虚拟机的书不多,这两本是权威。
关于虚拟机的更多资料可以看SUN的相关文档和JavaOne上一些讲座。
22 楼 patrickyao1988 2009-08-13  
很佩服楼主啊!请问下《The Java Virtual Machine Specification》中文名是叫《深入java虚拟机》吗?能不能推荐下关于java虚拟机的好书?很有兴趣深入学习一下
21 楼 ZangXT 2009-08-11  
shenjianwangyi 写道

现在就给你一个理由 你说是 谁说一定是public的呢? 你来告诉我 为什么不一定是public的

http://zangxt.iteye.com/admin/blogs/421510
20 楼 shenjianwangyi 2009-08-11  
ZangXT 写道
以讹传讹, 害死人啊...
堆栈的这篇烂文被批N次了
shenjianwangyi 写道
楼主 不要误导别人好吗

您好,可以给出我“误导”的理由吗?

现在就给你一个理由 你说是 谁说一定是public的呢? 你来告诉我 为什么不一定是public的
19 楼 java.lang.Object 2009-08-10  
dugu666 写道
Type mismatch: cannot convert from int to Integer

这个就是执行下面程序的结果
Integer i = 128;

Integer j = 128;

System.out.println(i==j);
其他的我就不说了,你自己认真检讨去吧!
楼主就是那种外行误导内行的那种人,我很憎恨,因为就是这种人,使很多真正的人才被埋没被压抑。。。很多外行的人很容易就被这种人给蒙蔽了。。。真正的人才被踩死也只能怨自己命不好了。。。呜呼哀哉。。。
这种人一旦握住权利之刀,对真正的人才是相当危险的。而他们在错杀人才的时候,可以说是浑然不知。。。

你用的JDK的版本有问题吧,或者你的设置的版本是兼容1.4吧。还有,楼主说得很好,不要在JSP里面写JAVA代码,测也不好测,为什么不写一个类,写一个main方法呢?
18 楼 kuchaguangjie 2009-08-10  
很好!!!虽然我看不太懂,但是提醒了我
17 楼 sjynt131 2009-08-10  
佩服!多谢你的分析,学到了不少知识
16 楼 wolfplanet 2009-08-08  
完全赞同您的观点。。。
15 楼 ZangXT 2009-08-08  
dugu666 写道

因为我用的是jsp!你应该对我说,jsp是不允许这样的!jsp必须这样:
Integer i = new Integer(128);

Integer j = new Integer(128);

System.out.println(i==j);

我给您回站内信了。请检查自己的设置,建议JSP中少写java代码。

相关推荐

    关于Java栈与堆的思考

    关于Java栈与堆的深入解析 Java作为一种广泛使用的编程语言,其内存管理机制是学习者必须掌握的核心概念之一。在Java中,栈(Stack)与堆(Heap)是用于存储数据的主要区域,它们各自承担着不同的职责,对于理解...

    详解java堆和栈

    ### 详解Java堆和栈 #### 一、引言 在Java编程中,理解堆(Heap)和栈(Stack)的概念及其区别对于程序员来说至关重要。本文将深入剖析这两个概念,并探讨它们之间的差异以及如何影响程序的运行。 #### 二、Java...

    区别Java中堆与栈区别Java中堆与栈

    Java 中堆与栈的区别 Java 中的堆和栈是两个不同的内存区域,分别用于存放不同类型的数据。堆是一个运行时数据区,类的对象从中分配空间,通过new、newarray、anewarray 和 multianewarray 等指令建立,垃圾回收器...

    解析Java栈与堆

    Java栈与堆的存储机制解析 Java栈和堆是Java语言中两个最基本的存储机制,它们都是Java用来在RAM中存放数据的地方。与C++不同,Java自动管理栈和堆,程序员不能直接地设置栈或堆。 1. 栈的存储机制: 栈的优势是...

    Java中堆内存和栈内存详解

    ### Java中堆内存和栈内存详解 #### 一、引言 在Java编程语言中,内存管理是一项核心技能。为了更好地理解和使用Java,必须清楚地了解堆内存与栈内存的区别及其工作原理。本文将深入探讨Java中堆内存与栈内存的概念...

    JAVA中堆和栈的概念

    声明变量是在栈空间开辟了一个位置,实例化之后才会开辟一个堆空间 被赋予空值的话则是将栈空间地址指向一个新的堆空间位置

    深入堆与栈 堆与栈的区别

    ### 深入理解堆与栈:堆与栈的区别 #### 一、基本概念与区别 在编程领域,特别是对于Java这种广泛使用的语言而言,理解内存管理中的“堆”与“栈”的概念及其区别至关重要。这不仅有助于我们更高效地编写代码,还...

    java中堆与栈的区别

    ### Java中堆与栈的区别详解 #### 一、概述 在Java编程语言中,了解堆与栈的区别对于理解和管理程序的内存使用至关重要。本文将详细解释Java中的堆和栈的概念,以及它们之间的区别。 #### 二、Java堆 Java堆是...

    Java中堆与栈的内存分配.pdf

    "Java中堆与栈的内存分配" Java是一种基于对象的编程语言,它的内存管理机制是自动的,开发者不需要手动分配和释放内存。但是,了解Java的内存管理机制是一件非常重要的事情。这篇文章将对Java中的堆和栈的内存分配...

    java堆、栈和常量池

    ### Java堆、栈和常量池详解 #### 一、Java内存模型概述 Java程序运行时,内存可以分为几个不同的区域: 1. **寄存器**:这部分内存由硬件直接支持,程序无法直接控制。 2. **栈**:用于存储基本类型的数据和对象...

    java里的堆和栈

    ### Java里的堆和栈 #### 一、概述 在Java编程中,“堆”与“栈”的概念至关重要。它们是程序运行时内存管理的核心部分。本文将深入探讨Java中堆和栈的区别及其工作原理。 #### 二、Java内存区域划分 在Java中,...

    详解java学习中堆与栈的内容

    ### 详解Java学习中堆与栈的内容 #### 一、引言 在Java学习过程中,堆(Heap)和栈(Stack)是两个非常重要的概念,它们对于理解Java内存管理机制至关重要。很多初学者在接触到这两个概念时往往感到困惑,本文将...

    Java中堆和栈的区别

    Java中的堆和栈是两种不同的内存区域,它们在程序运行时承担着不同的职责。栈主要用于存储基本类型变量和对象的引用,而堆则用于存储对象实例。了解它们的区别对于优化程序性能至关重要。 栈(Stack): 1. 栈是...

    java中的栈(深层了解java虚拟机对对象的内存分布)

    ### 深层解析Java虚拟机中的栈与堆:对象的内存分布 #### 核心概念:栈与堆的本质及作用 在Java编程语言中,理解栈(stack)和堆(heap)的概念及其工作原理对于深入掌握Java虚拟机(JVM)如何管理内存至关重要。栈和堆...

    String3.1-java堆和栈---马克-to-win Java视频

    String3.1-java堆和栈---马克-to-win java视频 马克Java社区 马克towin

    java中的堆和栈

    ### Java中的堆和栈 #### 一、概述 在Java编程语言中,堆和栈是两种重要的内存区域,它们各自负责存储不同类型的数据。理解这两者之间的区别对于掌握Java内存管理和性能优化至关重要。本文将深入探讨Java中堆和栈...

    java 内存中 堆、栈、常量池、方法区的总结

    在Java内存管理中,堆(Heap)、栈(Stack)、常量池(Constant Pool)和方法区(Method Area)是四个核心概念,它们在Java程序运行时扮演着不同的角色。 首先,方法区是用来存放类的信息、常量、静态变量等数据的...

    Java中堆与栈的区别.pdf

    在Java编程语言中,堆和栈是两种不同的内存区域,它们各自有着特定的功能和优势。栈主要用于存储基本类型的变量和对象引用,而堆则用于存储复杂对象实例。 栈内存主要特点是快速存取,因为它的数据结构是线性的,...

Global site tag (gtag.js) - Google Analytics