本文是按照Thinking in Java的章节温习的。仅记录有注释的章节。
第一章 对象入门 Introduction to Objects
1.
面向对象的三个基本特征
Java的类有个默认类型,只能被同一package的其他类访问。
第二章 一切皆对象 Everything is an Object
1. 栈和堆的区分
栈,因为本身是FILO - first in, last out. 先进后出,能够自动释放,不需要垃圾回收。而堆是内存池,保存数据比栈灵活,需要用new命令创建对象,所以垃圾回收GC是针对堆Heap。
同时因为栈的特性,所以方法/函数的变量,引用是存放在栈里,当退出作用域后则自动清理这些变量!而且栈的读取速度比堆所在的内存要快,因为堆需要定位数据,而栈不需要,所以Java代码优化--尽可能地使用stack(栈)变量,实际上就是尽量使用局部变量,把成员变量引用传递到函数方法里面的变量!
首先分清楚Stack,Heap的中文翻译:Stack—栈,Heap—堆。
在中文里,Stack可以翻译为“堆栈”,所以我直接查找了计算机术语里面堆和栈开头的词语:
堆存储: heapstorage 堆存储分配: heapstorage allocation 堆存储管理: heap storage management
栈编址: stack addressing 栈变换:stack transformation 栈存储器:stack memory 栈单元: stack cell
接着,总结在Java里面Heap和Stack分别存储数据的不同。
Heap(堆) | Stack(栈) | |
JVM中的功能 | 内存数据区 | 内存指令区 |
存储数据 | 对象实例(1) | 基本数据类型, 指令代码,常量,对象的引用地址(2) |
1. 保存对象实例,实际上是保存对象实例的属性值,属性的类型和对象本身的类型标记等,并不保存对象的方法(方法是指令,保存在stack中)。
对象实例在heap中分配好以后,需要在stack中保存一个4字节的heap内存地址,用来定位该对象实例在heap中的位置,便于找到该对象实例。
对象实例在heap中分配好以后,需要在stack中保存一个4字节的heap内存地址,用来定位该对象实例在heap中的位置,便于找到该对象实例。
2. 基本数据类型包括byte、int、char、long、float、double、boolean和short。
函数方法属于指令.
函数方法属于指令.
=======================
引用网上广泛流传的“Java堆和栈的区别”里面对堆和栈的介绍;
"Java 的堆是一个运行时数据区,类的(对象从中分配空间。这些对象通过new、newarray、anewarray和multianewarray等指令建立,它们不需要程序代码来显式的释放。堆是由垃圾回收来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。"
“栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类型的变量(,int, short, long, byte, float, double, boolean, char)和对象句柄。 ”
2. 成员变量的默认值和局部变量的初始化
Java和C++的其中不同之处:在Java里面,类里面的成员变量是有默认值的:
变量类型 默认值
Boolean false
Char '\u0000'(null) 显示为空白
byte (byte)0
short (short)0
int 0
long 0L
float 0.0f
double 0.0d
而对象引用则默认值为null。
但
函数方法里面的变量是不会分配默认值的,这点和C++是一致的。可以参考一下程序:
- public class Demo {
- int i;
- char c;
- String s;
- public void print() {
- //Error: 可能损失精度
- //找到: double
- //需要: float
- //float f = 1.1;
- float f = 1.1f;
- int num;
- String ss;
- //Error: 可能尚未初始化变量 num
- //System.out.println(num);
- System.out.println(i);
- System.out.println(c);
- //Error: 可能尚未初始化变量 ss
- //if (ss == null)
- if (s == null)
- System.out.println("null");
- }
- public static void main(String[] args) {
- new Demo().print();
- }
- }
输出结果为
0
空白
null
***所以函数方法里面的变量一定要初始化。
3. float 类型的赋值
从上一个程序可以看到,当需要给float类型的变量赋值,需要加上后缀
f 来表示。
4. 参数的传递
要小心把对象传递给方法的形参时,所传递的是引用,就是说在方法能够修改原来对象里面的数据。而为了避免这个问题,则对形参加上final修饰词,使得这个形参在方法里不能修改所引用的对象的数据。