吊打面试官(四)--Java语法基础运算符一文全掌握

createh54周前 (02-13)技术教程16

简介

本文介绍了Java运算符相关知识,包含运算规则,运算符使用经验,特殊运算符注意事项等,全文5400字。熟悉了这些内容,在运算符这块就可以吊打面试官了。

Java运算符的规则与特性


1. 贪心规则(Maximal Munch Rule)


贪心规则是指编译器在解析表达式时,会尽可能多地匹配字符,以形成最长的有效运算符或符号。编译器不会跳过空格合并,因此可以使用空格进行识别。


- **代码示例**:

```java

int a = 10; int b = a+++a; // 解析为 (a++) + a

```代码解释如下:

- 编译器会尽可能多地匹配运算符,因此 `a+++a` 被解析为 `(a++) + a`,而不是 `a + (++a)`。

- 最终结果是 `10 + 11 = 21`。

如何避免语句有歧义?:

1. **使用括号**:明确表达式的优先级,使用()表示优先。

```java int c = (a++) + b; ```


2. **添加空格**:编译器不会跳过空格合并,可以明确分隔运算符。

```java int c = a + ++b; ```


3. **拆分表达式**:将复杂的表达式拆分为多个简单的表达式,避免歧义。

```java int temp = a++; int c = temp + b; ```


2. 运算符优先级(Precedence)


运算符优先级决定了表达式中运算符的执行顺序。优先级高的运算符会先被计算。

- **常见运算符优先级(从高到低)**:

1. `()`(括号)

2. `++`、`--`(后缀递增/递减)、`+`、`-`(一元正负)、`!`、`~`(按位取反)

3. `*`、`/`、`%`(乘法、除法、取模)

4. `+`、`-`(加法、减法)

5. `<<`、`>>`、`>>>`(位移)

6. `<`、`<=`、`>`、`>=`、`instanceof`(比较)

7. `==`、`!=`(相等性比较)

8. `&`(按位与)

9. `^`(按位异或)

10. `|`(按位或)

11. `&&`(逻辑与)

12. `||`(逻辑或)

13. `?:`(三元运算符)

14. `=`、`+=`、`-=`、`*=`、`/=`、`%=`(赋值运算符)

- **示例**:

```java int result = 5 + 3 * 2;

// 3 * 2 先计算,结果是 5 + 6 = 11 ```


3. 运算符结合性(Associativity)


当表达式中出现多个相同优先级的运算符时,结合性决定了它们的计算顺序。结合性可以是左结合(从左到右)或右结合(从右到左)。

左结合运算符:

- 大多数运算符是左结合的,例如 `+`、`-`、`*`、`/`、`%`、`<<`、`>>`、`>>>`、`&`、`^`、`|`、`&&`、`||`。

- **示例**:

```java

int result = 10 - 5 - 2; // 从左到右计算,(10 - 5) - 2 = 3 ```

右结合运算符: - 赋值运算符(`=`、`+=`、`-=` 等)和三元运算符(`?:`)是右结合的。


- **示例**:

```java

int a, b, c; a = b = c = 5; // 从右到左计算,c = 5, b = c, a = b

```

4. 类型转换规则(Type Conversion Rules)

在表达式中,操作数的类型可能会自动转换,以满足运算符的要求。类型转换规则包括隐式转换(自动类型提升)和显式转换(强制类型转换)。


4.1 隐式类型转换(自动类型提升)当操作数的类型不同时,Java会自动将较小的类型提升为较大的类型。

- **规则**: - `byte` -> `short` -> `int` -> `long` -> `float` -> `double` - `char` -> `int`


- **示例**: ```java


int a = 5; double b = 2.5; double result = a + b; // a 被提升为 double,结果是 7.5

```


4.2 显式类型转换(强制类型转换)当需要将较大的类型转换为较小的类型时,必须使用显式类型转换。- **示例**: ```java double a = 5.7; int b = (int) a; // b 的值为 5,小数部分被截断 ```---


5. 短路规则(Short-Circuit Evaluation)

对于逻辑与(`&&`)和逻辑或(`||`)运算符,Java采用短路规则:如果第一个操作数已经能够确定整个表达式的结果,则不会计算第二个操作数。

- **示例**:

```java

int a = 5;

if (a > 0 || a / 0 == 1) {

// 不会抛出异常,因为 a > 0 为 true,短路规则生效

System.out.println("Short-circuit!"); }

```---


6. 赋值运算符的类型转换


复合赋值运算符(如 `+=`、`-=`、`*=` 等)会自动进行类型转换,即使可能导致数据丢失。

- **示例**:

```java

byte b = 10; b += 1000;

// 等价于 b = (byte)(b + 1000); System.out.println(b); // -46(数据丢失) ```


7. 字符串连接规则


`+` 运算符在用于字符串时会被重载,用于字符串连接。如果其中一个操作数是字符串,另一个操作数会被自动转换为字符串。

- **示例**:

```java

String result = "The answer is " + 42;

// "The answer is 42" ```---


8. 三元运算符的类型推断


三元运算符(`?:`)的结果类型是根据两个操作数的类型推断的。如果两个操作数的类型不同,结果类型将是它们的最小公共超类。

- **示例**:

```java

Number result = true ? 1 : 2.0;

// 结果是 Double 类型 System.out.println(result.getClass());

// class java.lang.Double ```


9. 位运算符的特殊规则


位运算符(如 `&`、`|`、`^`、`~`、`<<`、`>>`、`>>>`)直接操作二进制位,适用于整数类型。位运算符直接进行二进制操作,效率极高,常被用于进行特殊计算中。

9.1 按位与(`&`)

- **功能**:对两个操作数的每一位进行与操作,结果为1当且仅当两个位都为1。

- **示例**:

```java

int a = 5; // 二进制:0101

int b = 3; // 二进制:0011

int result = a & b; // 结果:0001 (1)

- **应用场景**:? - 判断特定位是否为1。? - 掩码操作,提取特定位。


9.2 按位或(`|`)

- **功能**:对两个操作数的每一位进行或操作,结果为1当且仅当至少一个位为1。

- **示例**:

```java

int a = 5; // 二进制:0101

int b = 3; // 二进制:0011

int result = a | b; // 结果:0111 (7) ```

- **应用场景**: - 设置特定位为1。 - 组合多个标志位。


9.3 按位异或(`^`)

- **功能**:对两个操作数的每一位进行异或操作,结果为1当且仅当两个位不同。

- **示例**:

```java

int a = 5; // 二进制:0101

int b = 3; // 二进制:0011

int result = a ^ b; // 结果:0110 (6)

```

- **应用场景**: - 交换两个变量的值。 - 切换特定位的状态。


9.4 按位取反(`~`)


- **功能**:对操作数的每一位进行取反操作,1变0,0变1。

- **示例**:

```java

int a = 5; // 二进制:0000 0101

int result = ~a; // 结果:1111 1010 (-6)

```

- **应用场景**: - 反转所有位。 - 计算补码。


9.5 左移(`<<`)

- **功能**:将操作数的二进制表示向左移动指定的位数,右侧空出的位用0填充。

- **示例**:

```java

int a = 5; // 二进制:0101

int result = a << 1; // 结果:1010 (10)

- **应用场景**:? - 快速乘以2的幂。? - 高效处理位掩码。9.6 算术右移(`>>`)- **功能**:将操作数的二进制表示向右移动指定的位数,左侧空出的位用符号位填充。- **示例**:? ```java? int a = -8; // 二进制:11111111111111111111111111111000? int result = a >> 1; // 结果:11111111111111111111111111111100 (-4)? ```- **应用场景**:? - 快速除以2的幂。? - 保留符号位。9.7 逻辑右移(`>>>`)- **功能**:将操作数的二进制表示向右移动指定的位数,左侧空出的位用0填充。- **示例**:? ```java? int a = -8; // 二进制:11111111111111111111111111111000? int result = a >>> 1; // 结果:01111111111111111111111111111100 (2147483644)? ```- **应用场景**:? - 无符号右移,忽略符号位。? - 处理无符号整数。---9.8 位运算符的常见应用场景


快速乘除法

- **左移(`<<`)**:用于快速乘以2的幂。 ```java

int a = 5; int b = a << 1; // 10,相当于 a * 2 ```


- **右移(`>>`)**:用于快速除以2的幂。 ```java

int a = 10; int b = a >> 1; // 5,相当于 a / 2 ```


位掩码操作位运算符常用于位掩码操作,用于选择性地操作特定位。

- **设置位**:

```java

int flags = 0;

flags |= 1 << 2; // 设置第2位

```

- **清除位**:

```java

flags &= ~(1 << 2); // 清除第2位

```


- **切换位**:

```java

flags ^= 1 << 2; // 切换第2位 ```


快速判断奇偶

使用按位与运算符 `&` 可以快速判断一个数是否为偶数。

int a = 5;boolean isEven = (a & 1) == 0; // false,5是奇数```交换两个变量的值使用按位异或运算符 `^` 可以在不使用临时变量的情况下交换两个变量的值。```javaint a = 5, b = 10;a ^= b; b ^= a; a ^= b; // 交换后 a=10, b=5```快速计算绝对值使用位运算可以快速计算一个整数的绝对值。```javaint a = -5;int abs = (a + (a >> 31)) ^ (a >> 31); // 5```---9.9 注意事项- **符号位**:在右移操作中,`>>` 会保留符号位,而 `>>>` 会用0填充。- **数据类型**:位运算符仅适用于整数类型(`byte`、`short`、`int`、`long`)。- **性能优化**:位运算在底层操作中非常高效,但过度使用可能会降低代码的可读性。11. `instanceof` 运算符的继承关系



`instanceof` 不仅可以检查对象是否是某个类的实例,还可以检查对象是否是某个类的子类的实例。

- **示例**:

```java

class Animal {} class Dog extends Animal {}

Animal myDog = new Dog(); System.out.println(myDog instanceof Dog); // true

System.out.println(myDog

instanceof Animal); // true

```


- **解释**:`instanceof` 运算符会检查对象的整个继承链,因此即使对象是子类的实例,它也会返回 `true`。


`instanceof` 使用注意点

`instanceof` 运算符在操作数为 `null` 时返回 `false`,而不是抛出异常。


- **示例**: ```java String str = null; if (str instanceof String) { // 结果为 false // ... } ```

- **解释**:`instanceof` 运算符在操作数为 `null` 时返回 `false`,因此可以安全地用于 `null` 检查。---


12. Lambda表达式与双冒号运算符

Lambda表达式

Java 8引入的Lambda表达式利用 `->` 运算符,使得函数式编程更加简洁。- **示例**: ```java List numbers = Arrays.asList(1, 2, 3, 4, 5); numbers.forEach(x -> System.out.println(x / 100.0)); ```

双冒号运算符

双冒号 `::` 用于方法引用,进一步简化Lambda表达式的写法。- **示例**: ```java List strings = Arrays.asList("1", "2", "3"); List numbers = strings.stream() .map(Integer::parseInt) .collect(Collectors.toList()); ```


Java位运算符使用经验



1. 按位与(`&`):

用于快速判断某个整数的特定位是否为1。 ```java int a = 5; // 0101 boolean isBitSet = (a & 1) != 0; // 判断最低位是否为1 ```2. 按位或(`|`):

用于将多个标志位组合成一个整数。 ```java int READ = 1, WRITE = 2, EXECUTE = 4; int permissions = READ | WRITE; // 组合权限 ```3. 按位异或(`^`):

用于交换两个变量的值。 ```java int a = 5, b = 10; a ^= b; b ^= a; a ^= b; // 交换a和b的值 ```4. 左移(`<<`)和右移(`>>`):

用于快速乘以或除以2的幂。 ```java int a = 8; int b = a << 1; // 16,相当于a * 2 int c = a >> 1; // 4,相当于a / 2 ```5. 使用位运算代替乘除法: ```java int a = 10; int b = a << 1; // 20,相当于a * 2 int c = a >> 1; // 5,相当于a / 2 ```---


Java运算符其他注意点

Java 本身不支持用户自定义的运算符重载,但某些运算符在特定场景下已经“重载”了。例如:

`+` 运算符的重载- **字符串连接**:`+` 运算符在用于字符串时会被重载,用于字符串连接。 ```java String result = "Hello, " + "World!"; // 结果是 "Hello, World!" ```- **数值加法**:`+` 运算符用于数值加法。 ```java int sum = 5 + 3; // 结果是 8 ```


`+=` 运算符的重载- **字符串连接**:`+=` 运算符可以用于字符串连接。 ```java String str = "Hello"; str += " World!"; // 结果是 "Hello World!" ```- **数值加法**:`+=` 运算符用于数值加法。 ```java int a = 5; a += 3; // 结果是 8 ```

`==` 和 `!=` 运算符的重载- **引用比较**:`==` 和 `!=` 运算符用于比较对象的引用。 ```java String s1 = new String("hello"); String s2 = new String("hello"); System.out.println(s1 == s2); // false,比较的是引用 ```- **数值比较**:`==` 和 `!=` 运算符用于比较数值。 ```java int a = 5, b = 5; System.out.println(a == b); // true,比较的是值 ```---


2. 自动装箱与拆箱

Java 提供了自动装箱(Autoboxing)和拆箱(Unboxing)功能,使得基本数据类型和其对应的包装类可以无缝转换。这在使用运算符时非常有用。自动装箱通过 valueOf() 方法实现,自动拆箱通过 xxxValue() 方法实现。自动装箱:


将基本数据类型自动转换为包装类。 ```java Integer a = 5; // 自动装箱,等价于 Integer a = Integer.valueOf(5); ```

自动拆箱:

将包装类自动转换为基本数据类型。 ```java int b = a; // 自动拆箱,等价于 int b = a.intValue(); ```- **运算符中的自动拆箱**: ```java Integer a = 10, b = 20; int sum = a + b; // 自动拆箱后计算 ```---


3. 字符串连接的性能优化


虽然 `+` 运算符可以用于字符串连接,但在大量字符串拼接时,性能较差。Java 提供了 `StringBuilder` 和 `StringBuffer` 来优化字符串拼接。- **使用 `+` 运算符**: ```java String result = "Hello, " + "World!"; // 适用于少量字符串拼接 ```- **使用 `StringBuilder`**: ```java StringBuilder sb = new StringBuilder(); sb.append("Hello, ").append("World!"); String result = sb.toString(); // 适用于大量字符串拼接 ```


总结

Java中的运算符规则包括贪心规则、优先级、结合性、类型转换规则、短路规则等。理解这些规则可以帮助开发者编写更高效、更健壮的代码,并避免因运算符使用不当而导致的错误。

在实际编程中,建议使用括号明确表达式的优先级,以提高代码的可读性和可维护性。同时,掌握运算符的巧妙使用经验可以进一步提升代码的性能和简洁性。

关注一下吧



?

相关文章

挥别代码拼接累,一键生成完整工程代码

在 Java 开发领域,传统的新增接口开发往往伴随着繁琐的代码拼接工作,犹如在荆棘丛中艰难前行,令众多开发者疲惫不堪。从细致入微的接口设计,到严谨复杂的表结构规划,从精准的业务规则编写,再到繁琐的数据...

45 个 Git 经典操作场景,专治不会合代码

文章转载于:江南一点雨git 对于大家应该都不太陌生,熟练使用git已经成为程序员的一项基本技能,尽管在工作中有诸如 Sourcetree这样牛X的客户端工具,使得合并代码变的很方便。但找工作面试和一...

告别繁琐代码拼接,一键完成新增接口开发

在 Java 开发的世界里,你是否还在为新增接口而苦恼?传统的开发方式,需要开发者花费大量时间在代码拼接上,从接口设计到表结构设计,从业务规则实现到数据处理流程,每一个环节都需要精心打磨,繁琐且容易出...

别了,代码拼接苦,一键完成接口开发

曾经,Java 开发的世界里,新增接口就像一场漫长的苦役。开发者们对着空白文档,眉头紧皱,苦思冥想接口的各种细节,然后就开始了那令人抓狂的代码拼接之旅。就好比你要搭建一座超级复杂的乐高城堡,可每一块积...

「Java必修课」Java 8之例说Stream的合并

本篇文章主要介绍几种合并stream的方法,主要有原生JDK的方法和使用第三方库StreamEx和Jooλ的方法。原生JDK原生的JDK 8提供的Stream里的一些静态方法,非常有用,让我们来了解一...

我的世界服务器开服Java版连接教程

本文摘抄莱卡云游戏服务器我的世界教程本教程使用Paper核心开服1、进入控制面板登录面板的信息在绿色的登陆面板按键下方,不是你的莱卡云账号1.2、第一次购买服务器会安装游戏端,大约5分钟左右,如果长时...