基本类型 包装类型 堆与栈 的区别

package cn.nubia;

public class Demo1
{

	/**
	 * ==号,根据JDK的说明,只有在两个引用都指向了同一个对象时才返回真值。而我们在这里要看的是, str1与str2是否都指向了同一个对象。
	 * from:http://baike.baidu.com/view/93201.htm?fr=aladdin
	 * 
	 * @param args
	 */
	public static void main(String[] args)
	{
		String str1 = "I love Java";
		String str2 = "I love Java";
		String str3 = "I love Java too";// String是基本类型,声明的变量存放在栈中
		System.out.println(str1 == str2);// 定义str2时,它的值与str1相同,程序将会在内存中查找,如果找到一样的,则直接指向它,相当于两个变量公用一个值
		String str4;
		str4 = str2;
		System.out.println(str3 == str4);
		String str5 = new String("I love Java");// 对象str5存放在堆中,详细说明:java中所有对象的存储空间都是在堆中分配的,但是这个对象的引用确实在堆栈中分配,
		// 也就是说在建立一个对象时在两个地方都分配内存,在堆中分配的内存实际简历这个对象,而在堆栈中分配的内存指示一个指向这个堆对象的指针(引用)而已
		// 只要是用new()来新建对象的,都会在堆中创建,而且其字符串是单独存值的,即使与栈中的数据相同,也不会与栈中的数据共享。
		System.out.println(str1 == str5);// false:说明只要是用new()来新建对象的,都会在堆中创建,
		// 而且其字符串是单独存值的,即使与栈中的数据相同,也不会与栈中的数据共享。
		System.out.println(new String("I lova Java") == str1);
		System.out.println(str1.equals(str5));
		// 详细再说:
		/**
		 * 堆与栈都是java用来在RAM中存放数据的地方,和C++不同,java自动管理栈和堆,程序员不能直接的设置栈或堆,C++管理堆和栈参照:
		 * http://www.cnblogs.com/lln7777/archive/2012/03/14/2396164.html
		 * 堆:是一个运行时数据区(程序运行了才知道分配堆的内存
		 *     ),类的对象从中分配空间,这些对象通过new、newarray等指令建立,java中的对象和数组都是存放在堆中;
		 * 栈:存取速度比堆快,又比寄存器慢,栈的数据可以共享,例如上面的str1和str2,编译器处理String str1 =
		 *     "I love Java";
		 *    首先它会在栈中创建一个变量为str1的引用,然后查找栈中是否有"I lova Java"这个值,如果没找到,就在栈中将这个值存放起来
		 * <span style="white-space:pre">	</span>(因为存放在栈中,所以str1不会等于str5,因为str5存放在堆中,栈和堆的数据是无法做到共享的),接着处理String
		 * <span style="white-space:pre">	</span>str2="I love java"
		 * <span style="white-space:pre">	</span>,在创建了str2这个引用变量后,因为在栈中已经有"I love java"这个值,所以str2直接指向
		 * <span style="white-space:pre">	</span>"I love java",这样就出现了str1和str2同时指向一个值的情况,因此str1==str2的值是true;
		 */

		// 了解这些java的细节,对调试一些奇怪错误有非常大的帮助!
	}

}

申请限制

栈:在Windows下, 是向 低地址扩展的 数据结构,是一块连续的内存的区域。这句话的意思是栈顶的地址和栈的最大容量是系统预先规定好的,在 WINDOWS下,栈的大小是2M(也有的说是1M,总之是一个编译时就确定的常数),如果申请的空间超过栈的剩余空间时,将提示overflow。因此,能从栈获得的空间较小。
堆:堆是向 高地址扩展的 数据结构,是不连续的内存区域。这是由于系统是用 链表来存储的空闲 内存地址的,自然是不连续的,而 链表的遍历方向是由低地址向高地址。堆的大小受限于 计算机系统中有效的 虚拟内存。由此可见,堆获得的空间比较灵活,也比较大。因此在android程序中,申请的数组大小有限制, Max Heap Size,是堆内存的上限值,Android的缺省值是16M(某些机型是24M)

   示例:

import java.util.Vector;

public class Demo1 {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		Vector<String> vs = new Vector<String>();
		while (true) {
			vs.add("a");
		}
	}

}

报错:

Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
	at java.util.Arrays.copyOf(Unknown Source)
	at java.util.Arrays.copyOf(Unknown Source)
	at java.util.Vector.ensureCapacityHelper(Unknown Source)
	at java.util.Vector.add(Unknown Source)
	at OOM错误.Demo1.main(Demo1.java:13)



你可能感兴趣的:(基本类型 包装类型 堆与栈 的区别)