3.3优先级
最简单的规则就是:先乘除后加减.
3.4赋值和方法调用别名(引用)
3.5操作运算符
一元加,减 和加号,减号拥有同样的符号.编译器能争取的识别.
3.6自加,自减
i++,++i/i--,--i
两种形式的自加,减.i的最后值都会变化.但是在表达式中的左值是不同的.表达式中运算的左值的顺序是这样的.
i++表示 i先赋值,然后i自身+1, 但是与这个马上运算的数是和赋值的先运算.
++i 表示,先自加,然后赋值
以下例子:
private void test() {
int a = 2;
int x = a++ * +a + a;
a = 2;
int y = ++a * +a + a;
System.out.println("x=" + x + " a=" + a);
System.out.println("y=" + y + " a=" + a);
}
//结果 :
x=9 a=3
y=12 a=3
这里面 a++ * +a + a 先a赋值=2,用这个值参与运算;a再自增,第二个变量开始的a=3了
所以表达式 等价与2(+3)+3= 9;
这里的 ++a * +a +a ,a先自加,用这个值进行参与运算,所以一开始 a=3,表达式等价与 3 (+3)+3 =12;
private void test2() {
int x = 1;
System.out.println("x " + x);
System.out.println("++x " + ++x);
System.out.println("x++ " + x++);
System.out.println("x " + x);
System.out.println("--x " + --x);
System.out.println("x-- " + x--);
System.out.println("x " + x);
}
//out
x 1
++x 2
x++ 2
x 3
--x 2
x-- 2
x 1
3.8.1 短路
在逻辑运算符中,如果前面的表达式能够明确无误的判断整个表达式的值,剩余部分的表达式是不会执行的.这就是短路.
false && XXX ==>false true || XXX==>true
3.9直接常量
八进制的数据: 以0开头,里面的数字只能在 0-7之间
private void test3() {
System.out.println("x " + 010);
System.out.println("x " + 0170);
}
//--
x 8
x 120
3.10按位运算符
^(亦或) xor : 数学公式a⊕b = (¬a ∧ b) ∨ (a ∧¬b) 如果a、b两个值不相同,则异或结果为1。如果a、b两个值相同,异或结果为0
按位操作可以与= 联合使用,以便合并运算和赋值 ,例如 &= ,|=, ^= 但是 ~ 不能与=合并,因为~是一元操作.
对于布尔类型值运算,可以进行 & , | , ^ 但是不能进行取反(~), 因为和 ! 冲突.在布尔类型的的运算中,中途不会短路.
private void test4() {
int a = 0x1;
int b = 0x0;
int x = (a | a | a & b);
//这里 可以转化成表达式 y=(a=(a|(a|a&b))) 所以 y=a
int y = (a |= a | a & b);
boolean booleanA = true;
boolean booleanB = booleanA ^ booleanA;
System.out.println("a" + Integer.toBinaryString(a));
System.out.println("b" + Integer.toBinaryString(b));
System.out.println("x" + Integer.toBinaryString(x));
System.out.println("y" + Integer.toBinaryString(y));
}
//output
a 1
b 0
x 1
y 1
3.11 移位操作
左移(<<) 向左移动,右侧补0
右移(>>) 按照右侧制定的位数,向左侧移动.符号位按照”符号扩展”方式:符号为+,如果位数不够在高位插0;若符号为,位数不够在高位插1.
无符号右移(>>>),不管符号位 是正负,位数不够都在高位插0
移位可以与= 组合使用(<<= , >>= ,>>>=).但是”无符号”右移操作时,可能会遇到问题,当short ,byte 值进行以为运算,得到的可能不是正确的值,因为需要转成int ,在进行右移操作,然后被截断,赋值给原来的类型.
3.13 字符串操作 + 和 +=
如果一个表达式是 字符串开头,那么后续的所有操作数都必须是字符串,编译器会自动把双引号里面的字符序列自动转成字符串.但是对于字符串前面的数据,能够按正常的运算处理,后面的数据需要用括号括起来.
private void testStr() {
int x = 10;
int y = 20;
int z = 30;
System.out.println("字符串后的数字 " + x + y);
System.out.println("字符串后的数字 " + (x + y));
System.out.println(x + y + " x+y 字符串前,x+y字符串后 " + x + y);
}