博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java学习之旅基础知识篇:数据类型及流程控制
阅读量:6616 次
发布时间:2019-06-25

本文共 8820 字,大约阅读时间需要 29 分钟。

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

/*    HelloWorld是每一个Java学习者的起点,    需要好好掌握*/public class HelloWorld {    public static void main(String[] args) {        // 输出Hello World!        System.out.println("Hello World!");    }}

那什么叫文档注释呢?简单的说,就是利用javadoc将源代码的文档注释生成一份API文档,供使用者查询和参考。JDK 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等)到这些成员上加以说明。

package miracle;/*** Description:* 
此程序主要测试Javadoc指令来生成文档注释*
程序名:TestJavadoc.java*
编写日期: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!"); }}
package miracleHe;/*** Description:* 
此程序主要测试Javadoc指令来生成文档注释*
程序名:TestJavadocTag.java*
编写日期: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 + ",你好!"; }}

我们这里添加了两个包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

public class TestInteger {    public static void main(String[] args) {        byte b = 56; //系统会自动转化为byte        //long big = 999999999999; //出错,系统不会当作long来处理        long big2 = 41433333313243133L; //使用L强制为long类型    }}

整数除了十进制来表示外,还可以使用八进制(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类型。

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    }}

但请注意,字符串虽然是由字符组成(可看作字符数组),但字符串是引用类型。接下来,我们讨论一下浮点数。浮点数分为单精度浮点数(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本身也不相等)

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);//抛出异常    }}

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

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

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

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    }}

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

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    }}

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

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

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    }}

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

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,此处由于字符串缓存机制,比较的仅仅是两者的内容    }}

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

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

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");        }    }}

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

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");        }    }}

刚才提到if之后的花括号不能省略,对于switch...case来说,case之后的花括号尽量省略,而break不要省略(否则将贯穿多个case执行),尽量加上default,此外对于switch(expression)的表达式只能为整数,不能为字符串(这点跟其他语言不一样)。

      对于循环结构,也不要省略循环体中的花括号(即使只有一句),还可以组成多层嵌套循环。可以使用break结束本层循环,进入循环之后的代码,使用continue结束本次循环,进入下一次循环,也可以使用return直接返回。但是有时还有特殊情况,就是从内层循环跳出到外层循环,需要使用Java标签(用:表示),不过通常此标签必须位于break所在循环的外层循环之前才起作用。以下程序将输出: i = 0,j = 0;i = 0,j = 1。

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;                }            }        }    }}

此外,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直接返回整个方法,而不管方法中嵌套有多少层循环。

转载于:https://www.cnblogs.com/hmiinyu/archive/2012/10/20/2730293.html

你可能感兴趣的文章
修改页面JS 360浏览器
查看>>
Git 跟 GitHub 是什么关系?
查看>>
IE6下jQuery选中select的BUG
查看>>
Tensorflow在win10下的安装(CPU版本)
查看>>
一次优化记录
查看>>
cgroup代码浅析(2)
查看>>
会计的思考(42):会计如何转变为公司的内部财务顾问
查看>>
利用钥匙串,在应用里保存用户密码的方法
查看>>
python 装饰器
查看>>
[辟谣]下蹲猛起来眼前发黑是心脏衰竭的表现?别扯了!
查看>>
paper 96:计算机视觉-机器学习近年部分综述
查看>>
vuex状态管理详细使用方法
查看>>
不要等有了足够的钱才选择去创业!!!
查看>>
手把手教你画嘴巴,以后再也不怕画嘴巴了
查看>>
selenium - webdriver - 截图方法get_screenshot_as_file()
查看>>
linux 命令 — archive
查看>>
强大的jQuery网格插件 ParamQuery
查看>>
io.lettuce.core.RedisCommandTimeoutException: Command timed out
查看>>
种子填充算法描述及C++代码实现
查看>>
Kali渗透测试——快速查找Metasploit的模块
查看>>