如何区分Java中的基本类型primitive types
和引用类型reference types
?
在Java虚拟机规范(Java Virtual Machine Specification
)中,数据类型分为
- 基本类型
primitive types
- 引用类型
reference types
引用类型又可以分为
- class型
- 数组
array
型 - 接口
interface
型
方法1:区分基本类型和引用类型,有一个简单的规则,就看它能不能赋null值:能,就是引用类型;不能,就是基本类型。例如:
- int x = null; //语法错误
- Integer y = null; //语法正确
- String z = null; //语法正确
例如:boolean, byte, char, short, int, long,float,double
这8个就不能赋值为null
,当我们在程序中定义它们的时候,JVM就直接赋了初始值了。
方法2:8大基本类型对应的包装类都是引用类型
方法3:引用类型都要用new关键字来创建实例
方法4:引用类型不能用加减乘除+ - * / %
数学运算符合。其他的编程语言可能支持操作符重载,Java是不支持的。
特殊的情形:语法设计者为了方便大家写代码,设计了几个引用类型的语法糖。
-
Integer x = 50;
编译后的字节码等效于Integer x = Integer.valueOf(50);
看Integer.valueOf
的源码可知本质上还是new
-
int[] y = {1, 2, 3};
编译后的字节码等效于int[] y = new int[]{1, 2, 3};
-
Integer a = Integer.valueOf(1) + Integer.valueOf(2);
虽然能用+
号,实际编译后的字节码使用了Integer.intValue
进行了拆箱操作 -
String s = "a" + "ab".substring(1);
虽然能用+号,实际编译后的字节码是用StringBuilder.append
实现的 -
String y = "abc";
Java对常量字符的编译有单独的优化, 这个可以单开博文来讨论 -
Runnable run = () -> {};
Java8开始的Lambda表达式,实际new了一个CallSite
,语法上没有new
而已