Java学习之旅基础知识篇:数据类型及流程控制

时间:2021-07-08 03:19:35

经过开篇对Java运行机制及相关环境搭建,本篇主要讨论Java程序开发的基础知识点,我简单的梳理一下。在讲解数据类型之前,我顺便提及一下Java注释:单行注释、多行注释以及文档注释,这里重点强调文档注释。通常单行注释独占一行并用"//"来开头,多行注释会占据多行并用/*...*/来表示。

Java学习之旅基础知识篇:数据类型及流程控制
/*
HelloWorld是每一个Java学习者的起点,
需要好好掌握
*/
public class HelloWorld {
public static void main(String[] args) {
// 输出Hello World!
System.out.println("Hello World!");
}
}
Java学习之旅基础知识篇:数据类型及流程控制

那什么叫文档注释呢?简单的说,就是利用javadoc将源代码的文档注释生成一份API文档,供使用者查询和参考。JDK API的在线文档查询:http://dlc.sun.com.edgesuite.net/jdk/jdk-api-localizations/jdk-api-zh-cn/builds/latest/html/zh_CN/api/,从文档界面不难看出分成三大区:包列表区(左上方)、类列表区(左下方)、详细说明区(右侧)。点击左侧"类列表区"中的类,右侧将显示该类对应的构造函数、属性、方法等详细信息。接下来,我们利用javadoc来生成一份API文档,需要说明的是,javadoc只处理以public或protected修饰的类、接口、方法、属性、构造器和内部类之前的文档注释而忽略其他地方的文档注释,如果开发者希望给使用者提供private成员的文档,则需要加入-private选项。文档注释用/**...*/来表示。javadoc的语法如下:

javadoc -d [API文档存放目录] -windowtitle [API文档浏览器标题] -doctitle [概述页面标题] -header [页面页眉] ... Java源文件/包 (可利用javadoc -help查询全部选项)

如源代码中包含中文,则需要设置-locale, -encoding, -charset如下:

javadoc -d [API文档存放目录] -windowtitle [API文档浏览器标题] -doctitle [概述页面标题] -header [页面页眉] -encoding utf-8 -charset utf-8 -locale zh_CN ... Java源文件/包 (可利用javadoc -help查询全部选项)

同时,为了更详细说明类、方法等成员的文档信息,经常添加以下javadoc标记(如@author、@version、@param、@return等)到这些成员上加以说明。

Java学习之旅基础知识篇:数据类型及流程控制
package miracle;
/**
* Description:
* <br/>此程序主要测试Javadoc指令来生成文档注释
* <br/>程序名:TestJavadoc.java
* <br/>编写日期:2012-08-22
* @author Miracle, He miracle@sina.com
* @version 1.0
*/
public class TestJavadoc {
/**
* 测试属性
*/
protected String name;
/**
* 主方法,程序入口
* @param args 输入参数列表
*/
public static void main(String[] args) {
System.out.println("Hello, Miracle!");
}
}
Java学习之旅基础知识篇:数据类型及流程控制
Java学习之旅基础知识篇:数据类型及流程控制
package miracleHe;
/**
* Description:
* <br/>此程序主要测试Javadoc指令来生成文档注释
* <br/>程序名:TestJavadocTag.java
* <br/>编写日期:2012-08-22
* @author Miracle, He miracle@sina.com
* @version 1.0
*/
public class TestJavadocTag {
/**
* 此方法用来打招呼
* @param name 打招呼的名称
* @return 返回打招呼的字符串
*/
private String hello(String name) {
return name + ",你好!";
}
}
Java学习之旅基础知识篇:数据类型及流程控制

我们这里添加了两个包miracle和miracleHe(目的为生成概述),执行以下命令将输出API文档。

javadoc -d apidoc -windowtitle "Generate Javadoc" -doctitle "Learning HelloWorld Class by Javadoc" -header "Javadoc Test" -charset "utf-8" -encoding "utf-8" Test*.java

默认情况下不会提取@author、@version等信息,如需提取则需要添加。

javadoc -d apidoc -windowtitle "Generate Javadoc" -doctitle "Learning HelloWorld Class by Javadoc" -header "Javadoc Test"-charset "utf-8" -encoding "utf-8" -author -version Test*.java

Java是强类型语言,也就是变量或表达式在编译时就已经明确其类型,即先声明后使用。数据类型分为: 基本类型(Primitive Type)和引用类型(Reference Type)。其组织关系如下:基本类型包含整数类型(byte、short、int、long)、字符类型(char)、浮点类型(float、double)和布尔类型(true、false);引用类型包含字符串(String)、数组、类、接口和空类型(null)。先从整数类型谈起,如果一个较小的整数(在byte或short的范围之内)赋值给byte或short时,系统会自动转化为对应的类型;如果一个巨大的整数(超出int的范围),系统则不会自动当做long来进行处理,需要添加后缀L来进行标识,即使这个整数在int的范围之内声明的long类型变量不添加L仍然当做是int

Java学习之旅基础知识篇:数据类型及流程控制
public class TestInteger {
public static void main(String[] args) {
byte b = 56; //系统会自动转化为byte
//long big = 999999999999; //出错,系统不会当作long来处理
long big2 = 41433333313243133L; //使用L强制为long类型
}
}
Java学习之旅基础知识篇:数据类型及流程控制

整数除了十进制来表示外,还可以使用八进制(0开头)以及十六进制(0x开头,A~F代表10~15)来进行表示。

int octalValue = 013;
int hexValue = 0x2F;

虽然字符型被单独处理,但其实它就是一种整数(0~65535之间的无符号整数),字符通常可以使用''(如'A')、转义字符('\n'、'\r')和Unicode('\uXXXX',前256个字符与ASCII码一致)来表示。如果将0~65535之间的整数赋给char变量将直接将int转化为char类型。

Java学习之旅基础知识篇:数据类型及流程控制
public class TestChar {
public static void main(String[] args) {
//定义字符型
char a = 'a';
char enter = '\r';
char ch = '\u24af';
System.out.println(ch);//输出?
char yu = '宇';
int yuValue = yu;
System.out.println(yuValue);//23431
char c = 97;
System.out.println(c);//a
}
}
Java学习之旅基础知识篇:数据类型及流程控制

但请注意,字符串虽然是由字符组成(可看作字符数组),但字符串是引用类型。接下来,我们讨论一下浮点数。浮点数分为单精度浮点数(float)和双精度浮点数(double)。其中float占4个字节,第一位是符号位,接下来8位是指数位,最后23位是尾数,必须要添加后缀F;double占8个字节,第一位是符号位,接下来11位是指数位,最后52位是尾数,是默认类型,可以不添加后缀D来标识。浮点数可用十进制(如5.12,.512)和科学计数法(5.12E2)来表示。特别需要注意的是:浮点数还包含三个特殊的值,正无穷大(POSITIVE_INFINITY,通过正数除以0得到)、负无穷大(NEGATIVE_INFINITY,通过负数除以0得到)和非数(NaN,通过0.0除以0.0或对负数开方得到)。所有正无穷大值都相等,所有负无穷大值相等,非数不与任何数相等(包含NaN本身也不相等)

Java学习之旅基础知识篇:数据类型及流程控制
public class TestFloat {
public static void main(String[] args) {
float f = 5.12F;
double zero = 0.0;
float p = Float.POSITIVE_INFINITY;
double n = Double.NEGATIVE_INFINITY;
System.out.println(p==n);//false
System.out.println(f/zero);//Infinity
System.out.println(f/zero==p);//true
System.out.println(0.0/zero==Double.NaN);//false
System.out.println(6.0/0==8.2/0);//true
System.out.println(1/0);//抛出异常
}
}
Java学习之旅基础知识篇:数据类型及流程控制

另外提一句,bool类型只能为true或false,不能用0或非0来表示,其他基本类型都不能转化为bool类型,如果bool类型与字符串相连,将直接转化为字符串

接下来,我们讨论一下数据类型转换相关知识。数据类型转换分为自动转换和强制转换,自动转换的关系如下图:

Java学习之旅基础知识篇:数据类型及流程控制

箭头左边的类型可以自动转化为右边的类型。此外,当基本类型与字符串进行连接时,基本类型会自动转换为字符串。反之,将字符串转化为基本类型则调用xxx.pareseXxx方法(如Integer.parseInt("12"))。

Java学习之旅基础知识篇:数据类型及流程控制
public class AutoConversion {
public static void main(String[] args) {
int a = 6;
float f = a;
System.out.println(f);//6.0
byte b = 9;
//char c = b;//出错,byte不能自动转化为char
double d = b;
System.out.println(d);//9.0
//基本类型与字符串进行转化
String s = 5.3F + "";
System.out.println(s);//5.3
System.out.println(3 + 4 + "Hello!");//7Hello!
System.out.println("Hello!" + 3 + 4);//Hello!34
}
}
Java学习之旅基础知识篇:数据类型及流程控制

反之,如果想把箭头右边的类型转换为左边的类型,就需要强制类型转换,这样做可能导致数据溢出或精度丢失。因此在进行强制类型转换时需格外小心。

Java学习之旅基础知识篇:数据类型及流程控制
public class ForceConversion {
public static void main(String[] args) {
int i = 234;
byte b = (byte)i;
System.out.println(b);//-22
double d = 3.56;
int n = (int)d;
System.out.println(n);//3
}
}
Java学习之旅基础知识篇:数据类型及流程控制

可能大家会问234咋转换为byte就变成了-22了呢?我们都知道,byte的范围是-128~127,显示超出了表示的范围,234的二进制表示为00..0011101010,转换后截取后8位之后变成11101010,而第一位是符号位(这里是个负数),而负数在计算机中以补码形式存在,需要转换为原码(补码减1成反码再按位取反,符号位不变,因此:11101010-->11101001-->10010110)。

在进行表达式计算时,数据类型会发生自动提升:如所有byte、short和char将自动提升为int类型,表达式的数据类型将提升为最高等级操作数的数据类型。对于整数相除时,即使不能除尽也要舍弃小数部分,对于字符串与数字或字符相加时,此时应该从左自右进行运算,以判断是否为字符串连接还是加法运算。

Java学习之旅基础知识篇:数据类型及流程控制
public class AutoPremotion {
public static void main(String[] args) {
short s = 5;
//s = s - 2;//出错,表达式被提升为int
byte b = 10;
char c = 'a';
double d = .12;
System.out.println(b + c + d);//表达式被提升为double
System.out.println(23 / 3);//7
System.out.println(b + c + "Hello!");//107Hello!
System.out.println("Hello!" + b + c);//Hello!10a
}
}
Java学习之旅基础知识篇:数据类型及流程控制

在讲解流程控制之前,我在这里补充一下平常容易出错的知识点。我们来看以下程序:

Java学习之旅基础知识篇:数据类型及流程控制
public class CompareString {
public static void main(String[] args) {
String a = new String("Miracle");
String b = new String("Miracle");
System.out.println(a == b);//false,因此a与b指向不同的实例(尽管内容一致)
String c = "Miracle";
String d = "Miracle";
System.out.println(c == d);//true,此处由于字符串缓存机制,比较的仅仅是两者的内容
}
}
Java学习之旅基础知识篇:数据类型及流程控制

另外,对于短路运算符(如||)与不短路运算符(如|)的区别: ||先计算左边的操作数,如果为true将不再继续计算,而|不管左边结果如何都会计算之后的操作数

Java学习之旅基础知识篇:数据类型及流程控制
public class TestLogicOperator {
public static void main(String[] args) {
int a = 5;
int b = 10;
if(a > 4 || b++ > 10)
{
System.out.println("a=" + a + ",b=" + b);//a=5,b=10
}
if(a > 4 | b++ > 10)
{
System.out.println("a=" + a + ",b=" + b);//a=5,b=11
}
}
}
Java学习之旅基础知识篇:数据类型及流程控制

接下来,进入流程控制的讲解,任何编程语言(Java也不例外)的流程控制结构包含:顺序结构和分支结构(if、if...else、if...else if...else、switch)和循环结构(while、do...while、for、foreach)。这里不再举例说明,只是强调一点,对于分支结构if...else,尽量不要省略之后的花括号,即使只有一条语句也不能省略,其中对于非常简单的if...else结构,可以用三目运算符(if(a>b)?a:b)来替代。为了避免发生逻辑错误,应该遵守:总是优先把包含范围小的的条件放在前面处理

Java学习之旅基础知识篇:数据类型及流程控制
public class TestIf {
public static void main(String[] args) {
int age = 45;
if(age > 20) {
System.out.println("Young");
} else if(age > 40) {
System.out.println("Middle");
} else if(age > 60) {
System.out.println("Old");
}
}
}
Java学习之旅基础知识篇:数据类型及流程控制

我们发现运行之后输出Young,明显与预期不符(应该输出Middle)。就是因为刚才提到的范围问题导致(age > 20比age > 60范围大)。我们改写一下:

Java学习之旅基础知识篇:数据类型及流程控制
public class TestIf {
public static void main(String[] args) {
int age = 45;
if(age > 60) {
System.out.println("Old");
} else if(age > 40) {
System.out.println("Middle");
} else if(age > 20) {
System.out.println("Young");
}
}
}
Java学习之旅基础知识篇:数据类型及流程控制

刚才提到if之后的花括号不能省略,对于switch...case来说,case之后的花括号尽量省略,而break不要省略(否则将贯穿多个case执行),尽量加上default,此外对于switch(expression)的表达式只能为整数,不能为字符串(这点跟其他语言不一样)。
      对于循环结构,也不要省略循环体中的花括号(即使只有一句),还可以组成多层嵌套循环。可以使用break结束本层循环,进入循环之后的代码,使用continue结束本次循环,进入下一次循环,也可以使用return直接返回。但是有时还有特殊情况,就是从内层循环跳出到外层循环,需要使用Java标签(用:表示),不过通常此标签必须位于break所在循环的外层循环之前才起作用。以下程序将输出: i = 0,j = 0;i = 0,j = 1。

Java学习之旅基础知识篇:数据类型及流程控制
public class TestBreak {
public static void main(String[] args) {
outer:
for(int i = 0; i < 5; i++) {
for(int j = 0; j < 2; j++) {
System.out.println("i = " + i + ",j = " + j);
if(j == 1) {
break outer;
}
}
}
}
}
Java学习之旅基础知识篇:数据类型及流程控制

此外,continue也可添加标签,表示立即结束continue所在循环,跳到标签所在位置进入下一次循环。如果将以上的break改成continue的话,将输出: i = 0, j = 0; i = 0, j = 1; i = 1, j = 0; i = 1, j = 1; i = 2, j = 0; i = 2, j = 1; i = 3, j = 0; i = 3, j = 1; i = 4, j = 0; i = 4, j = 1。最后,return直接返回整个方法,而不管方法中嵌套有多少层循环。