第三章 Java基本程序设计结构
-
逐一声明每一个变量可以提高程序的可读性。
-
在Java中,变量的声明尽可能地靠近变量第一次使用的地方,这是一种良好的程序编写风格。
-
常量名使用全大写。使用关键字static final设置一个类常量。
public static final double CM_PER_INCH = 2.54;
类常量定义位于main方法的外部。 -
我们建议不要在其他表达式的内部使用
++
,这样编写的代码很容易令人困惑,并会产生烦人的bug。(一个笑话C++) -
&&
和||
按照短路的方式求值。&
,|
运算符应用于布尔值,得到的结果也是布尔值。不按短路方式计算。 -
>>>
运算符将用0填充高位;>>
运算符用符号位填充高位。没有<<<
运算符。 -
强制类型转换
得到最接近的整数 double x = 9.997; int nx = (int) Math.round(x);//round返回的结果是long类型
-
&&
优先级比||
的优先级高。+=
右结合运算符。 -
substring方法的第二个参数是不想复制的第一个位置,substring可以计算字符串长度。
-
当一个字符串和一个非字符串的值进行拼接时,后者被转换成字符串(任何一个对象都可以转换成字符串。
-
在java文档中将String类对象称为不可变字符串。java的设计者认为共享带来的高效率远远胜过于提取,拼接字符串带来的低效率。
-
检测字符串是否相等
s.equals(t)
s,t可以是字符串变量可以是字符串常量。==
判断字符串是否放在同样位置上,如果在同样位置上,它们必然相等。 -
if(str.length() == 0) 或 if(str.equals("")) 是否为空
-
if(str != null && str.length() != 0) 检查字符串既不是null也不是空串
-
length()方法返回采用UTF-16编码表示到给定字符串所需要到代码单元数量.//3.6.6
int cpCount = greeting.codePointCount(0, greeting.length());//得到实际长度,即代码点得数量. char first = greeting.chatAt(0);//调用s.char.charAt(n)将返回位置n到代码单元. //得到第i个代码点. int index = gretting.offsetByCodePoint(0, i); int cp = gertting.codePintAt(index);// //遍历一个字符串,依次查看每一个代码点 int cp = sentence.codePointAt(i); if(Chartacter.isSupplementaryCodePoint(cp)) i += 2; else i++; //回退操作 i--; if(Chartacter.isSupplementaryCodePoint(cp)) i--; int cp = sentence.codePointAt(i);
-
常用的字符串API
API java.lang.Stirng 1.0 int codePointAt(int index);//返回给定位置开始或结束到代码点 int offsetByCodePoints(int startIndex, int cpCount);//返回从startIndex代码点开始,位移cpCount后的代码点索引 int compareTo(String other);//按照字典顺序,如果字符串位于other之前,返回一个负数;如果字符串位于other之后返回一个正数;如果两个字符串相等,返回0 int indexOf(String str); int indexOf(Sting str, int fromIndex); int indexOf(int cp); int indexOf(int cp, int fromIndex);//返回与字符串str或代码点cp匹配的第一个子串得开始位置.这个位置从索引0或fromIndex开始计算.如果在原始串中不存在str,返回-1. int codePointCount(int startIndex, int endIndex);//返回startIndex和endIndex-1 之间的代码点数量.没有配成对到代用字符将计入代码点. Stirng trim();//返回一个新字符串.这个字符串将删除年原始字符串头部和尾部到空格.
-
StringBuilder(JDK5.0引入StringBilder类,前身是StringBuffer)
StringBuilder builder = new StringBuilder();//构造空的字符串构建器 bulider.append(ch);添加内容 String completedString = builder.toStirng();//构建字符串,得到一个String对象,其中包含了构建器中的字符序列。
API java.lang.StringBuilder 5.0 StringBuilder();//构造一个空的字符串构建器 int length();//返回构建器或缓冲器中的代码单元 StringBuilder append(String str);//追加一个字符串并返回this StringBuilder append(char c);//追加一个代码单元并返回this StringBuilder appendCodePoint(int cp);//追加一个代码点,并将其转换为一个或两个代码单元并返回this String toStrinig();//返回一个与构造器或缓冲区内容相同的字符串。
-
标准输入流System.in,Scanner
API java.util.Scanner 5.0 Scanner in = new Scanner(System.in);//用给定的输入流创建一个scanner对象。 System.out.print("What is your name?"); String name = in.nextLine();//输入一行,包括空格。 String firstName = in.next();//读取一个单词,以空格符作为分隔符。 System.out.print("How old are you?"); int age = in.nextInt();//读取一个整数。nextDouble()读取下一个浮点数。
-
当使用的类不是定义在基本java.lang包中时,一定要使用import指定字将相应的包加载进来。
-
Scanner类,输入是可见的,不适合用于控制台读取密码。Java SE 6引入Console类实现读取密码
API java.io.Console 6 Console cons = System.console(); String username = cons.readLine("User name: "); char[] passwd = cons.readPassword("Password: ");//Console每次只能读取一行,没有读取一个单词或者一个数值的方法。
API java.lang.System 1.0 staic Console console() 6 //交互操作
-
每一个以%字符开始的格式说明符都用相应的参数替换。格式说明符尾部的转换符将指示被格式化的数值类型。
-
printf方法中日期与时间的格式化选项。以t开始,以表3-7中任意字母结束的两个字母格式。
System.printf("%tc", new Date());//Mon Feb 09 18:05:19 PST 2004 System.printf("%1$s %2StB %2$te, %2$tY", "Due date", new Date());//Due date: February 9, 2004 System.out.printf("%s %tB %<te, %<tY", "Due date: ", new Dte());//<标志,它指示前面格式说明中的参数将再次被使用。
-
参数索引值从1开始,而不是从0开始,
%1$...
对第1个参数格式化。 -
文件输入输出
Scanner in = new Scanner(Paths.get("myfile.txt")); PrintWriter out = new PrintWriter("myfile.txt"); Scanner in = new Scanner("myfile.txt");//字符串
API java.io.PrintWrier 1.1 PrintWriter(String fileName);//构造一个将数据写入文件的PrintWriter。文件名由参数决定。 API java.nio.file.Pahts 7 static Path get(String pathname);//根据给定的路径名构造一个Path。
-
else与最邻近的if构成一组
if(x <= 0) if(x == 0) sign = 0; else sign = -1;
-
不成文的规定:for语句中的3个部分应该对同一个计数器变量进行初始化,检测,更新。
-
在循环中检测两个浮点数是否相等需要格外小心。
for(double x = 0; x != 10; x += 0.1)...//可能永远不会结束,由于舍入的误差,最终可能得不到精确值。循环中,因为0.1无法精确地用二进制表示,x将从9.99...98跳到10.09...98
-
从n个数字中抽取k个数字
$$\frac{n*(n-1)(n-2)...(n-k+1)}{123456}$$
int lotterOdds = 1;
for(int i = 1; i <= k; i++) {
lotterOdds = lotterOdds * (n-i+1) / i;
}
- switch的case标签可以是:
类型为char, byte, short或int(或其包装类Character, Byte, Short和Integer)的常量表达式. 枚举变量 从Java SE 7 开始,case标签还可以是字符串字面量.
-
当在switch语句中使用枚举常量时,不必在每个标签中指明枚举名,可以由switch得表达式值确定.
Size sz = ...; switch(sz) { case SMALL: //no need to use Size.SMALL ... breakl ... }
-
带标签的break,continue(标签必须放在希望跳出到最外层循环之前,并且必须紧跟一个冒号.)
-
大数值BigInteger(实现任意精度的整数运算),BigDecimal(实现任意精度的浮点数运算)
API java.math.BigInteger 1.1 BigInteger add(BigInteger other);//和 BigInteger subtract(BigInteger other);//差 BigInteger multiply(BigInteger other);//积 BigInteger divide(BigInteger other);//商 BigInteger mod(BigInteger other);//余数 //返回一个大整数和另一个大整数other的和,差,积,商以及余数。 static BigInteger valueOf(long x);//返回值等于x的大整数。 BigDecimal add(BigDecimal other);//和 BIgDecimal subtract(BigDecimal other);//差 BigDecimal multiply(BigDecimal other);//积 BigDecimal divide(BigDecimal other RoundingMode mode);//商 //返回这个大实数与另一个大实数other.HALF_UP是在学校学习的四舍五入方式。 static BigDecimal valueOf(long x); static BigDecimal valueOf(long x, int scale); //返回值位x或x/10^scale的大实数
$$x/10^{scale}$$
int[] a;//声明整型数组a
int[] a = new int[100];//初始化一个数组,数组的长度不要求是常量,new int[n]创建一个长度位n的数组。
-
创建一个数字数组时,所有的元素都会被初始化为0.boolean数组的元素会初始化称false。对象数组的元素则初始化为一个特殊值null,这表示元素还未存放任何对象。
-
一旦创建数组就不能再改变它的大小,如果经常需要在运行过程中扩展数组的大小,就应该使用另一种数据结构-数组列表(array list)。
-
增强for循环的语句格式为:for(variable : collection) statement.定义一个变量用于暂存集合中的每一个元素,并执行相应的语句(或者语句块)。collection这一集合表达式必须是一个数组或者是一个实现类Iterable接口的类对象(如ArrayList).(for each element in a)
-
更简单的方式打印数组中的所有值,Arrays.toString(a),返回一个包含数组元素的字符串,这些元素放在括号内,并用逗号分开,如:"[2, 3, 5, 7, 11, 13]".
int[] smallPrimes = {2, 3, 5, 7, 11, 13};//创建数组并初始化的简化形式,不需要new new int[] {17, 19, 23, 29, 31, 37};//初始化一个匿名数组 samllPrimes = new int[] {17, 19, 23, 29, 31, 37};
-
在java中允许数组的长度为0.
int[] copiedLuckyNumbers = Array.copyOf(luckNumbers, luckNumbers.length);//将一个数组的所有值拷贝到一个新的数组中去。 luckNumbers = Arrays.copyOf(luckNumbers, 2 * luckyNUmbers.length);//这种方法通常用来增加数组的大小。如果数组是数值型,那么多余的元素将被赋值为0;布尔型,将赋值为false;如果长度小于原始数组的长度,则只拷贝最前面的数据元素。
-
java中的[]运算符被预定义为检查数组边界,而且没有指针运算,即不能通过a加1得到数组的下一个元素。
public class Message { public static void main(String[] args) { if... } } java Message -g cruel world;//args数组将包含下列内容:args[0]:"-g";args[1]:"cruel";args[2]:"world"
-
在java运行应用程序的应用程序到main方法中,程序名并没有存储在args数组中.例如,当使用下列命令运行程序时.java Message -h world;//args[0]是"-h",而不是"Message"或"java".
-
Arrays.sort(a);//对数值型数组进行排序,使用了优化的快速排序算法.
import java.util.*; /** *3-7 */ public class LotteryDrawing { public static void main(String[] args) { Scanner in = new Scanner(System.in); System.out.print("How many numbers do you need to draw?"); int k = in.nextInt(); System.out.print("What is the highest number you can deaw?"); int n = in.nextInt(); int[] numbers = new int[n]; for(int i = 0; i < numbers.length; i++) { int r = (int) (Math.random() * n); result[i] = number[r]; numbers[r] = numbers[n - 1]; n--; Arrays.sort(result); System.out.println("Bet the following combination.It'll make you rish!"); for(int r : result) System.out.println(r); } } }
API java.util.Arrays 1.2 //参数a 类型为int,long,char,byte,boolean,float或double的数组. //start 包含这个值. //end 不包含这个值. //v 同a的数据元素类型相同的值. static String toString(type[] a);//返回包含a中数据元素的字符串,这些数据元素被放在括号中,并用逗号分隔. static type copyOf(type[] a, int length); static type copyOf(type[] a, int start, int end); static void sort(type[] a); static int binarySearch(type[] a, type v); static int binarySearch(type[] a, int start, int end, type v);//采用二分搜索算法查找v.如果查找成功,则返回相应到下标值;否则,返回一个负数值,-r-1是为保持a有序v应插入到位置. static void fill(type[] a, type v);//将数组的所有数据元素值设置为v. static boolean equals(type[] a, type[] b);//如果两个数组大小相同,并且下标相同的元素都对应相等,返回true.
- 二维数组(3.10.6),要想快速打印二位数组Arrays.deepToString(a);//格式:[[1, 2, 3], [2, 3, 4], [5, 6, 7]]
- 不规则数组:只能单独创建行数组.(3.10.7)
public class LotteryArray { public static void main(String[] args) { final int NMAX = 10; int[][] odds = new int[NMAX + 1][]; for(int n = 0; n <= NMAX; n++) odds[n] = new int[n + 1]; for(int n = 0; n < odds.length; n++) { for(int k = 0;k < odds[n].length; k++) { int lotteryOdds = 1; for(int i = 1; i <= k; i++) lotteryOdds = lotteryOdds * (n - i + 1) / i; odds[n][k] = lotteryOdds; } for(int[] row : odds) System.out.printf("%4d", odd); System.out.println(); } } }
表3-3 特殊字符的转义序列符 图3-1 数值类型之间的合法转换 表3-4 运算符优先级 表3-5 用于printf的转换符 表3-6 用于printf的标志 表3-7 日期和时间的转换符 图3-6 格式说明符语法