- 浏览: 34923 次
- 性别:
- 来自: 大连
-
最新评论
-
qufulin:
活用ajax,如何发送异步请求我就不给你说了, 我这给你贴段使 ...
如何通过代码实现查看网页源程序 -
Komes:
zhangdaiping 写道啥也不说了,上个图给楼主看。
先 ...
如何通过代码实现查看网页源程序 -
zhangdaiping:
啥也不说了,上个图给楼主看。
如何通过代码实现查看网页源程序 -
iamsk:
做个记号,赞
到底传值还是传引用 -
yuankai:
我也说一个吧。在写Blog的时候,如果有附件,而且附件比较大的 ...
javaeye让人多少有些不爽的bug们
好东西分享
基本类型,或者叫做内置类型,是JAVA中不同于类的特殊类型。它们是我们编程中使用最频繁的类型,因此面试题中也总少不了它们的身影,在这篇文章中我们将从面试中常考的几个方面来回顾一下与基本类型相关的知识。
基本类型共有九种,它们分别都有相对应的包装类。关于它们的详细信息请看下表:
[img]http://zangweiren.iteye.com/upload/picture/pic/18450/8071c6c2-7cfb-3783-829a-a6abb3ae55e5.jpg [/img]
对于基本类型void以及它的包装类java.lang.Void,我们都无法直接进行操作。基本类型可以分为三类,字符类型char,布尔类型boolean以及数值类型byte、short、int、long、float、double。数值类型又可以分为整数类型byte、short、int、long和浮点数类型float、double。JAVA中的数值类型不存在无符号的,它们的取值范围是固定的,不会随着机器硬件环境或者操作系统的改变而改变。对于数值类型的基本类型的取值范围,我们无需强制去记忆,因为它们的值都已经以常量的形式定义在对应的包装类中了。请看下面的例子:
- public class PrimitiveTypeTest {
- public static void main(String[] args) {
- // byte
- System.out.println("基本类型:byte 二进制位数:" + Byte.SIZE);
- System.out.println("包装类:java.lang.Byte");
- System.out.println("最小值:Byte.MIN_VALUE=" + Byte.MIN_VALUE);
- System.out.println("最大值:Byte.MAX_VALUE=" + Byte.MAX_VALUE);
- System.out.println();
- // short
- System.out.println("基本类型:short 二进制位数:" + Short.SIZE);
- System.out.println("包装类:java.lang.Short");
- System.out.println("最小值:Short.MIN_VALUE=" + Short.MIN_VALUE);
- System.out.println("最大值:Short.MAX_VALUE=" + Short.MAX_VALUE);
- System.out.println();
- // int
- System.out.println("基本类型:int 二进制位数:" + Integer.SIZE);
- System.out.println("包装类:java.lang.Integer");
- System.out.println("最小值:Integer.MIN_VALUE=" + Integer.MIN_VALUE);
- System.out.println("最大值:Integer.MAX_VALUE=" + Integer.MAX_VALUE);
- System.out.println();
- // long
- System.out.println("基本类型:long 二进制位数:" + Long.SIZE);
- System.out.println("包装类:java.lang.Long");
- System.out.println("最小值:Long.MIN_VALUE=" + Long.MIN_VALUE);
- System.out.println("最大值:Long.MAX_VALUE=" + Long.MAX_VALUE);
- System.out.println();
- // float
- System.out.println("基本类型:float 二进制位数:" + Float.SIZE);
- System.out.println("包装类:java.lang.Float");
- System.out.println("最小值:Float.MIN_VALUE=" + Float.MIN_VALUE);
- System.out.println("最大值:Float.MAX_VALUE=" + Float.MAX_VALUE);
- System.out.println();
- // double
- System.out.println("基本类型:double 二进制位数:" + Double.SIZE);
- System.out.println("包装类:java.lang.Double");
- System.out.println("最小值:Double.MIN_VALUE=" + Double.MIN_VALUE);
- System.out.println("最大值:Double.MAX_VALUE=" + Double.MAX_VALUE);
- System.out.println();
- // char
- System.out.println("基本类型:char 二进制位数:" + Character.SIZE);
- System.out.println("包装类:java.lang.Character");
- // 以数值形式而不是字符形式将Character.MIN_VALUE输出到控制台
- System.out.println("最小值:Character.MIN_VALUE="
- + (int) Character.MIN_VALUE);
- // 以数值形式而不是字符形式将Character.MAX_VALUE输出到控制台
- System.out.println("最大值:Character.MAX_VALUE="
- + (int) Character.MAX_VALUE);
- }
- }
public class PrimitiveTypeTest { public static void main(String[] args) { // byte System.out.println("基本类型:byte 二进制位数:" + Byte.SIZE); System.out.println("包装类:java.lang.Byte"); System.out.println("最小值:Byte.MIN_VALUE=" + Byte.MIN_VALUE); System.out.println("最大值:Byte.MAX_VALUE=" + Byte.MAX_VALUE); System.out.println(); // short System.out.println("基本类型:short 二进制位数:" + Short.SIZE); System.out.println("包装类:java.lang.Short"); System.out.println("最小值:Short.MIN_VALUE=" + Short.MIN_VALUE); System.out.println("最大值:Short.MAX_VALUE=" + Short.MAX_VALUE); System.out.println(); // int System.out.println("基本类型:int 二进制位数:" + Integer.SIZE); System.out.println("包装类:java.lang.Integer"); System.out.println("最小值:Integer.MIN_VALUE=" + Integer.MIN_VALUE); System.out.println("最大值:Integer.MAX_VALUE=" + Integer.MAX_VALUE); System.out.println(); // long System.out.println("基本类型:long 二进制位数:" + Long.SIZE); System.out.println("包装类:java.lang.Long"); System.out.println("最小值:Long.MIN_VALUE=" + Long.MIN_VALUE); System.out.println("最大值:Long.MAX_VALUE=" + Long.MAX_VALUE); System.out.println(); // float System.out.println("基本类型:float 二进制位数:" + Float.SIZE); System.out.println("包装类:java.lang.Float"); System.out.println("最小值:Float.MIN_VALUE=" + Float.MIN_VALUE); System.out.println("最大值:Float.MAX_VALUE=" + Float.MAX_VALUE); System.out.println(); // double System.out.println("基本类型:double 二进制位数:" + Double.SIZE); System.out.println("包装类:java.lang.Double"); System.out.println("最小值:Double.MIN_VALUE=" + Double.MIN_VALUE); System.out.println("最大值:Double.MAX_VALUE=" + Double.MAX_VALUE); System.out.println(); // char System.out.println("基本类型:char 二进制位数:" + Character.SIZE); System.out.println("包装类:java.lang.Character"); // 以数值形式而不是字符形式将Character.MIN_VALUE输出到控制台 System.out.println("最小值:Character.MIN_VALUE=" + (int) Character.MIN_VALUE); // 以数值形式而不是字符形式将Character.MAX_VALUE输出到控制台 System.out.println("最大值:Character.MAX_VALUE=" + (int) Character.MAX_VALUE); } }
运行结果:
- 基本类型:byte 二进制位数:8
- 包装类:java.lang.Byte
- 最小值:Byte.MIN_VALUE=-128
- 最大值:Byte.MAX_VALUE=127
- 基本类型:short 二进制位数:16
- 包装类:java.lang.Short
- 最小值:Short.MIN_VALUE=-32768
- 最大值:Short.MAX_VALUE=32767
- 基本类型:int 二进制位数:32
- 包装类:java.lang.Integer
- 最小值:Integer.MIN_VALUE=-2147483648
- 最大值:Integer.MAX_VALUE=2147483647
- 基本类型:long 二进制位数:64
- 包装类:java.lang.Long
- 最小值:Long.MIN_VALUE=-9223372036854775808
- 最大值:Long.MAX_VALUE=9223372036854775807
- 基本类型:float 二进制位数:32
- 包装类:java.lang.Float
- 最小值:Float.MIN_VALUE=1.4E-45
- 最大值:Float.MAX_VALUE=3.4028235E38
- 基本类型:double 二进制位数:64
- 包装类:java.lang.Double
- 最小值:Double.MIN_VALUE=4.9E-324
- 最大值:Double.MAX_VALUE=1.7976931348623157E308
- 基本类型:char 二进制位数:16
- 包装类:java.lang.Character
- 最小值:Character.MIN_VALUE=0
- 最大值:Character.MAX_VALUE=65535
Float和Double的最小值和最大值都是以科学记数法的形式输出的,结尾的“E+数字”表示E之前的数字要乘以10的多少倍。比如3.14E3就是3.14×1000=3140,3.14E-3就是3.14/1000=0.00314。
大家将运行结果与上表信息仔细比较就会发现float、double两种类型的最小值与Float.MIN_VALUE、Double.MIN_VALUE的值并不相同,这是为什么呢?实际上Float.MIN_VALUE和Double.MIN_VALUE分别指的是float和double类型所能表示的最小正数。也就是说存在这样一种情况,0到±Float.MIN_VALUE之间的值float类型无法表示,0到±Double.MIN_VALUE之间的值double类型无法表示。这并没有什么好奇怪的,因为这些范围内的数值超出了它们的精度范围。
基本类型存储在栈中,因此它们的存取速度要快于存储在堆中的对应包装类的实例对象。从Java5.0(1.5)开始,JAVA虚拟机(Java Virtual Machine)可以完成基本类型和它们对应包装类之间的自动转换。因此我们在赋值、参数传递以及数学运算的时候像使用基本类型一样使用它们的包装类,但这并不意味着你可以通过基本类型调用它们的包装类才具有的方法。另外,所有基本类型(包括void)的包装类都使用了final修饰,因此我们无法继承它们扩展新的类,也无法重写它们的任何方法。
各种数值类型之间的赋值与转换遵循什么规律呢?我们来看下面这个例子:
- public class PrimitiveTypeTest {
- public static void main(String[] args) {
- // 给byte类型变量赋值时,数字后无需后缀标识
- byte byte_a = 1;
- // 编译器会做范围检查,如果赋予的值超出了范围就会报错
- // byte byte_b = 1000;
- // 把一个long型值赋值给byte型变量,编译时会报错,即使这个值没有超出byte类型的取值范围
- // byte byte_c = 1L;
- // 给short类型变量赋值时,数字后无需后缀标识
- short short_a = 1;
- // 编译器会做范围检查,如果赋予的值超出了范围就会报错
- // short short_b = 70000;
- // 把一个long型值赋值给short型变量,编译时会报错,即使这个值没有超出short类型的取值范围
- // byte short_c = 1L;
- // 给short类型变量赋值时,数字后无需后缀标识
- int int_a = 1;
- // 编译器会做范围检查,如果赋予的值超出了范围就会报错
- // int int_b = 2200000000;
- // 把一个long型值赋值给int型变量,编译时会报错,即使这个值没有超出int类型的取值范围
- // int int_c = 1L;
- // 可以把一个int型值直接赋值给long型变量,数字后无需后缀标识
- long long_a = 1;
- // 如果给long型变量赋予的值超出了int型值的范围,数字后必须加L(不区分大小写)标识
- long long_b = 2200000000L;
- // 编译器会做范围检查,如果赋予的值超出了范围就会报错
- // long long_c = 9300000000000000000L;
- // 可以把一个int型值直接赋值给float型变量
- float float_a = 1;
- // 可以把一个long型值直接赋值给float型变量
- float float_b = 1L;
- // 没有F(不区分大小写)后缀标识的浮点数默认为double型的,不能将它直接赋值给float型变量
- // float float_c = 1.0;
- // float型数值需要有一个F(不区分大小写)后缀标识
- float float_d = 1.0F;
- // 把一个double型值赋值给float型变量,编译时会报错,即使这个值没有超出float类型的取值范围
- // float float_e = 1.0D;
- // 编译器会做范围检查,如果赋予的值超出了范围就会报错
- // float float_f = 3.5000000E38F;
- // 可以把一个int型值直接赋值给double型变量
- double double_a = 1;
- // 可以把一个long型值直接赋值给double型变量
- double double_b = 1L;
- // 可以把一个float型值直接赋值给double型变量
- double double_c = 1F;
- // 不带后缀标识的浮点数默认为double类型的,可以直接赋值
- double double_d = 1.0;
- // 也可以给数字增加一个D(不区分大小写)后缀标识,明确标出它是double类型的
- double double_e = 1.0D;
- // 编译器会做范围检查,如果赋予的值超出了范围就会报错
- // double double_f = 1.8000000000000000E308D;
- // 把一个double型值赋值给一个byte类型变量,编译时会报错,即使这个值没有超出byte类型的取值范围
- // byte byte_d = 1.0D;
- // 把一个double型值赋值给一个short类型变量,编译时会报错,即使这个值没有超出short类型的取值范围
- // short short_d = 1.0D;
- // 把一个double型值赋值给一个int类型变量,编译时会报错,即使这个值没有超出int类型的取值范围
- // int int_d = 1.0D;
- // 把一个double型值赋值给一个long类型变量,编译时会报错,即使这个值没有超出long类型的取值范围
- // long long_d = 1.0D;
- // 可以用字符初始化一个char型变量
- char char_a = 'a';
- // 也可以用一个int型数值初始化char型变量
- char char_b = 1;
- // 把一个long型值赋值给一个char类型变量,编译时会报错,即使这个值没有超出char类型的取值范围
- // char char_c = 1L;
- // 把一个float型值赋值给一个char类型变量,编译时会报错,即使这个值没有超出char类型的取值范围
- // char char_d = 1.0F;
- // 把一个double型值赋值给一个char类型变量,编译时会报错,即使这个值没有超出char类型的取值范围
- // char char_e = 1.0D;
- // 编译器会做范围检查,如果赋予的值超出了范围就会报错
- // char char_f = 70000;
- }
- }
public class PrimitiveTypeTest { public static void main(String[] args) { // 给byte类型变量赋值时,数字后无需后缀标识 byte byte_a = 1; // 编译器会做范围检查,如果赋予的值超出了范围就会报错 // byte byte_b = 1000; // 把一个long型值赋值给byte型变量,编译时会报错,即使这个值没有超出byte类型的取值范围 // byte byte_c = 1L; // 给short类型变量赋值时,数字后无需后缀标识 short short_a = 1; // 编译器会做范围检查,如果赋予的值超出了范围就会报错 // short short_b = 70000; // 把一个long型值赋值给short型变量,编译时会报错,即使这个值没有超出short类型的取值范围 // byte short_c = 1L; // 给short类型变量赋值时,数字后无需后缀标识 int int_a = 1; // 编译器会做范围检查,如果赋予的值超出了范围就会报错 // int int_b = 2200000000; // 把一个long型值赋值给int型变量,编译时会报错,即使这个值没有超出int类型的取值范围 // int int_c = 1L; // 可以把一个int型值直接赋值给long型变量,数字后无需后缀标识 long long_a = 1; // 如果给long型变量赋予的值超出了int型值的范围,数字后必须加L(不区分大小写)标识 long long_b = 2200000000L; // 编译器会做范围检查,如果赋予的值超出了范围就会报错 // long long_c = 9300000000000000000L; // 可以把一个int型值直接赋值给float型变量 float float_a = 1; // 可以把一个long型值直接赋值给float型变量 float float_b = 1L; // 没有F(不区分大小写)后缀标识的浮点数默认为double型的,不能将它直接赋值给float型变量 // float float_c = 1.0; // float型数值需要有一个F(不区分大小写)后缀标识 float float_d = 1.0F; // 把一个double型值赋值给float型变量,编译时会报错,即使这个值没有超出float类型的取值范围 // float float_e = 1.0D; // 编译器会做范围检查,如果赋予的值超出了范围就会报错 // float float_f = 3.5000000E38F; // 可以把一个int型值直接赋值给double型变量 double double_a = 1; // 可以把一个long型值直接赋值给double型变量 double double_b = 1L; // 可以把一个float型值直接赋值给double型变量 double double_c = 1F; // 不带后缀标识的浮点数默认为double类型的,可以直接赋值 double double_d = 1.0; // 也可以给数字增加一个D(不区分大小写)后缀标识,明确标出它是double类型的 double double_e = 1.0D; // 编译器会做范围检查,如果赋予的值超出了范围就会报错 // double double_f = 1.8000000000000000E308D; // 把一个double型值赋值给一个byte类型变量,编译时会报错,即使这个值没有超出byte类型的取值范围 // byte byte_d = 1.0D; // 把一个double型值赋值给一个short类型变量,编译时会报错,即使这个值没有超出short类型的取值范围 // short short_d = 1.0D; // 把一个double型值赋值给一个int类型变量,编译时会报错,即使这个值没有超出int类型的取值范围 // int int_d = 1.0D; // 把一个double型值赋值给一个long类型变量,编译时会报错,即使这个值没有超出long类型的取值范围 // long long_d = 1.0D; // 可以用字符初始化一个char型变量 char char_a = 'a'; // 也可以用一个int型数值初始化char型变量 char char_b = 1; // 把一个long型值赋值给一个char类型变量,编译时会报错,即使这个值没有超出char类型的取值范围 // char char_c = 1L; // 把一个float型值赋值给一个char类型变量,编译时会报错,即使这个值没有超出char类型的取值范围 // char char_d = 1.0F; // 把一个double型值赋值给一个char类型变量,编译时会报错,即使这个值没有超出char类型的取值范围 // char char_e = 1.0D; // 编译器会做范围检查,如果赋予的值超出了范围就会报错 // char char_f = 70000; } }
从上面的例子中我们可以得出如下几条结论:
- 未带有字符后缀标识的整数默认为int类型;未带有字符后缀标识的浮点数默认为double类型。
- 如果一个整数的值超出了int类型能够表示的范围,则必须增加后缀“L”(不区分大小写,建议用大写,因为小写的L与阿拉伯数字1很容易混淆),表示为long型。
- 带有“F”(不区分大小写)后缀的整数和浮点数都是float类型的;带有“D”(不区分大小写)后缀的整数和浮点数都是double类型的。
- 编译器会在编译期对byte、short、int、long、float、double、char型变量的值进行检查,如果超出了它们的取值范围就会报错。
- int型值可以赋给所有数值类型的变量;long型值可以赋给long、float、double类型的变量;float型值可以赋给float、double类型的变量;double型值只能赋给double类型变量。
下图显示了几种基本类型之间的默认逻辑转换关系:
图中的实线表示无精度损失的转换,而虚线则表示这样的转换可能会损失一定的精度。如果我们想把一个能表示更大范围或者更高精度的类型,转换为一个范围更小或者精度更低的类型时,就需要使用强制类型转换(Cast)了。不过我们要尽量避免这种用法,因为它常常引发错误。请看下面的例子,如果不运行代码,你能预测它的结果吗?
- public class PrimitiveTypeTest {
- public static void main(String[] args) {
- int a = 123456;
- short b = (short) a;
- // b的值会是什么呢?
- System.out.println(b);
- }
- }
public class PrimitiveTypeTest { public static void main(String[] args) { int a = 123456; short b = (short) a; // b的值会是什么呢? System.out.println(b); } }
运行结果:
- -7616
运算符对基本类型的影响
当使用+、-、*、/、%运算符对基本类型进行运算时,遵循如下规则:
- 只要两个操作数中有一个是double类型的,另一个将会被转换成double类型,并且结果也是double类型;
- 否则,只要两个操作数中有一个是float类型的,另一个将会被转换成float类型,并且结果也是float类型;
- 否则,只要两个操作数中有一个是long类型的,另一个将会被转换成long类型,并且结果也是long类型;
- 否则,两个操作数(包括byte、short、int、char)都将会被转换成int类型,并且结果也是int类型。
当使用+=、-=、*=、/=、%=、运算符对基本类型进行运算时,遵循如下规则:
- 运算符右边的数值将首先被强制转换成与运算符左边数值相同的类型,然后再执行运算,且运算结果与运算符左边数值类型相同。
了解了这些,我们就能解答下面这个常考的面试题了。请看:
乍一看,觉得它们都应该没有错误,可以正常运行。我们来写个例子试试:
- public class PrimitiveTypeTest {
- public static void main(String[] args) {
- short s1 = 1;
- // 这一行代码会报编译错误
- // s1 = s1 + 1;
- // 这一行代码没有报错
- s1 = 1 + 1;
- // 这一行代码也没有报错
- s1 += 1;
- }
- }
public class PrimitiveTypeTest { public static void main(String[] args) { short s1 = 1; // 这一行代码会报编译错误 // s1 = s1 + 1; // 这一行代码没有报错 s1 = 1 + 1; // 这一行代码也没有报错 s1 += 1; } }
从例子中我们可以看出结果了。利用上面列举的规律,也很容易解释。在s1=s1+1;中,s1+1运算的结果是int型,把它赋值给一个short型变量s1,所以会报错;而在s1+=1;中,由于是s1是short类型的,所以1首先被强制转换为short型,然后再参与运算,并且结果也是short类型的,因此不会报错。那么,s1=1+1;为什么不报错呢?这是因为1+1是个编译时可以确定的常量,“+”运算在编译时就被执行了,而不是在程序执行的时候,这个语句的效果等同于s1=2,所以不会报错。前面讲过了,对基本类型执行强制类型转换可能得出错误的结果,因此在使用+=、-=、*=、/=、%=等运算符时,要多加注意。
当使用“==”运算符在基本类型和其包装类对象之间比较时,遵循如下规则:
- 只要两个操作数中有一个是基本类型,就是比较它们的数值是否相等。
- 否则,就是判断这两个对象的内存地址是否相等,即是否是同一个对象。
下面的测试例子则验证了以上的规则:
- public class EqualsTest {
- public static void main(String[] args) {
- // int类型用int类型初始化
- int int_int = 0;
- // int类型用Integer类型初始化
- int int_Integer = new Integer(0);
- // Integer类型用Integer类型初始化
- Integer Integer_Integer = new Integer(0);
- // Integer类型用int类型初始化
- Integer Integer_int = 0;
- System.out.println("int_int == int_Integer结果是:"
- + (int_int == int_Integer));
- System.out.println("Integer_Integer == Integer_int结果是:"
- + (Integer_Integer == Integer_int));
- System.out.println();
- System.out.println("int_int == Integer_Integer结果是:"
- + (int_int == Integer_Integer));
- System.out.println("Integer_Integer == int_int结果是:"
- + (Integer_Integer == int_int));
- System.out.println();
- // boolean类型用boolean类型初始化
- boolean boolean_boolean = true;
- // boolean类型用Boolean类型初始化
- boolean boolean_Boolean = new Boolean(true);
- // Boolean类型用Boolean类型初始化
- Boolean Boolean_Boolean = new Boolean(true);
- // Boolean类型用boolean类型初始化
- Boolean Boolean_boolean = true;
- System.out.println("boolean_boolean == boolean_Boolean结果是:"
- + (boolean_boolean == boolean_Boolean));
- System.out.println("Boolean_Boolean == Boolean_boolean结果是:"
- + (Boolean_Boolean == Boolean_boolean));
- System.out.println();
- System.out.println("boolean_boolean == Boolean_Boolean结果是:"
- + (boolean_boolean == Boolean_Boolean));
- System.out.println("Boolean_Boolean == boolean_boolean结果是:"
- + (Boolean_Boolean == boolean_boolean));
- }
- }
public class EqualsTest { public static void main(String[] args) { // int类型用int类型初始化 int int_int = 0; // int类型用Integer类型初始化 int int_Integer = new Integer(0); // Integer类型用Integer类型初始化 Integer Integer_Integer = new Integer(0); // Integer类型用int类型初始化 Integer Integer_int = 0; System.out.println("int_int == int_Integer结果是:" + (int_int == int_Integer)); System.out.println("Integer_Integer == Integer_int结果是:" + (Integer_Integer == Integer_int)); System.out.println(); System.out.println("int_int == Integer_Integer结果是:" + (int_int == Integer_Integer)); System.out.println("Integer_Integer == int_int结果是:" + (Integer_Integer == int_int)); System.out.println(); // boolean类型用boolean类型初始化 boolean boolean_boolean = true; // boolean类型用Boolean类型初始化 boolean boolean_Boolean = new Boolean(true); // Boolean类型用Boolean类型初始化 Boolean Boolean_Boolean = new Boolean(true); // Boolean类型用boolean类型初始化 Boolean Boolean_boolean = true; System.out.println("boolean_boolean == boolean_Boolean结果是:" + (boolean_boolean == boolean_Boolean)); System.out.println("Boolean_Boolean == Boolean_boolean结果是:" + (Boolean_Boolean == Boolean_boolean)); System.out.println(); System.out.println("boolean_boolean == Boolean_Boolean结果是:" + (boolean_boolean == Boolean_Boolean)); System.out.println("Boolean_Boolean == boolean_boolean结果是:" + (Boolean_Boolean == boolean_boolean)); } }
运行结果:
- int_int == int_Integer结果是:true
- Integer_Integer == Integer_int结果是:false
- int_int == Integer_Integer结果是:true
- Integer_Integer == int_int结果是:true
- boolean_boolean == boolean_Boolean结果是:true
- Boolean_Boolean == Boolean_boolean结果是:false
- boolean_boolean == Boolean_Boolean结果是:true
- Boolean_Boolean == boolean_boolean结果是:true
为了便于查看,上例中变量命名没有采用规范的方式,而是采用了“变量类型”+“_”+“初始化值类型”的方式。
Math.round()方法
java.lang.Math类里有两个round()方法,它们的定义如下:
- public static int round(float a) {
- //other code
- }
- public static long round(double a) {
- //other code
- }
public static int round(float a) { //other code } public static long round(double a) { //other code }
它们的返回值都是整数,且都采用四舍五入法。运算规则如下:
- 如果参数为正数,且小数点后第一位>=5,运算结果为参数的整数部分+1。
- 如果参数为负数,且小数点后第一位>5,运算结果为参数的整数部分-1。
- 如果参数为正数,且小数点后第一位<5;或者参数为负数,且小数点后第一位<=5,运算结果为参数的整数部分。
我们可以通过下面的例子来验证:
- public class MathTest {
- public static void main(String[] args) {
- System.out.println("小数点后第一位=5");
- System.out.println("正数:Math.round(11.5)=" + Math.round(11.5));
- System.out.println("负数:Math.round(-11.5)=" + Math.round(-11.5));
- System.out.println();
- System.out.println("小数点后第一位<5");
- System.out.println("正数:Math.round(11.46)=" + Math.round(11.46));
- System.out.println("负数:Math.round(-11.46)=" + Math.round(-11.46));
- System.out.println();
- System.out.println("小数点后第一位>5");
- System.out.println("正数:Math.round(11.68)=" + Math.round(11.68));
- System.out.println("负数:Math.round(-11.68)=" + Math.round(-11.68));
- }
- }
public class MathTest { public static void main(String[] args) { System.out.println("小数点后第一位=5"); System.out.println("正数:Math.round(11.5)=" + Math.round(11.5)); System.out.println("负数:Math.round(-11.5)=" + Math.round(-11.5)); System.out.println(); System.out.println("小数点后第一位<5"); System.out.println("正数:Math.round(11.46)=" + Math.round(11.46)); System.out.println("负数:Math.round(-11.46)=" + Math.round(-11.46)); System.out.println(); System.out.println("小数点后第一位>5"); System.out.println("正数:Math.round(11.68)=" + Math.round(11.68)); System.out.println("负数:Math.round(-11.68)=" + Math.round(-11.68)); } }
运行结果:
- 小数点后第一位=5
- 正数:Math.round(11.5)=12
- 负数:Math.round(-11.5)=-11
- 小数点后第一位<5
- 正数:Math.round(11.46)=11
- 负数:Math.round(-11.46)=-11
- 小数点后第一位>5
- 正数:Math.round(11.68)=12
- 负数:Math.round(-11.68)=-12
根据上面例子的运行结果,我们还可以按照如下方式总结,或许更加容易记忆:
- 参数的小数点后第一位<5,运算结果为参数整数部分。
- 参数的小数点后第一位>5,运算结果为参数整数部分绝对值+1,符号(即正负)不变。
- 参数的小数点后第一位=5,正数运算结果为整数部分+1,负数运算结果为整数部分。
但是上面的结论仍然不是很好记忆。我们来看看round()方法的内部实现会给我们带来什么启发?我们来看这两个方法内部的代码:
- public static int round(float a) {
- return (int)floor(a + 0.5f);
- }
- public static long round(double a) {
- return (long)floor(a + 0.5d);
- }
public static int round(float a) { return (int)floor(a + 0.5f); } public static long round(double a) { return (long)floor(a + 0.5d); }
看来它们都是将参数值+0.5后交与floor()进行运算,然后取返回值。那么floor()方法的作用又是什么呢?它是取一个小于等于参数值的最大整数。比如经过floor()方法运算后,如果参数是10.2则返回10,13返回13,-20.82返回-21,-16返回-16等等。既然是这样,我们就可以用一句话来概括round()方法的运算效果了:
- Math类的round()方法的运算结果是一个<=(参数值+0.5)的最大整数。
switch语句
哪些类型可以用于switch语句的判断呢?我们做个测试就知道了:
- public class MathTest {
- // 枚举类型,Java5.0以上版本可用
- static enum enum_e {
- A, B
- }
- public static void main(String[] args) {
- // byte
- byte byte_n = 0;
- switch (byte_n) {
- case 0:
- System.out.println("byte可以用于switch语句");
-
</sp
发表评论
-
dom4j操作xml基础
2010-09-15 21:38 908DOM是用与平台和语言无关的方式表示XML文档的官方W3C标准 ... -
手工搭建SSH集成工程
2010-09-15 21:23 860//搭建SSH集成工程://1.建立普通web工程,导入SSH ... -
tomcat+eclipse工程发布的两种方式
2010-09-15 20:52 2154tomcat+eclipse工程发布的 ... -
优秀的Java程序员必须了解GC的工作原理(转)
2010-05-05 20:14 1040一个优秀的Java程序 ... -
详细解析Java中抽象类和接口的区别(转)
2010-05-05 20:11 689在Java语言中, abs ... -
Vector、ArrayList和List的异同(转)
2010-05-05 20:07 1101线性表,链表,哈希表 ... -
JAVA面试题解惑系列(十一)——这些运算符你是否还记得?
2010-05-01 00:13 694好东西分享 有些运算符在JAVA语言中存在着,但是在实际开发 ... -
JAVA面试题解惑系列(十)——话说多线程
2010-05-01 00:11 751好东西分享 线程或者说多线程,是我们处理多任务的强大工具。线 ... -
JAVA面试题解惑系列(九)——继承、多态、重载和重写
2010-05-01 00:09 727好东西分享 什么是多 ... -
JAVA面试题解惑系列(七)——日期和时间的处理
2010-05-01 00:06 809好东西分享 日期和时间的处理不仅在面试题中会考到,在实际项目 ... -
JAVA面试题解惑系列(六)——字符串(String)杂谈
2010-05-01 00:04 718好东西分享上一次我们已经一起回顾了面试题中常考的到底创建了几个 ... -
JAVA面试题解惑系列(五)——传了值还是传了引用?
2010-05-01 00:02 724好东西分享 JAVA中的传递都是值传递吗?有没有引用传递呢? ... -
JAVA面试题解惑系列(四)——final、finally和finalize的区别
2010-05-01 00:00 688好东西分享final、finally和finalize的区别是 ... -
JAVA面试题解惑系列(三)——变量(属性)的覆盖
2010-04-30 23:58 586好东西分享我们来看看这么一道题: Java代码 ... -
JAVA面试题解惑系列(二)——到底创建了几个String对象?
2010-04-30 23:56 758好东西分享 我们首先来看一段代码: Java代码 ... -
JAVA面试题解惑系列(一)——类的初始化顺序
2010-04-30 23:54 694好东西分享大家在去参加面试的时候,经常会遇到这样的考题:给你两 ... -
关于import org.apache.xpath.XPathAPI;
2009-06-23 12:31 4034import org.apache.xpath.XPathAP ... -
到底传值还是传引用
2009-06-10 10:32 1019不少同学总是质疑java编程中参数的传递,到底是传 ... -
自学Java 之 组合键
2009-06-09 10:10 1598今天练习了一个小的聊天程序,实现客户端与服务器端相 ... -
Socket 异常 Connection refused 解决办法
2009-06-02 17:57 5702今天练习Socket编程,在new Socket时无意 ...
相关推荐
.46 1.8 JAVA面试题解惑系列(八)——聊聊基本类型(内置类型) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .63 1.9 JAVA面试题解惑系列(九)——继承、多态、重载和重写 . . . . . . . . . . ....
**1.8 JAVA面试题解惑系列(八)——聊聊基本类型(内置类型)** - **知识点**:讨论Java的基本数据类型,包括整型、浮点型、字符型、布尔型等,以及它们的包装类和自动装箱/拆箱机制。 **1.9 JAVA面试题解惑系列...
“JAVA面试题解惑系列(八)——聊聊基本类型(内置类型)”涵盖了Java的原始数据类型,包括int、char、boolean等,它们与引用类型的不同之处以及它们的默认值和范围。 “JAVA面试题解惑系列(九)——继承、多态、...
### Java面试题解惑系列——知识点详解 #### 1. 类的初始化顺序 在Java中,类的初始化顺序是一个常见的面试考察点。当一个类被加载时,Java虚拟机(JVM)会按照一定的顺序执行类中的不同部分,包括静态变量、静态...
1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。
8c71b76fb2ec10cf50fc6b0308d3dcfc_9545878e2b97a84b2e089ece58da9e82
Android逆向过程学习
内容概要:本文详细介绍了基于西门子S7-200 PLC的糖果包装控制系统的设计与实现。首先阐述了PLC在工业自动化领域的优势及其在糖果包装生产线中的重要性。接着深入探讨了系统的硬件连接方式,包括传感器、执行机构与PLC的具体接口配置。随后展示了关键的编程实现部分,如糖果计数、包装执行、送膜控制、称重判断以及热封温度控制等具体梯形图代码片段。此外,还分享了一些实用的经验技巧,如防止信号抖动、PID参数优化、故障诊断方法等。最后总结了该系统的优势,强调其对提高生产效率和产品质量的重要作用。 适合人群:从事工业自动化控制、PLC编程的技术人员,尤其是对小型PLC系统感兴趣的工程师。 使用场景及目标:适用于糖果制造企业,旨在提升包装生产线的自动化程度,确保高效稳定的生产过程,同时降低维护成本并提高产品一致性。 其他说明:文中不仅提供了详细的理论讲解和技术指导,还结合实际案例进行了经验分享,有助于读者更好地理解和掌握相关知识。
内容概要:本文详细介绍了参与西门子杯比赛中关于三部十层电梯系统的博图V15.1程序设计及其WinCC画面展示的内容。文中不仅展示了电梯系统的基本架构,如抢单逻辑、方向决策、状态机管理等核心算法(采用SCL语言编写),还分享了许多实际调试过程中遇到的问题及解决方案,例如未初始化变量导致的异常行为、状态机遗漏空闲状态、WinCC画面动态显示的挑战以及通信配置中的ASCII码解析错误等问题。此外,作者还特别提到一些创意性的设计,如电梯同时到达同一层时楼层显示器变为闪烁爱心的效果,以及节能模式下电梯自动停靠中间楼层的功能。 适合人群:对PLC编程、工业自动化控制、电梯调度算法感兴趣的工程技术人员,尤其是准备参加类似竞赛的学生和技术爱好者。 使用场景及目标:适用于希望深入了解PLC编程实践、掌握电梯群控系统的设计思路和技术要点的人士。通过学习本文可以更好地理解如何利用PLC进行复杂的机电一体化项目的开发,提高解决实际问题的能力。 其他说明:文章风格幽默诙谐,将严肃的技术话题融入轻松的生活化比喻之中,使得原本枯燥的专业知识变得生动有趣。同时,文中提供的经验教训对于从事相关领域的工作者来说非常宝贵,能够帮助他们少走弯路并激发更多创新思维。
慧荣量产工具合集.zip
内容概要:本文详细介绍了永磁同步电机(PMSM)的FOC(磁场定向控制)和SVPWM(空间矢量脉宽调制)算法的仿真模型。首先解释了FOC的基本原理及其核心的坐标变换(Clark变换和Park变换),并给出了相应的Python代码实现。接下来探讨了SVPWM算法的工作机制,包括扇区判断和占空比计算的方法。此外,文章还讨论了电机的PI双闭环控制结构,即速度环和电流环的设计与实现。文中不仅提供了详细的理论背景,还分享了一些实用的编程技巧和注意事项,帮助读者更好地理解和应用这些算法。 适合人群:电气工程专业学生、从事电机控制系统开发的技术人员以及对永磁同步电机控制感兴趣的科研人员。 使用场景及目标:① 学习和掌握永磁同步电机的FOC控制和SVPWM算法的具体实现;② 提供丰富的代码示例和实践经验,便于快速搭建和调试仿真模型;③ 探讨不同参数设置对电机性能的影响,提高系统的稳定性和效率。 其他说明:文章强调了在实际应用中需要注意的一些细节问题,如坐标变换中的系数选择、SVPWM算法中的扇区判断优化以及PI控制器的参数调整等。同时,鼓励读者通过动手实验来加深对各个模块的理解。
# 压缩文件中包含: 中文文档 jar包下载地址 Maven依赖 Gradle依赖 源代码下载地址 # 本文件关键字: jar中文文档.zip,java,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,中文API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压最外层zip,再解压其中的zip包,双击 【index.html】 文件,即可用浏览器打开、进行查看。 # 特殊说明: ·本文档为人性化翻译,精心制作,请放心使用。 ·只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; ·不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 # 温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件;
Android逆向过程学习
1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。
3dmax插件
# 【spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7.jar中文文档.zip】 中包含: 中文文档:【spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7-javadoc-API文档-中文(简体)版.zip】 jar包下载地址:【spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7.jar下载地址(官方地址+国内镜像地址).txt】 Maven依赖:【spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7.jar Maven依赖信息(可用于项目pom.xml).txt】 Gradle依赖:【spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7.jar Gradle依赖信息(可用于项目build.gradle).txt】 源代码下载地址:【spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7-sources.jar下载地址(官方地址+国内镜像地址).txt】 # 本文件关键字: spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7.jar中文文档.zip,java,spring-ai-autoconfigure-vector-store-qdrant-1.0.0-M7.jar,org.springframework.ai,spring-ai-autoconfigure-vector-store-qdrant,1.0.0-M7,org.springframework.ai.vectorstore.qdr
1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。
内容概要:本文详细介绍了平方根容积卡尔曼滤波(SRCKF)在永磁同步电机(PMSM)控制系统中的应用及其相对于传统CKF的优势。文章首先指出传统CKF在处理协方差矩阵时存在的数值不稳定性和非正定问题,导致系统性能下降。接着,作者通过引入SRCKF,利用Cholesky分解和QR分解来确保协方差矩阵的正定性,从而提高状态估计的精度和稳定性。文中展示了具体的电机模型和状态方程,并提供了详细的代码实现,包括状态预测、容积点生成以及观测更新等关键步骤。此外,文章还分享了实际调试过程中遇到的问题及解决方案,如选择合适的矩阵分解库和处理电机参数敏感性。最终,通过实验数据对比,证明了SRCKF在突加负载情况下的优越表现。 适合人群:从事永磁同步电机控制研究的技术人员、研究生及以上学历的研究者。 使用场景及目标:适用于需要高精度状态估计的永磁同步电机控制系统的设计与优化,特别是在处理非线性问题和提高数值稳定性方面。 其他说明:文章引用了相关领域的权威文献,如Arasaratnam的TAC论文和Zhong的《PMSM无传感器控制综述》,并强调了实际工程实践中代码调试的重要性。
# 【tokenizers-***.jar***文档.zip】 中包含: ***文档:【tokenizers-***-javadoc-API文档-中文(简体)版.zip】 jar包下载地址:【tokenizers-***.jar下载地址(官方地址+国内镜像地址).txt】 Maven依赖:【tokenizers-***.jar Maven依赖信息(可用于项目pom.xml).txt】 Gradle依赖:【tokenizers-***.jar Gradle依赖信息(可用于项目build.gradle).txt】 源代码下载地址:【tokenizers-***-sources.jar下载地址(官方地址+国内镜像地址).txt】 # 本文件关键字: tokenizers-***.jar***文档.zip,java,tokenizers-***.jar,ai.djl.huggingface,tokenizers,***,ai.djl.engine.rust,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,djl,huggingface,中文API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压 【tokenizers-***.jar***文档.zip】,再解压其中的 【tokenizers-***-javadoc-API文档-中文(简体)版.zip】,双击 【index.html】 文件,即可用浏览器打开、进行查看。 # 特殊说明: ·本文档为人性化翻译,精心制作,请放心使用。 ·只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; ·不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 # 温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件; # Maven依赖: ``` <dependency> <groupId>ai.djl.huggingface</groupId> <artifactId>tokenizers</artifactId> <version>***</version> </dependency> ``` # Gradle依赖: ``` Gradle: implementation group: 'ai.djl.huggingface', name: 'tokenizers', version: '***' Gradle (Short): implementation 'ai.djl.huggingface:tokenizers:***' Gradle (Kotlin): implementation("ai.djl.huggingface:tokenizers:***") ``` # 含有的 Java package(包): ``` ai.djl.engine.rust ai.djl.engine.rust.zoo ai.djl.huggingface.tokenizers ai.djl.huggingface.tokenizers.jni ai.djl.huggingface.translator ai.djl.huggingface.zoo ``` # 含有的 Java class(类): ``` ai.djl.engine.rust.RsEngine ai.djl.engine.rust.RsEngineProvider ai.djl.engine.rust.RsModel ai.djl.engine.rust.RsNDArray ai.djl.engine.rust.RsNDArrayEx ai.djl.engine.rust.RsNDArrayIndexer ai.djl.engine.rust.RsNDManager ai.djl.engine.rust.RsSymbolBlock ai.djl.engine.rust.RustLibrary ai.djl.engine.rust.zoo.RsModelZoo ai.djl.engine.rust.zoo.RsZooProvider ai.djl.huggingface.tokenizers.Encoding ai.djl.huggingface.tokenizers.HuggingFaceTokenizer ai.djl.huggingface.tokenizers.HuggingFaceTokenizer.Builder ai.djl.hu
3