java 中的数据类型分为两大类:值类型(基本数据类型)和引用类型(复合数据类型) 
    值类型分为 1,整数类型(byte,short,int,long) 2,浮点类型(float,double) 3,布尔类型(boolean) 4,字符类型(char)    栈
    引用值类型 除了值类型外就是引用值类型了,引用值类型有1,类类型(class) 2,数组 3,接口类型(interface)     堆

  Java中一切都是对象。所有的函数,变量,常量都必须在一个类里定义。这本来很好,很完美的规定。但是Java在一开始设计的时候,出于性能上的考虑,却留下了一道口子:primitive类型,我们在开发中也常称为值类型,或者干脆按字面意思翻译为基础类型或者原始类型。

  primitive类型包括byte, char, short, int, long, float, double, boolean。我们经常会用到这些类型,并不陌生。Java提供了这些基础类型以后,为了仍然保持强对象的要求(即Java中的所有东西都是对象),又引入了它们的包装类(wrapper class)。分别是

  byte -> Byte
  char -> Character
  short -> Short
  int -> Integer
  long -> Long
  float -> Float
  double -> Double
  boolean -> Boolean

自动拆装箱:
  这就是说,同样一个整数,在Java有两种表示方法,一个是值,一个是对象。Java还怕你用得不顺手,又提供了一个自动拆装箱的功能。所谓自动装箱,就是说,在需要使用对象的地方,传一个值,JVM会自动帮你转成对象,例如:
public class Test {
public static void main(String args[]) {
Integer a = 1;
Integer b = new Integer(1); Integer c = inc(0);
} public static int inc(Integer x) {
return x + 1;
}
}

所谓自动拆箱,就是反过来,在需要使用值的地方,允许你传一个对象,比如:

public class Test {
public static void main(String args[]) {
int a = new Integer(1);
int b = Integer.valueOf(1);
int c = inc(0);
} public static Integer inc(Integer x) {
return x + 1;
}
}
public class Test {
public static void main(String args[]) {
Integer a = new Integer(1);
Integer b = Integer.valueOf(1);
Integer c = inc(0);
Integer d = 1;
System.out.println(a == b); // false
System.out.println(a.equals(b)); // true System.out.println(a == c); // false
System.out.println(a.equals(c)); // true System.out.println(a == d); // false
System.out.println(a.equals(d)); // true System.out.println(b == c); // true
System.out.println(b.equals(c)); //true System.out.println(b == d); // true
System.out.println(b.equals(d)); // true System.out.println(((Long)1L) == 1); // true
System.out.println(new Long(1).equals(1)); // false Long e = 100;
Long f = 100;
System.out.println(e == f); // true
e = 1000;
f = 1000;
System.out.println(e == f); // false
} public static Integer inc(Integer x) {
return x + 1;
}
}

wrapper类留下了多少坑,由此可见一斑,这也许还不是全部的,因为我的例子构造的未必那么全。这已经不是一个设计的品味或者取舍权衡的问题了。这完全就是错误的。为了使这种错误少出现,我们能不使用外包类的时候还是尽量不要使用外包类,除非在使用容器,泛型等等的时候,不得不使用。跟这么多出乎意料的结果比起来,审美和品位已经不是最重要的了。

我们还是尝试着解答一下上面的各个结果。首先明确一点,"=="判断的是两边的对象是不是同一个对象,即两个引用指向同一个对象。"equals"是函数,如何实现取决于类里如何定义。我们以JDK中Long的源代码举例:

class Long {
public Long(long value) {
this.value = value;
} private static class LongCache {
private LongCache(){} static final Long cache[] = new Long[-(-128) + 127 + 1]; static {
for(int i = 0; i < cache.length; i++)
cache[i] = new Long(i - 128);
}
} public static Long valueOf(long l) {
final int offset = 128;
if (l >= -128 && l <= 127) { // will cache
return LongCache.cache[(int)l + offset];
}
return new Long(l);
} // 神坑!!!!
public boolean equals(Object obj) {
if (obj instanceof Long) {
return value == ((Long)obj).longValue();
}
return false;
}
}

可以看到,直接使用new构造一个长整型的时候,会生成一个新的对象。而使用 valueOf 的时候,如果参数位于[-128, 127]这个区间内,则直接从cache数组中取一个对象,而不是新建,这就回答了为什么e 和 f 为100时,"=="为true,而 e 和 f 为1000时, "=="为false。

自动装箱时,Java调用了valueOf函数,拆箱时,调用 xxxValue 函数。这就回答了,为什么b, c, d 是同一个对象,而与a却是不同的对象。

比较难理解的是

System.out.println(((Long)1L) == 1); // true
System.out.println(new Long(1).equals(1)); // false

“==”操作符的左边是一个Long型的对象,为了比较就使用了自动拆箱,左边拆箱成了整数1,所以与右边的值是相等的。而第二行,equals参数接受一个Object对象,所以1就被自动装箱成了Integer.valueOf(1),我们看一下Long的equals的实现。如果传进来的是Long型的,那么就比较值,如果传进来的是其他类型,直接返回false。。。。我隔着屏幕似乎都听到了你说“我擦嘞"的声音。。。

字面量:

字面量(literal)是用于表达源代码中一个固定值的记号。

定义看上去不那么直观。其实很简单,整数1,2, 3是整型字面量;用引号引起来的"hello world",是字符串(String)字面量;1.0, 2.1是浮点型字面量;'a', 'c'是字符型字面量。可以使用字面量初始化primitive类型,也可以用于初始化某些特定的对象。例如:

public class Test {
public static void main(String args[]) {
int a = 1;
char c = 'z';
String s = "hello java!";
double[] t = {1.0, 2.1, 3.4};
}
}

Java允许通过字面量给String类型的对象赋值,却不允许字面量本身就是一个对象。对于一个强对象的语言,这其实是很落后的做法。我们来看几个其他语言的例子:

# Python,字符串连接的写法
"-".join["hello", "world"] # hello-world
//javascript
var d = {"hello" : 1, "world" : 2};

其实JSON的结构不过就是javascript的字典类型的字面量而已。这种数据即代码的做法才是现代语言应该具备的特征。

我们看到,在很多种语言中,都可以把字面量当做一个对象进行调用。例如groovy和ruby里,都可以直接在一个整数上调用times,代表后面的代码块要执行多少次。在这些语言里,才真正的一切都是对象,就连函数都是对象(这个扯远了,我们会尝试在课程里引入一些scala的知识,会介绍函数做为对象带来了哪些变化)。

对比起来,Java不仅字面量不是对象,而且字面量向对象的转变还留下了一个个不大不小的坑。这已经不是bad taste的问题,而是强对象这件事情,Java 根本就没做对

java的数据类型、自动拆装箱、字面量的更多相关文章

  1. 一文读懂什么是Java中的自动拆装箱

    基本数据类型 基本类型,或者叫做内置类型,是Java中不同于类(Class)的特殊类型.它们是我们编程中使用最频繁的类型. Java是一种强类型语言,第一次申明变量必须说明数据类型,第一次变量赋值称为 ...

  2. Java中的自动拆装箱(转)

    出处: 一文读懂什么是Java中的自动拆装箱 本文主要介绍Java中的自动拆箱与自动装箱的有关知识.  基本数据类型 基本类型,或者叫做内置类型,是Java中不同于类(Class)的特殊类型.它们是我 ...

  3. 什么是Java中的自动拆装箱

    1.基本数据类型 基本类型,或者叫做内置类型,是Java中不同于类(Class)的特殊类型.它们是我们编程中使用最频繁的类型. Java是一种强类型语言,第一次申明变量必须说明数据类型,第一次变量赋值 ...

  4. java中的自动拆装箱与缓存(Java核心技术阅读笔记)

    最近在读<深入理解java核心技术>,对于里面比较重要的知识点做一个记录! 众所周知,Java是一个面向对象的语言,而java中的基本数据类型却不是面向对象的!为了解决这个问题,Java为 ...

  5. Java包装类的自动拆装箱

    题目: Integer i = 42; Long l = 42l; Double d = 42.0; 下面为true的是 A.(i == l) B.(i == d) C.(l == d) D.i.eq ...

  6. Java的自动拆/装箱

    作者:Alvin 关键字:语法糖 类 对象 参考 Java 中的语法糖 语法糖--这一篇全了解 浅谈 Integer 类 什么是Java中的自动拆装箱 深入剖析Java中的装箱和拆箱 前言 我们知道, ...

  7. java自动拆装箱

    介绍 Java 5增加了自动装箱与自动拆箱机制,方便基本类型与包装类型的相互转换操作.(关于基本类型与包装类型之前有记录过https://www.cnblogs.com/xiuzhublog/p/12 ...

  8. 自动拆装箱(int,Integer)

    包装类型Java语言是一个面向对象的语言,但是Java中的基本数据类型却是不面向对象的,这在实际使用时存在很多的不便,为了解决这个不足,在设计类时为每个基本数据类型设计了一个对应的类进行代表,这样八个 ...

  9. Java基础(二) 基本类型数据类型、包装类及自动拆装箱

    我们知道基本数据类型包括byte, short, int, long, float, double, char, boolean,对应的包装类分别是Byte, Short, Integer, Long ...

随机推荐

  1. Java8 改进的匿名内部类:

    1.匿名内部类适合创建那种只需要一次使用的类 2.匿名内部类定义格式: new 实现接口() | 父类构造器(实参列表){ //匿名内部类类体部分 } 3.从上面定义格式可以看出,匿名内部类必须实现一 ...

  2. highcharts点击事件系列

    http://www.highcharts.com/demo/line-ajax 参考设置(bar 柱状图) plotOptions: {                series: {       ...

  3. 刷新物化视图sql

    在plsql中新建command window,执行如下语句: exec dbms_mview.refresh('V_CTRL_POINT_PLAN_DATE');   -- V_CTRL_POINT ...

  4. 541. Reverse String II

    static int wing=[]() { std::ios::sync_with_stdio(false); cin.tie(NULL); ; }(); class Solution { publ ...

  5. 2018.09.20 atcoder 1D Reversi(模拟)

    传送门 考虑每次摆石头都会消去最外层的一个连续颜色串. 所以只用统计一下有多少段颜色即可. 代码: #include<bits/stdc++.h> using namespace std; ...

  6. cmake-add_definitions

    add_definitions: Adds -D define flags to the compilation of source files. add_definitions(-DFOO -DBA ...

  7. C#中验证sql语句是否正确(不执行语句)

    SET PARSEONLY检查每个 Transact-SQL 语句的语法并返回任何错误消息,但不编译和执行语句.SET PARSEONLY { ON | OFF }当 SET PARSEONLY 为 ...

  8. 理解Defer、Panic和Recover

    刚开始的时候理解如何使用Defer和Recover有一点怪异,尤其是使用了try/catch块的时候.有一种模式可以在Go中实现和try/catch语句块一样的效果.不过之前你需要先领会Defer.P ...

  9. Java 容器 LinkedHashMap源码分析1

    同 HashMap 一样,LinkedHashMap 也是对 Map 接口的一种基于链表和哈希表的实现.实际上, LinkedHashMap 是 HashMap 的子类,其扩展了 HashMap 增加 ...

  10. RabbitMQ 初探

    有哪些优点 可靠性:RabbitMQ 提供了多种技术可以让你在性能和可靠性之间进行权衡.这些技术包括持久性.投递确认.发布者证实和高可用性. 灵活的路由:提供了多种内置交换机类型.如果你有更复杂的路由 ...