原文:http://zetcode.com/lang/java/datatypes/

在 Java 教程的这一部分中,我们将讨论数据类型。

计算机程序(包括电子表格,文本编辑器,计算器或聊天客户端)可以处理数据。 用于各种数据类型的工具是现代计算机语言的基本组成部分。 数据类型是一组值以及对这些值的允许操作。

Java 编程语言是一种静态类型的语言。 这意味着每个变量和每个表达式都具有在编译时已知的类型。 Java 语言也是一种强类型语言,因为类型会限制变量可以容纳的值或表达式可以产生的值,限制对这些值支持的操作,并确定操作的含义。 强大的静态类型有助于在编译时检测错误。 动态类型语言(例如 Ruby 或 Python)中的变量会随着时间的推移接收不同的数据类型。 在 Java 中,一旦变量声明为某种数据类型,就无法保存其他数据类型的值。

Java 中有两种基本数据类型:基本类型和引用类型。 基本类型为:

  • boolean
  • char
  • byte
  • short
  • int
  • long
  • float
  • double

Java 中每种类型都有一个特定的关键字。 基本类型不是 Java 中的对象。 原始数据类型不能存储在仅适用于对象的 Java 集合中。 可以将它们放置在数组中。

引用类型为:

  • 类类型
  • 接口类型
  • 数组类型

还有一个特殊的null类型,它代表不存在的值。

在 Ruby 编程语言中,一切都是对象。 甚至是基本数据类型。

  1. #!/usr/bin/ruby
  2. 4.times { puts "Ruby" }

这个 Ruby 脚本会在控制台上打印四次"Ruby"字符串。 我们在 4 号上调用次方法。 该数字是 Ruby 中的对象。

Java 有不同的方法。 它具有原始数据类型和包装器类。 包装器类将原始类型转换为对象。 包装器类将在下一章中介绍。

布尔值

我们的世界建立了双重性。 有天与地,水与火,阴与阳,男人与女人,爱与恨。 在 Java 中,boolean数据类型是具有以下两个值之一的原始数据类型:truefalse

快乐的父母正在等待孩子的出生。 他们为两种可能性都选择了名称。 如果要成为男孩,他们选择了罗伯特。 如果要成为女孩,他们会选择维多利亚。

com/zetcode/BooleanType.java

  1. package com.zetcode;
  2. import java.util.Random;
  3. public class BooleanType {
  4. public static void main(String[] args) {
  5. String name = "";
  6. Random r = new Random();
  7. boolean male = r.nextBoolean();
  8. if (male == true) {
  9. name = "Robert";
  10. }
  11. if (male == false) {
  12. name = "Victoria";
  13. }
  14. System.out.format("We will use name %s%n", name);
  15. System.out.println(9 > 8);
  16. }
  17. }

该程序使用随机数生成器来模拟我们的情况。

  1. Random r = new Random();
  2. boolean male = r.nextBoolean();

Random类用于产生随机数。 nextBoolean()方法随机返回一个布尔值。

  1. if (male == true) {
  2. name = "Robert";
  3. }

如果布尔变量male等于true,则将名称变量设置为"Robert"if关键字适用于布尔值。

  1. if (male == false) {
  2. name = "Victoria";
  3. }

如果随机生成器选择false,则将名称变量设置为"Victoria"

  1. System.out.println(9 > 8);

关系运算符导致布尔值。 此行在控制台上显示为true

  1. $ java BooleanType.java
  2. We will use name Robert
  3. true
  4. $ java BooleanType.java
  5. We will use name Robert
  6. true
  7. $ java BooleanType.java
  8. We will use name Victoria
  9. true

多次运行该程序。

整数

整数是实数的子集。 它们写时没有小数或小数部分。 整数落入集合Z = {..., -2, -1, 0, 1, 2, ......}中整数是无限的。

在计算机语言中,整数(通常)是原始数据类型。 实际上,计算机只能使用整数值的子集,因为计算机的容量有限。 整数用于计算离散实体。 我们可以有 3、4 或 6 个人,但不能有 3.33 个人。 我们可以拥有 3.33 公斤,4.564 天或 0.4532 公里。

类型 大小 范围
byte 8 位 -128 至 127
short 16 位 -32,768 至 32,767
char 16 位 0 至 65,535
int 32 位 -2,147,483,648 至 2,147,483,647
long 64 位 -9,223,372,036,854,775,808 至 9,223,372,036,854,775,807

Table: Integer types in Java

可以根据我们的需要使用这些整数类型。 然后,我们可以将byte类型用于存储妇女生育的孩子数量的变量。 经过验证的年龄最大的人死于 122 岁,因此我们可能至少选择short类型作为年龄变量。 这将为我们节省一些内存。

整数字面值可以用十进制,十六进制,八进制或二进制表示法表示。 如果数字后缀为 ASCII 字母Ll,则其类型为long。 否则为int类型。 大写字母L首选用于指定长数字,因为小写字母l容易与数字 1 混淆。

  1. int a = 34;
  2. byte b = 120;
  3. short c = 32000;
  4. long d = 45000;
  5. long e = 320000L;

我们有五个赋值。 值 34、120、32000 和 45000 是int类型的整数字面值。 byteshort类型没有整数字面值。 如果这些值适合目标类型,则编译器不会抱怨并自动执行转换。 对于小于Integer.MAX_VALUElong数字,L后缀是可选的。

  1. long x = 2147483648L;
  2. long y = 2147483649L;

对于大于Integer.MAX_VALUElong数字,我们必须添加L后缀。

当我们使用整数时,我们处理离散项。 例如,我们可以使用整数来计算苹果。

com/zetcode/Apples.java

  1. package com.zetcode;
  2. public class Apples {
  3. public static void main(String[] args) {
  4. int baskets = 16;
  5. int applesInBasket = 24;
  6. int total = baskets * applesInBasket;
  7. System.out.format("There are total of %d apples%n", total);
  8. }
  9. }

在我们的程序中,我们计算了苹果的总量。 我们使用乘法运算。

  1. int baskets = 16;
  2. int applesInBasket = 24;

篮子数和每个篮子中的苹果数是整数值。

  1. int total = baskets * applesInBasket;

将这些值相乘,我们也得到一个整数。

  1. $ java Apples.java
  2. There are total of 384 apples

这是程序的输出。

可以在 Java 中使用四种不同的表示法指定整数:十进制,八进制,十六进制和二进制。 二进制符号是在 Java 7 中引入的。众所周知,通常使用小数。 八进制数字以0字符开头,后跟八进制数字。 十六进制数字以0x字符开头,后跟十六进制数字。 二进制数字以0b开头,后跟二进制数字(零和一)。

com/zetcode/IntegerNotations.java

  1. package com.zetcode;
  2. public class IntegerNotations {
  3. public static void main(String[] args) {
  4. int n1 = 31;
  5. int n2 = 0x31;
  6. int n3 = 031;
  7. int n4 = 0b1001;
  8. System.out.println(n1);
  9. System.out.println(n2);
  10. System.out.println(n3);
  11. System.out.println(n4);
  12. }
  13. }

我们有四个整数变量。 每个变量都被分配了一个具有不同整数符号的值。

  1. int n1 = 31;
  2. int n2 = 0x31;
  3. int n3 = 031;
  4. int n4 = 0b1001;

第一个是十进制,第二个十六进制,第三个八进制和第四个二进制。

  1. $ java IntegerNotations.java
  2. 31
  3. 49
  4. 25
  5. 9

我们看到了程序的输出。

大数字很难阅读。 如果我们有一个像 245342395423452 这样的数字,我们会发现很难快速阅读它。 在计算机外部,大数字之间用空格或逗号分隔。 从 Java SE 1.7 开始,可以用下划线分隔整数。

下划线不能在数字的开头或结尾,浮点字面量中的小数点附近以及FL后缀之前使用。

com/zetcode/UsingUnderscores.java

  1. package com.zetcode;
  2. public class UsingUnderscores {
  3. public static void main(String[] args) {
  4. long a = 23482345629L;
  5. long b = 23_482_345_629L;
  6. System.out.println(a == b);
  7. }
  8. }

此代码示例演示了 Java 中下划线的用法。

  1. long a = 23482345629L;
  2. long b = 23_482_345_629L;

我们有两个相同的长数字。 在第二个数字中,我们将数字中的每三个数字分开。 比较这两个数字,我们得到一个布尔值trueL后缀告诉编译器我们有一个长整数。

使用的 Java byteshortintlong类型确实表示固定精度数字。 这意味着它们可以表示有限数量的整数。 长类型可以代表的最大整数是 9223372036854775807。如果要处理更大的数字,则必须使用java.math.BigInteger类。 它用于表示不可变的任意精度整数。 任意精度整数仅受可用计算机内存量的限制。

com/zetcode/VeryLargeIntegers.java

  1. package com.zetcode;
  2. import java.math.BigInteger;
  3. public class VeryLargeIntegers {
  4. public static void main(String[] args) {
  5. System.out.println(Long.MAX_VALUE);
  6. BigInteger b = new BigInteger("92233720368547758071");
  7. BigInteger c = new BigInteger("52498235605326345645");
  8. BigInteger a = b.multiply(c);
  9. System.out.println(a);
  10. }
  11. }

java.math.BigInteger类的帮助下,我们将两个非常大的数字相乘。

  1. System.out.println(Long.MAX_VALUE);

我们打印可以用long类型表示的最大整数值。

  1. BigInteger b = new BigInteger("92233720368547758071");
  2. BigInteger c = new BigInteger("52498235605326345645");

我们定义了两个BigInteger对象。 它们都具有long类型可以容纳的更大的值。

  1. BigInteger a = b.multiply(c);

使用multiply()方法,我们将两个数字相乘。 请注意,BigInteger数字是不可变的。 该操作返回一个新值,我们将其分配给新变量。

  1. System.out.println(a);

计算出的整数将打印到控制台。

  1. $ java VeryLargeIntegers.java
  2. 9223372036854775807
  3. 4842107582663807707870321673775984450795

这是示例输出。

算术溢出

算术溢出是在计算产生的结果的大小大于给定寄存器或存储位置可以存储或表示的结果时发生的条件。

com/zetcode/Overflow.java

  1. package com.zetcode;
  2. public class Overflow {
  3. public static void main(String[] args) {
  4. byte a = 126;
  5. System.out.println(a);
  6. a++;
  7. System.out.println(a);
  8. a++;
  9. System.out.println(a);
  10. a++;
  11. System.out.println(a);
  12. }
  13. }

在此示例中,我们尝试分配一个超出数据类型范围的值。 这导致算术溢出。

  1. $ java Overflow.java
  2. 126
  3. 127
  4. -128
  5. -127

发生溢出时,变量将重置为负的上限值。

浮点数字

实数测量连续的数量,例如重量,高度或速度。 浮点数表示计算中实数的近似值。 在 Java 中,我们有两种原始浮点类型:floatdoublefloat是单精度类型,以 32 位存储数字。 double是双精度类型,以 64 位存储数字。 这两种类型具有固定的精度,不能完全表示所有实数。 在必须使用精确数字进行处理的情况下,可以使用BigDecimal类。

带有F/f后缀的浮点数是float类型,double数字有D/d后缀。 double数字的后缀是可选的。

假设一个短跑运动员跑了 1 个小时,跑了 9.87 秒。 他的公里/小时速度是多少?

com/zetcode/Sprinter.java

  1. package com.zetcode;
  2. public class Sprinter {
  3. public static void main(String[] args) {
  4. float distance;
  5. float time;
  6. float speed;
  7. distance = 0.1f;
  8. time = 9.87f / 3600;
  9. speed = distance / time;
  10. System.out.format("The average speed of a sprinter is %f km/h%n", speed);
  11. }
  12. }

在此示例中,必须使用浮点值。 在这种情况下,浮点数据类型的低精度不会造成问题。

  1. distance = 0.1f;

100m 是 0.1km。

  1. time = 9.87f / 3600;

9.87s 是9.87 / 60 * 60 h

  1. speed = distance / time;

为了获得速度,我们将距离除以时间。

  1. $ java Sprinter.java
  2. The average speed of a sprinter is 36.474163 km/h

这是程序的输出。 小数舍入误差不会影响我们对短跑运动员速度的理解。

floatdouble类型不精确。

com/zetcode/FloatingInPrecision.java

  1. package com.zetcode;
  2. public class FloatingInPrecision {
  3. public static void main(String[] args) {
  4. double a = 0.1 + 0.1 + 0.1;
  5. double b = 0.3;
  6. System.out.println(a);
  7. System.out.println(b);
  8. System.out.println(a == b);
  9. }
  10. }

该代码示例说明了浮点值的不精确性质。

  1. double a = 0.1 + 0.1 + 0.1;
  2. double b = 0.3;

我们定义两个double值。 D/d后缀是可选的。 乍一看,它们应该相等。

  1. System.out.println(a);
  2. System.out.println(b);

打印它们将显示很小的差异。

  1. System.out.println(a == b);

该行将返回false

  1. $ java FloatingInPrecision.java
  2. 0.30000000000000004
  3. 0.3
  4. false

边距误差很小。 因此,比较运算符返回布尔值false

当我们使用货币,货币以及通常用于业务应用时,我们需要使用精确的数字。 基本浮点类型的舍入误差是不可接受的。

com/zetcode/CountingMoney.java

  1. package com.zetcode;
  2. public class CountingMoney {
  3. public static void main(String[] args) {
  4. float c = 1.46f;
  5. float sum = 0f;
  6. for (int i=0; i<100_000; i++) {
  7. sum += c;
  8. }
  9. System.out.println(sum);
  10. }
  11. }

1.46f代表 1 欧元和 46 美分。 我们从 100000 个这样的金额中创建一个总和。

  1. for (int i=0; i<100_000; i++) {
  2. sum += c;
  3. }

在此循环中,我们从 100000 这样的金额中创建一个总和。

  1. $ java CountingMoney.java
  2. 146002.55

计算得出的错误为 2 欧元和 55 美分。

为了避免此裕度误差,我们利用了BigDecimal类。 它用于保存不可变的,任意精度的带符号十进制数字。

com/zetcode/CountingMoney2.java

  1. package com.zetcode;
  2. import java.math.BigDecimal;
  3. public class CountingMoney2 {
  4. public static void main(String[] args) {
  5. BigDecimal c = new BigDecimal("1.46");
  6. BigDecimal sum = new BigDecimal("0");
  7. for (int i=0; i<100_000; i++) {
  8. sum = sum.add(c);
  9. }
  10. System.out.println(sum);
  11. }
  12. }

我们用相同的金额执行相同的操作。

  1. BigDecimal c = new BigDecimal("1.46");
  2. BigDecimal sum = new BigDecimal("0");

我们定义两个BigDecimal数字。

  1. for (int i=0; i<100_000; i++) {
  2. sum = sum.add(c);
  3. }

BigDecimal数字是不可变的,因此在每个循环中始终将一个新对象分配给sum变量。

  1. $ java CountingMoney2.java
  2. 146000.00

在此示例中,我们获得了精确值。

Java 支持浮点值的科学语法。 也称为指数表示法,它是一种写数字太大或太小而不能方便地用标准十进制表示法写的方式。

com/zetcode/ScientificNotation.java

  1. package com.zetcode;
  2. import java.math.BigDecimal;
  3. import java.text.DecimalFormat;
  4. public class ScientificNotation {
  5. public static void main(String[] args) {
  6. double n = 1.235E10;
  7. DecimalFormat dec = new DecimalFormat("#.00");
  8. System.out.println(dec.format(n));
  9. BigDecimal bd = new BigDecimal("1.212e-19");
  10. System.out.println(bd.toEngineeringString());
  11. System.out.println(bd.toPlainString());
  12. }
  13. }

我们使用科学计数法定义两个浮点值。

  1. double n = 1.235E10;

这是double类型的浮点值,以科学计数法表示。

  1. DecimalFormat dec = new DecimalFormat("#.00");
  2. System.out.println(dec.format(n));

我们使用DecimalFormat类将double值排列为标准十进制格式。

  1. BigDecimal bd = new BigDecimal("1.212e-19");
  2. System.out.println(bd.toEngineeringString());
  3. System.out.println(bd.toPlainString());

BigDecimal类采用科学计数法中的浮点值作为参数。 我们使用该类的两种方法来打印工程字符串和纯字符串中的值。

  1. $ java ScientificNotation.java
  2. 12350000000.00
  3. 121.2E-21
  4. 0.0000000000000000001212

This is the example output.

枚举

枚举类型是一种特殊的数据类型,它使变量成为一组预定义的常量。 可以将任何枚举器分配为已声明为具有枚举类型的变量作为值。 枚举使代码更具可读性。 当我们处理只能从一小部分可能的值中取出一个的变量时,枚举很有用。

com/zetcode/Enumerations.java

  1. package com.zetcode;
  2. public class Enumerations {
  3. enum Days {
  4. MONDAY,
  5. TUESDAY,
  6. WEDNESDAY,
  7. THURSDAY,
  8. FRIDAY,
  9. SATURDAY,
  10. SUNDAY
  11. }
  12. public static void main(String[] args) {
  13. Days day = Days.MONDAY;
  14. if (day == Days.MONDAY) {
  15. System.out.println("It is Monday");
  16. }
  17. System.out.println(day);
  18. for (Days d : Days.values()) {
  19. System.out.println(d);
  20. }
  21. }
  22. }

在我们的代码示例中,我们为工作日创建一个枚举。

  1. enum Days {
  2. MONDAY,
  3. TUESDAY,
  4. WEDNESDAY,
  5. THURSDAY,
  6. FRIDAY,
  7. SATURDAY,
  8. SUNDAY
  9. }

使用enum关键字创建代表星期几的枚举。 枚举项是常量。 按照约定,常量用大写字母表示。

  1. Days day = Days.MONDAY;

我们有一个名为day的变量,其类型为Days。 它被初始化为星期一。

  1. if (day == Days.MONDAY) {
  2. System.out.println("It is Monday");
  3. }

与将日变量与某个数字进行比较相比,此代码更具可读性。

  1. System.out.println(day);

该行将在星期一打印到控制台。

  1. for (Days d : Days.values()) {
  2. System.out.println(d);
  3. }

此循环将整天打印到控制台。 静态values()方法按声明顺序返回包含此enum类型常量的数组。 此方法可用于通过增强的for语句迭代常量。 增强的for逐元素遍历数组,并将其打印到终端。

  1. $ java Enumerations.java
  2. It is Monday
  3. MONDAY
  4. MONDAY
  5. TUESDAY
  6. WEDNESDAY
  7. THURSDAY
  8. FRIDAY
  9. SATURDAY
  10. SUNDAY

This is the example output.

可以给枚举常量一些值。

com/zetcode/Enumerations2.java

  1. package com.zetcode;
  2. enum Season {
  3. SPRING(10),
  4. SUMMER(20),
  5. AUTUMN(30),
  6. WINTER(40);
  7. private int value;
  8. private Season(int value) {
  9. this.value = value;
  10. }
  11. public int getValue() {
  12. return value;
  13. }
  14. }
  15. public class Enumerations2 {
  16. public static void main(String[] args) {
  17. for (Season season : Season.values()) {
  18. System.out.println(season + " " + season.getValue());
  19. }
  20. }
  21. }

该示例包含一个Season枚举,该枚举具有四个常量。

  1. SPRING(10),
  2. SUMMER(20),
  3. AUTUMN(30),
  4. WINTER(40);

在这里,我们定义enum的四个常数。 给常数指定特定值。

  1. private int value;
  2. private Season(int value) {
  3. this.value = value;
  4. }

当定义常量时,我们还必须创建一个构造器。 构造器将在本教程的后面部分介绍。

  1. SPRING 10
  2. SUMMER 20
  3. AUTUMN 30
  4. WINTER 40

This is the example output.

字符串和字符

String是代表计算机程序中文本数据的数据类型。 Java 中的字符串是字符序列。 char是单个字符。 字符串用双引号引起来。

由于字符串在每种编程语言中都非常重要,因此我们将为它们专门整整一章。 在这里,我们仅举一个小例子。

com/zetcode/StringsChars.java

  1. package com.zetcode;
  2. public class StringsChars {
  3. public static void main(String[] args) {
  4. String word = "ZetCode";
  5. char c = word.charAt(0);
  6. char d = word.charAt(3);
  7. System.out.println(c);
  8. System.out.println(d);
  9. }
  10. }

程序将Z字符打印到终端。

  1. String word = "ZetCode";

在这里,我们创建一个字符串变量,并为其分配"ZetCode"值。

  1. char c = word.charAt(0);

charAt()方法返回指定索引处的char值。 序列的第一个char值在索引 0 处,第二个在索引 1 处,依此类推。

  1. $ java StringsChars.java
  2. Z
  3. C

该程序将"ZetCode"字符串的第一个和第四个字符打印到控制台。

数组

数组是处理元素集合的复杂数据类型。 每个元素都可以通过索引访问。 数组的所有元素必须具有相同的数据类型。

我们将整章专门介绍数组。 这里我们仅显示一个小例子。

com/zetcode/ArraysEx.java

  1. package com.zetcode;
  2. public class ArraysEx {
  3. public static void main(String[] args) {
  4. int[] numbers = new int[5];
  5. numbers[0] = 3;
  6. numbers[1] = 2;
  7. numbers[2] = 1;
  8. numbers[3] = 5;
  9. numbers[4] = 6;
  10. int len = numbers.length;
  11. for (int i = 0; i < len; i++) {
  12. System.out.println(numbers[i]);
  13. }
  14. }
  15. }

在此示例中,我们声明一个数组,用数据填充它,然后将数组的内容打印到控制台。

  1. int[] numbers = new int[5];

我们创建一个整数数组,该数组最多可以存储 5 个整数。 因此,我们有五个元素组成的数组,索引为0..4

  1. numbers[0] = 3;
  2. numbers[1] = 2;
  3. numbers[2] = 1;
  4. numbers[3] = 5;
  5. numbers[4] = 6;

在这里,我们为创建的数组分配值。 我们可以通过数组访问符号访问数组的元素。 它由数组名称和方括号组成。 在方括号内,我们指定所需元素的索引。

  1. int len = numbers.length;

每个数组都有一个length属性,该属性返回数组中的元素数。

  1. for (int i = 0; i < len; i++) {
  2. System.out.println(numbers[i]);
  3. }

我们遍历数组并将数据打印到控制台。

  1. $ java ArraysEx.java
  2. 3
  3. 2
  4. 1
  5. 5
  6. 6

This is the output of the program.

在 Java 教程的这一部分中,我们介绍了 Java 中的数据类型。