2.1 运算符与表达式
Java提供了丰富的运算符,如算术运算符、关系运算符、逻辑运算符、位运算符等。Java语言中的绝大多数运算符和C语言相同,基本语句如条件分支语句,循环语句等,也和C语言类似。
2.1.1算术运算符与算术表达式
1.加减运算符
加减运算符+、-是二目运算符,即连接两个操作元的运算符。加减运算符的结合方向是从左到右。例如2+3-8,先计算2+3,然后再将得到的结果减8.加减运算符的操作元实证性或浮点型数据,加减运算符的优先级是4级。
2.乘除和求余运算符
乘除和求余运算符*、/、%是二目运算符,结合方向是从左到右,例如2*3/8,先计算2*3,然后再将得到的结果除以8。乘除和求余运算符的操作元是整型或浮点型数据,*、/、%运算符的优先级是3级。
用算术运算符和括号连接起来的符合Java语法规则的式子,称为算术表达式。
2.1.2 自增、自减运算符
自增、自减运算符++、--是单目运算符,可以放在操作元之前,也可以放在操作元之后。操作元必须是一个整型或浮点型变量,作用是使变量的值增1或减1,例如:
++x(--x)表示在使用x之前,先使x的值增(减)1。
x++(x-0)表示在使用x之后,使x的值增(减)1。
粗略来看,++x和x++的作用相当于x=x+1。但++x和x++的不同指出在于,++x实现执行x=x+1再使用x的值,而x++是先使用x的值再执行x=x+1。如果x的原值是5,则对于“y=++x;”,y的值为6,对于“y=x++;”,y的值为5。
2.1.3 算术混合运算的精度
精度从“低”到“高”排列的顺序是:
byte short char int long float double
Java在计算算术表达式的值时,使用下列运算精度规则:
- 如果表达式中有双精度浮点数(double型数据),则按双精度进行运算。例如,表达式5.0/2+10的结果12.5是double型数据。
- 如果表达式中最高精度是单精度浮点数(float型数据),则按单精度进行运算,例如,表达式5.0f/2+10的结果12.5是float型数据。
- 如果表达式中最高精度是long型整数,则按long精度进行运算。例如,表达式12L+100+’a’的结果2009是long型数据。
- 如果表达式中最高精度低于int型整数,则按int精度进行运算。例如,表达式(byte)10+’a’的结果是1007,都是int型数据。
Java允许把不超过byte、short、char的取值范围的算术表达式的值赋给byte、short和char型变量。例如,(byte)30+’a’是结果为127的int型变量。
byte x=(byte)20+'a';
是正确的,但
byte x=(byte)30+'a';
却无法通过编译,编译错误是“可能损失精度,找到int需要byte”,其原因是(byte)30+’b’的结果是int型变量,其值超过了byte变量的取值范围。
2.1.4 关系运算符与关系表达式
关系运算符是二目运算符,用于比较两个值的关系。关系运算符的运算结果是boolean型,当运算符对应的关系成立时,运算结果时true,否则false。例如,10<9的结果是false,5>1的结果是true,3!=5的结果是true,10>20-17的结果为true。
结果为数值型的变量或表达式可以通过关系运算符形成关系表达式。
2.1.5 逻辑运算符与逻辑表达式
逻辑运算符包括&&,||,!。其中&&、||为二目运算符,实现逻辑与、逻辑或;!为单目运算符,实现逻辑非。逻辑运算符的操作元必须是boolean型数据,逻辑运算符可以用来连接关系表达式。
结果为boolean型的变量或表达式可以通过逻辑运算符型岑那个逻辑表达式。
逻辑运算符&&和||也称作短路逻辑运算符,这是因为op1的值是false时,&&运算符子啊进行运算时不再去计算op2的值,直接就得出op1&&op2的结果时false;当op1的值为true时,||运算符在进行运算时不再去计算op2的值,直接就得到op1||op2的结果时true。
2.1.6 赋值运算符与赋值表达式
赋值运算符=是二目运算符,左面的操作元必须是变量,不能是常量或表达式。设x是一个整型变量,y是一个boolean型变量,x=20和y=true都是正确的赋值表达式,赋值运算符的优先级较低,是14级,结合方向是从右往左。
赋值表达式的值就是=左面变量的值。例如:假如a、b是两个int型变量,那么表达式b=12和a=b=100的值分别为12和100。
注意不要将赋值运算符=与等号关系运算符==混淆。
2.1.7 位运算符
整型数据在内存中以二进制的形式表示,例如一个int型变量在内存中占用4个字节共32位,int型数据7的二进制表示是:
00000000000000000000000000000111
左面最高位是符号位,最高位是0表示正数,是1表示负数。负数采用补码表示,例如,-8的补码表示是:
11111111111111111111111111111000
这样就可以对两个整型数据实施位运算,即对两个整型数据对应的位进行运算得到一个新的整型数据。
1.按位与运算
按位与运算符&是双目运算符,对两个整型数据a、b按位进行运算,运算结果是一个整型数据c。运算法则是:如果a、b两个数据对应位都是1,则c的该位是1,否则为0.如果b的精度高于a,那么结果c的精度和b相同。
2.按位或运算
按位或运算符|是二目运算符,对两个整型数据a、b按位进行运算,运算结果是一个整型数据c。运算法则是:如果a、b两个数据对应位都是0,则c的该位是0,否则是1.如果b的精度高于a,那么结果c的精度和b相同。
3.按位非运算
按位非运算符~是单目运算符,对一个整型数据a按位进行运算,运行结果是一个整型数据c。运算法则是:如果a对应位是0,则c的该位是1,否则为0。
4.按位异或运算
按位异或运算符^是一个二目运算符,对两个整型数据a、b按位进行运算,运算结果是一个整型数据c。运算法则是:如果a、b两个数据对应位相同,则c的该位是0,否则是1。如果b的精度高于a,那么结果c的精度和b相同。
由异或运算法则可知:a^a=0,a^0=a。
因此,如果c=a^b,那么a=c^b,也就是说,^的逆运算仍然是^,即a^b^b等于a。
位运算符也可以操作逻辑型数据,法则是:
当a、b都是true时,a&b是true,否则是false。
当a、b都是false时,a|b是false,否则是true。
当a是true时,~a是false;当a是false时,~a是true。
位运算符在操作逻辑型数据时,与逻辑运算符&&、||、!不同的是:位运算符要计算完成a和b的值之后再给出运算的结果。
例:
public class Main {
public static void main(String[] args) {
char a1='+',a2='点',a3='进',a4='攻';
char secret='A';
a1=(char)(a1^secret);
a2=(char)(a2^secret);
a3=(char)(a3^secret);
a4=(char)(a4^secret);
System.out.println("密文:"+a1+a2+a3+a4);
a1=(char)(a1^secret);
a2=(char)(a2^secret);
a3=(char)(a3^secret);
a4=(char)(a4^secret);
System.out.println("原文:"+a1+a2+a3+a4);
//密文:j烸辚敺
//原文:+点进攻
}
}
2.1.8 instanceof运算符
该运算符是二目运算符,左面的操作元是一个对象,右面是一个类。当左面的对象是右面的类或子类创建的对象时,该运算符运算的结果时true,否则是false(后面会详细讲解)。
2.1.9 运算符综述
Java表达式就是用运算符连接起来的符合Java规则的式子。运算符的优先级决定了表达式中运算执行的先后顺序。没必要去记忆运算符的优先级别,在编写程序时尽量地使用括号()运算符号来实现想要地运算次序。
2.2 语句概述
Java里的语句分为以下6类:
1.方法调用语句
如:
System.put.println("hello")
2.表达式语句
由一个表达式构成一个语句,即表达式尾加上分号。例如赋值语句:
x=23;
3.复合语句
可以用{}把一些语句括起来构成复合语句,例如:
{
z=123+x;
System.out.println("How are you");
}
4.空语句
一个分号就是一个语句,称为空语句。
5.控制语句
控制语句分为条件语句、开关语句和循环语句。
6.package语句和import语句
package语句和import语句与类、对象有关。
2.3 if条件分支语句
条件分支语句按语法格式可细分尾三种形式,以下是这三种形式的详细讲解。
2.3.1 if语句
if语句是单条件分支语句,即根据一个条件来控制程序执行的流程。
if语句的语法格式:
if(表达式){
若干语句
}
在if语句中,关键字if后面的一对小括号()内的表达式的值必须是boolean类型,当值尾true时,则执行紧跟着的复合语句,结束当前if语句的执行;如果表达式的值为false,结束当前if语句的执行。
需要注意的是,在if语句中,其中的复合语句中如果只有一条语句,{}可以省略不写,但为了增强程序的可读性最好不要省略(这是一个很好的编程风格)。
例:将变量a、b、c中的数值大小顺序进行互换(从小到大排列)
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
int a=9,b=5,c=7,t=0;
if(b<a){
t=a;
a=b;
b=t;
}
if(c<a){
t=a;
a=c;
c=t;
}
if(c<b){
t=b;
b=c;
c=t;
}
System.out.println("a="+a+",b="+b+",c="+c);//a=5,b=7,c=9
}
}
2.3.2 if-else语句
if-else语句是单条件分支语句,即根据一个条件来控制程序执行的流程。
if-else语句的语法格式:
if(表达式) {
若干语句
}
else{
若干语句
}
在if-else语句中,关键字if后面的一对小括号()内的表达式的值必须是boolean类型,当值为true时,则执行紧跟着的复合语句,结束当前if-else语句的执行;如果表达式的值为false,则执行关键字else后面的复合语句,结束当前if-else语句的执行。
下面是有语法错误的if-else语句。
if(x>0)
y=10;
z=20;
else
y=-100;
正确的写法是:
if(x>0){
y=10;
z=20;
}
else
y=100;
需要注意的是,在if-else语句中,其中的复合语句中如果只有一条语句,{}可以省略不写,但为了增强程序的可读性最好不要省略(这是一个很好的编程风格)。
例:写两个if-else语句,其作用是根据成绩输出相应的信息。
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
int math=65,english=85;
if(math>60){
System.out.println("数学及格了");
}
else{
System.out.println("数学不及格");
}
if(english>90){
System.out.println("英语是优");
}
else{
System.out.println("英语不是优");
}
System.out.println("我在学习if-else语句");
}
}
2.3.3 if-else if-else语句
if-else if-else语句是多条件分支语句,即根据多个条件来控制程序执行的流程。
if-else if-else语句的语法格式:
if(表达式){
若干语句
}
else if(表达式){
若干语句
}
.
.
.
else{
若干语句
}
在if-else if-else语句中,if以及多个else if后面的一对括号()内的表达式的值必须是boolean类型。程序执行if-else if-else时,按该语句中表达式的顺序,首先计算第一个表达式的值,如果计算结果为true,则执行紧跟着的复合语句,结束当前if-else if-else语句的执行,如果计算结果false,则继续计算第二个表达式的值,依此类推,假设计算第m个表达式的值为true,则执行进根着的复合语句,结束之前if-else if -else语句的执行,否则继续计算第m+1个表达式的值,如果所有表达式的值都是false,执行关键字else后面的复合语句,结束当前if-else if-else语句的执行。
if-else if-else语句中的else部分是可选项,如果没有else部分,当所有表达式的值都是false时,结束当前if-else if-else语句的执行(该语句什么都没有做)。
需要注意的时,在if-else if-else语句中,其中的复合语句中如果只有一条语句,{}可以省略不写,但为了增强程序的可读性最好不要省略。
2.4 switch开关语句
switch语句是单条件多分支的开关语句,它的一般格式定义如下(其中break语句是可选的)。
switch(表达式)
{
case 常量值1:
若干个语句
break;
case 常量值2:
若干个语句
break;
...
case 常量值n:
若干个语句
break;
default:
若干语句
switch语句中“表达式“的值可以是byte、short、int、char型;“常量值1”到“常量值n”也是byte、short、int、char型,而且要互不相同。
switch语句首先计算表达式的值,如果表达式的值和某个case后面的常量值相等,就执行该case里的若干语句直到碰到break语句为止。如果某个case中没有使用break语句,一旦表达式的值和该case后面的常量值相等,程序不仅执行该case里的若干语句,而且继续执行后继case里的若干个语句,直到碰到break语句为止。若switch语句中的表达式的值不与任何case的常量值相等,则执行default后面的若干个语句。switch语句中的default是可选的,如果它不存在,并且switch语句中表达式的值不与任何case的常量值相等,那么switch语句就不会进行任何处理。
几个分支语句的共同特点是根据一个条件选择执行一个分支操作,而不是选择执行多个分支操作。在switch语句中,通过合理地使用break语句,可以达到根据一个条件选择执行一个分支操作(一个case)或多个分支操作(多个case)地结果。
例:使用switch语句判断用户从键盘输入地正整数是否为中奖号码。
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
int number=0;
System.out.println("输入正整数(回车确定)");
Scanner reader=new Scanner(System.in);
number=reader.nextInt();
switch(number){
case 9:
case 131:
case 12:
System.out.println(number+"是三等奖");
break;
case 209:
case 596:
case 27:
System.out.println(number+"是二等奖");
break;
case 875:
case 316:
case 59:
System.out.println(number+"是一等奖");
break;
default:
System.out.println(number+"不中奖");
}
}
}
需要情调地是,switch语句中表达式地值是byte、short、int、char型,但不可以是long型数据。如果将例子4中的
int number=0;
更改为:
long number=0;
将导致编译错误。
2.5 循环语句
循环语句是根据条件,要求程序反复执行某些操作,直到程序“满意”为止。
2.5.1 for循环语句
for语句的语法格式:
for(表达式1;表达式2;表达式3){
若干语句
}
for语句由关键字for、一对小括号()中用分号分隔的三个表达式,以及一个复合语句组成,其中的表达式2必须是一个求值为boolean型数据的表达式,而复合语句称为循环体。循环体只有一条语句时,大括号可以省略,但最好不要省略,以便增加程序的可读性。表达式1负责完成变量的初始化;表达式2是值为boolean型的表达式,称为循环条件;表达式3用来修整变量,改变循环条件。for语句的执行规则是:
- 计算表达式1,完成必要的初始化工作。
- 判断表达式2的值,若表达式2的值为true,则进行(3),否则进行(4)。
- 执行循环体,然后计算表达式3,以便改变循环条件,进行(2)。
- 结束for语句的执行。
例:计算8+88+888+8888+…的前12项和
public class Main{
public static void main(String[] args) {
long sum=0,a=8,item=a,n=12,i=1;
for(i=1;i<=n;i++) {
sum=sum+item;
item*=item*10+a;
}
}
}
2.5.2while循环语句
while语句的语法格式:
while(表达式){
若干语句
}
while语句由关键字while、一对括号()中的一个求值为boolean类型数据的表达式和一个复合语句组成,其中的复合语句称为循环体,循环体只有一条语句时,大括号{}可以省略,但最好不要省略,以便增加程序的可读性。表达式称为循环条件。while语句的执行规则是:
- 计算表达式的值,如果该值是true时,就进行(2),否则执行(3)。
- 执行循环体,再进行(1).
- 结束while语句的执行。
2.5.3 do-while循环语句
do-while循环语法格式如下:
do {
若干语句
}while(表达式);
do-while循环和while循环的区别是do-while的循环体至少被执行一次。
例:用while语句计算1+1/2!+1/3!+1/4!+…的前20项和
public class Main{
public static void main(String[] args){
double sum=0,item=1;
int i=1,n=20;
while(i<=n){
sum+=item;
i+=1;
item=item*(1.0/1);
}
System.out.print(“sum=”=sum);
}
}
2.6 break和continue语句
break和continue语句是用关键字break和continue加上分号构成的语句,例如:
break;
在循环体中可以使用break语句和continue语句。在一个循环中,例如循环50次的循环语句中,如果在某次循环中执行break语句,那么整体循环语句就结束。如果在某次循环中执行了continue语句,那么本次循环就结束,即不再执行本次循环中循环体中的continue语句后面的语句,而转入进行下一次循环。
例:使用了break和continue语句
public class Main{
public static void main(String [] args){
int sum=0,i,j;
for(i=1;i<=10;i++){
if(i%2==0){
//计算1+3+5+7+9
continue;
}
sum+=i;
}
System.out.println("sum+"+sum);
for(j=2;j<=100;j++){
//求100以内的系数
for(i=2;i<=j/2;i++){
if(j%i==0){
break;
}
}
if(i>j/2){
System.out.println(" "+j+"是素数");
}
//sum+25
// 2是素数
// 3是素数
// 5是素数
// 7是素数
// 11是素数
// 13是素数
// 17是素数
// 19是素数
// 23是素数
// 29是素数
// 31是素数
// 37是素数
// 41是素数
// 43是素数
// 47是素数
// 53是素数
// 59是素数
// 61是素数
// 67是素数
// 71是素数
// 73是素数
// 79是素数
// 83是素数
// 89是素数
// 97是素数
}
}
}
2.7 for语句及数组
for语句的功能给予扩充、增强,以便更好地遍历数组。调用格式如下:
for(声明循环变量:数组的名字){
...
}
其中,声明的循环变量的类型必须和数组的类型相同。这种形式的for语句类似自然语言中的“for each”语句,为了便于理解上述for语句,可以将这种形式的for语句翻译为“对于循环变量依次取数组的每一个元素的值”。
例:分别使用for语句的传统方式和改进方式遍历数组
public class Main{
public static void main(String[] args){
int a[]={1,2,3,4};
char b[]={'a','b','c','d'};
for(int n=0;n<a.length;n++){
//传统方式
System.out.println(a[n]);
}
for(int n=0;n<b.length;n++){
//传统方式
System.out.println(b[n]);
}
for(int i:a){
//循环变量i依次取数组a的每一个元素的值(改进方式)
System.out.println(i);
}
for(char ch:b){
//循环变量ch依次取数组b的每一个元素的值(改进方式)
System.out.println(ch);
}
//1
//2
//3
//4
//a
//b
//c
//d
//1
//2
//3
//4
//a
//b
//c
//d
}
}
需要特别注意的是:for(声明循环变量:数组的名字)中的“声明循环变量”必须是变量声明,不可以使用已经声明过的变量。例如,上述例子中第三个for语句不可以如下分开写成一条变量声明和一条for语句:
int i=0;//变量声明
for(i:a) {//for语句
System.out.println(i);
}