动手动脑与动手

动手动脑一:

EnumTest.java:

程序代码:

public class EnumTest {

       public static void main(String[] args) {

              Size s=Size.SMALL;

              Size t=Size.LARGE;

              //s和t引用同一个对象?

              System.out.println(s==t);  //

              //是原始数据类型吗?

              System.out.println(s.getClass().isPrimitive());

              //从字符串中转换

              Size u=Size.valueOf("SMALL");

              System.out.println(s==u);  //true

              //列出它的所有值

              for(Size value:Size.values()){

                     System.out.println(value);

              }

       }

}

 enum Size{SMALL,MEDIUM,LARGE};

仔细阅读示例: EnumTest.java,运行它,分析运行结果?

System.out.println(s==t); string类型的判等,有数值和存储位置两部分。

System.out.println(s.getClass().isPrimitive());ava.lang.Class.isPrimitive()确定是否在指定的Class对象表示一个基本类型。有9个预定义的Class对象来表示的八种原始类型和无效(void)。这些都是由Java虚拟机创建的原始类型具有相同的名称,它们代表即布尔,字节,字符,短整型,长,浮点和双精度。

System.out.println(s==u);true;

二、你能得到什么结论?你掌握了枚举类型的基本用法了吗?

枚举类型是一种基本数据类型,而不是一种构造类型,因为它不能再分解为任何基本类型。不要和联合混淆。1、枚举值是常量,不是变量。不能在程序中用赋值语句再对它赋值。2、枚举元素本身由系统定义了一个表示序号的数值,从0开始顺序定义为012…。如在weekday中,sun值为0,mon值为1,…,sat值为6。

课后练习三

阅读相应教材,或者使用互联网搜索引擎,弄清楚反码、补码跟原码这几个概念,然后编写示例程序,对正数、负数进行各种位操作,观察输出结果,与手工计算的结果进行比对,看看Java中的数是采用上述哪种码表示的。

1. 原码

        原码就是符号位加上真值的绝对值, 即用第一位表示符号, 其余位表示值. 比如如果是8位二进制:

                [+1]原 = 0000 0001

                [-1]原 = 1000 0001

        第一位是符号位. 因为第一位是符号位, 所以8位二进制数的取值范围就是:

        [1111 1111 , 0111 1111]

        即

        [-127 , 127]

        原码是人脑最容易理解和计算的表示方式.

2. 反码

        反码的表示方法是:

                正数的反码是其本身

                负数的反码是在其原码的基础上, 符号位不变,其余各个位取反.

                [+1] = [00000001]原 = [00000001]反

                [-1] = [10000001]原 = [11111110]反

        可见如果一个反码表示的是负数, 人脑无法直观的看出来它的数值. 通常要将其转换成原码再计算.

3. 补码

        补码的表示方法是:

                正数的补码就是其本身

                负数的补码是在其原码的基础上, 符号位不变, 其余各位取反, 最后+1. (即在反码的基础上+1)

                [+1] = [00000001]原 = [00000001]反 = [00000001]补

                [-1] = [10000001]原 = [11111110]反 = [11111111]补

        对于负数, 补码表示方式也是人脑无法直观看出其数值的. 通常也需要转换成原码在计算其数值.

课后练习四

Java变量遵循“同名变量的屏蔽原则”,请课后阅读相关资料弄清楚相关知识,然后自己编写一些测试代码,就象本示例一样,有意识地在不同地方定义一些同名变量,看看输出的到底是哪个值。

每个变量都有一个“有效”的区域(称为“作用域”),出了这个区域,变量将不再有效。

请看左边的示例代码,输出结果是什么?

2

private static int value=1的作用域为外括号和内括号中间的部分,里面的作用域为value=2;。

动手动脑五

Java中的类型转换

看着这个图,再查查Java中每个数据类型所占的位数,和表示数值的范围,你能得出什么结论?

数据类型    位数      表示数值的范围

Int         32位        取值范围为 -2的31次方到2的31次方减1之间的任意整数(-2147483648~2147483647)

Short       16位       取值范围为 -32768~32767之间的任意整数

long        64位       取值范围为 -2的63次方到2的63次方减1之间的任意整数         (-9223372036854774808~9223372036854774807)

float        32位       取值范围为 3.402823e+38 ~ 1.401298e-45

double      64位       取值范围为 1.797693e+308~ 4.9000000e-324

char        8位       取值范围为  -128~127

7.byte        8位       取值范围为 -128~127之间的任意整数

我所得出的结论:不同的数据类型有着不同的数值表示范围,所以数据有可能超出表示范围(即数据溢出),所以我们要根据实际情况选择适当的数据类型。

动手实验六

你看到了什么样的输出,意外吗?

和我想的有点不一样。double类型的精度不是很精确,所以结果不是很准确,我们在计算时要根据实际要求选取适当的数据类型。

七、你知道原因吗?

为什么double类型的数值进行运算得不到“数学上精确”的结果?

这个涉及到二进制与十进制的转换问题。

N进制可以理解为:数值×基数的幂,例如我们熟悉的十进制数123.4=1×10²+2×10+3×(10的0次幂)+4×(10的-1次幂);其它进制的也是同理,例如二进制数11.01=1×2+1×(2的0次幂)+0+1×(2的-2次幂)=十进制的3.25。

double类型的数值占用64bit,即64个二进制数,除去最高位表示正负符号的位,在最低位上一定会与实际数据存在误差(除非实际数据恰好是2的n次方)。

举个例子来说,比如要用4bit来表示小数3.26,从高到低位依次对应2的1,0,-1,-2次幂,根据最上面的分析,应当在二进制数11.01(对应十进制的3.25)和11.10(对应十进制的3.5)之间选择。

简单来说就是我们给出的数值,在大多数情况下需要比64bit更多的位数才能准确表示出来(甚至是需要无穷多位),而double类型的数值只有64bit,后面舍去的位数一定会带来误差,无法得到“数学上精确”的结果。

八、怎样处理精度损失?

解决方法——使用BigDecimal类

Demo:TestBigDecimal.java

注意:在构建BigDecimal对象时应使用字符串而不是double数值,否则,仍有可能引发计算精度问题。(为什么会这样呢?)

double类型有它的表示范围,所以在表示时有一定的限制,必要时我们要选取string类型的字符串类型来计算更加精确的结果。

动手动脑九

以下代码的输出结果是什么?

              int X=100;

              int Y=200;

              System.out.println("X+Y="+X+Y);

              System.out.println(X+Y+"=X+Y");

System.out.println(“”+X+Y);是将X和Y连起来(int类型当成字符串相连)。

System.out.println(X+Y+””)是指X+Y后连上字符串,有一个string+int+int和int+int+string区别就是第一个是直接三个相连,而第二个是先进行int类型的相加,再进行int与string类型当成string和string类型的相连。

原文地址:https://www.cnblogs.com/1902143333-hy/p/5937480.html