前言:面对“byte b1=3;byteb2=7;byte b=b1+b2;”报错,而“int i1=3;int i2=7;int i=i1+i2;”不报错,进行了深入探究,从而引申出java基本类型之间赋值与运算操作的规律。通过自己制作的图例分析,达到对不同基本数据类型之间的赋值与运算结果的正确判断。不会再出现事实而非的回答,知道运算底层。好了,言归正传。
先送给大家今天我看到的一句话,觉得很有意义
如果你自己都模棱两可的话,不要指望Java虚拟机会明白你的意思。
一、 认识赋值运算符。
赋值使用操作符“=”。它的意思是“取右边的值(即右值),把它复制给左边(即左值)”。右值可以是任何常数、变量或者表达式(只要它能生成一个值就行)。但左值必须是一个明确的,已命名的变量。也就是说,必须有一个物理空间可以存储等号右边的值。
分类 基本数据类型 与 类数据类型 的不同
1、 对基本数据类型的赋值是很简单的。基本数据存储了实际的数值,而并非指向一个对象的引用,所以在为其赋值的时候,是直接将一个地方的内容复制到了另一个地方。
2、 但是在为对象“赋值”的时候,情况却放生了变化。对一个对象进行操作,我们真正操作的是对对象的引用。----这种特殊的现象通常称作“别名现象”,是Java操作对象的一种基本方式。
----摘抄自《Thinking In Java》P39
二、 Java中byte、short和char类型运算的细节。
Java中涉及byte、short和char类型的运算操作首先会把这些值转换为int类型,然后对int类型值进行运算,最后得到int类型的结果。因此,如果把两个byte类型值相加,最后会得到一个int类型的结果。如果需要得到byte类型结果,必须将这个int类型的结果显式转换为byte类型。
Java虚拟机中没有byte类型!!!
Java虚拟机对基本类型的操作基本都是在栈上完成的(这个是可信的,因为不是我说的)。我们知道,Java在处理一个语句的时候,首先它会先把用到的操作数压到栈中,然后再从栈中弹出进行计算,最后将结果再压回到栈中。任何对byte的操作也会如此。因此,Java对byte类型操作之前会将其压入到栈中。实际上,Java压入栈的不是byte类型,而是一个标准的int类型(32位,byte是8位),也就是说,Java虚拟机将我们短小可爱的byte类型压入栈之前,会先把它拉长成int类型再压栈。不过事实上在压栈之前也是int类型.这样一来,我们不管是在栈里还是从栈里弹出的byte类型实际上都是用int的长度存储的。这也就是我为什么说,Java虚拟机中没有byte类型。因为它们都被变成了int。
----摘抄自网络文章《int与byte的区别》
下面是我做的一些测试
三、 测试一数字值赋值给变量。
格式例如 Xxx x = 数值;//检验是否正确。
变量\数值 |
byte |
short |
int |
long |
float |
double |
byte |
|
|
√ |
X |
X |
X |
short |
|
|
√ |
X |
X |
X |
int |
|
|
√ |
X |
X |
X |
long |
|
|
√ |
√ |
X |
X |
float |
|
|
√ |
√ |
√ |
X |
double |
|
|
√ |
√ |
√ |
√ |
说明:例如byte b = 14;14是int类型,能成功,对应表格就打钩。 |
附带说明:
1、 检测数值是否超出变量范围。
2、 右值是常数
是可以确定,编译器可判断数值大小。
四、 检测二变量赋值给变量。
格式例如 Xxx 变量B=变量A; //检验是否正确。
变量B\变量A |
byte |
short |
int |
long |
float |
double |
byte |
√ |
X |
X |
X |
X |
X |
short |
√ |
√ |
X |
X |
X |
X |
int |
√ |
√ |
√ |
X |
X |
X |
long |
√ |
√ |
√ |
√ |
X |
X |
float |
√ |
√ |
√ |
√ |
√ |
X |
double |
√ |
√ |
√ |
√ |
√ |
√ |
说明:例如 |
||||||
int i = 14; |
||||||
byte b = i; //如果通过编译,就在对应的表格打钩。 |
附带说明:
1、 右值为变量
不确定数值大小,
只能简单判断空间的大小。
根据已知理论,结合两个图片做一些练习:
例1:
byte b = 4;
b = 3+7;
//编译通过,因为“3+7”在编译时直接变为int的“10”,根据表//一,可行。
例2:
byte b1 = 3;
byte b2 = 7;
byte b = b1+b2; //不能通过编译。因为根据原理二,“b1+b2”的//运算操作使其结果为int类型。
//根据表二,int类型的变量不能赋值给byte变量。
例3:
byte b1 = 3;
byte b2 = 7;
int i = b1+b2; //能通过编译。因为根据原理二,“b1+b2”的运//算操作使其结果为int类型。
//根据表二,int类型的变量能赋值给int变量。
例4:
int i1 = Integer.MAX_VALUE;
int i2 = 2;
int I = i1+i2; //能通过编译。因为根据原理二,“i1+i2”的运//算操作使其结果为int类型。
//根据表二,int类型的变量能赋值给int变量。
例5:
short s1=1;
s1 = s1+1; //不能通过编译。因为根据原理二,“s1+1”的运算//操作使其结果为int类型。
//根据表二,int类型的变量不能赋值给short变量。
例6:
short s1=1;
s1 = 1+1; //能通过编译。因为根据原理二,“1+1”的运算操作//使其结果为int类型。
//根据表一,int类型的数值能赋值给short变量。
例7:
short s1=1;
s1 += 1;
/*
能通过编译。这里“+=”运算符与“+”运算符原理一样,先检查右边,右边不符合对其强转。“s1=1”为一次运算;“s1+=1”为一次运算;“s1=s1+1”为两次运算。所以例题中是将“1”强转为“short”类型参与运算的。
*/
通过例题,获取小结
1、 右边不能判断,就不能赋值、转化。
2、 判断是否可以赋值,查看右边是常数(3,3+7)还是变量(x,x+7,x+y)。