Skip to content

Commit a06a408

Browse files
committed
[docs update]添加对移位运算符的基本介绍(重要)
1 parent 9f66ae8 commit a06a408

File tree

3 files changed

+76
-5
lines changed

3 files changed

+76
-5
lines changed

docs/java/basis/java-basic-questions-01.md

Lines changed: 74 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -228,6 +228,76 @@ Java 中的注释有三种:
228228
229229
++-- 运算符可以放在变量之前,也可以放在变量之后,当运算符放在变量之前时(前缀),先自增/减,再赋值;当运算符放在变量之后时(后缀),先赋值,再自增/减。例如,当 `b = ++a` 时,先自增(自己增加 1),再赋值(赋值给 b);当 `b = a++` 时,先赋值(赋值给 b),再自增(自己增加 1)。也就是,++a 输出的是 a+1 的值,a++输出的是 a 值。用一句口诀就是:“符号在前就先加/减,符号在后就后加/减”。
230230
231+
### 移位运算符
232+
233+
移位运算符是最基本的运算符之一,几乎每种编程语言都包含这一运算符。移位操作中,被操作的数据被视为二进制数,移位就是将其向左或向右移动若干位的运算。
234+
235+
移位运算符在各种框架以及 JDK 自身的源码中使用还是挺广泛的,`HashMap`(JDK1.8) 中的 `hash` 方法的源码就用到了移位运算符:
236+
237+
```java
238+
static final int hash(Object key) {
239+
int h;
240+
// key.hashCode():返回散列值也就是hashcode
241+
// ^ :按位异或
242+
// >>>:无符号右移,忽略符号位,空位都以0补齐
243+
return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
244+
}
245+
246+
```
247+
248+
在 Java 代码里使用 `<<``>>``>>>`转换成的指令码运行起来会更高效些。
249+
250+
掌握最基本的移位运算符知识还是很有必要的,这不光可以帮助我们在代码中使用,还可以帮助我们理解源码中涉及到移位运算符的代码。
251+
252+
Java 中有三种移位运算符:
253+
254+
- `<<` :左移运算符,向左移若干位,高位丢弃,低位补零。`x << 1`,相当于 x 乘以 2(不溢出的情况下)。
255+
- `>>` :带符号右移,向右移若干位,高位补符号位,低位丢弃。正数高位补 0,负数高位补 1。`x >> 1`,相当于 x 除以 2。
256+
- `>>>` :无符号右移,忽略符号位,空位都以 0 补齐。
257+
258+
由于 `double``float` 在二进制中的表现比较特殊,因此不能来进行移位操作。
259+
260+
移位操作符实际上支持的类型只有`int``long`,编译器在对`short``byte``char`类型进行移位前,都会将其转换为`int`类型再操作。
261+
262+
**如果移位的位数超过数值所占有的位数会怎样?**
263+
264+
当 int 类型左移/右移位数大于等于 32 位操作时,会先求余(%)后再进行左移/右移操作。也就是说左移/右移 32 位相当于不进行移位操作(32%32=0),左移/右移 42 位相当于左移/右移 10 位(42%32=10)。当 long 类型进行左移/右移操作时,由于 long 对应的二进制是 64 位,因此求余操作的基数也变成了 64。
265+
266+
也就是说:`x<<42`等同于`x<<10``x>>42`等同于`x>>10`,``x >>>42`等同于`i4 >>> 10`。
267+
268+
**左移运算符代码示例**
269+
270+
```java
271+
int i = -1;
272+
System.out.println("初始数据: " + i);
273+
System.out.println("初始数据对应的二进制字符串: " + Integer.toBinaryString(i));
274+
i <<= 10;
275+
System.out.println("左移 10 位后的数据 " + i);
276+
System.out.println("左移 10 位后的数据对应的二进制字符 " + Integer.toBinaryString(i));
277+
```
278+
279+
输出:
280+
281+
```
282+
初始数据: -1
283+
初始数据对应的二进制字符串: 11111111111111111111111111111111
284+
左移 10 位后的数据 -1024
285+
左移 10 位后的数据对应的二进制字符 11111111111111111111110000000000
286+
```
287+
288+
由于左移位数大于等于 32 位操作时,会先求余(%)后再进行左移操作,所以下面的代码左移 42 位相当于左移 10 位(42%32=10),输出结果和前面的代码一样。
289+
290+
```java
291+
int i = -1;
292+
System.out.println("初始数据: " + i);
293+
System.out.println("初始数据对应的二进制字符串: " + Integer.toBinaryString(i));
294+
i <<= 42;
295+
System.out.println("左移 10 位后的数据 " + i);
296+
System.out.println("左移 10 位后的数据对应的二进制字符 " + Integer.toBinaryString(i));
297+
```
298+
299+
右移运算符使用类似,篇幅问题,这里就不做演示了。
300+
231301
### continue、break 和 return 的区别是什么?
232302

233303
在循环结构中,当循环条件不满足或者循环次数达到要求时,循环会正常结束。但是,有时候可能需要在循环的过程中,当发生了某种条件之后 ,提前终止循环,这就需要用到下面几个关键词:
@@ -826,6 +896,7 @@ System.out.println(l + 1 == Long.MIN_VALUE); // true
826896
827897
## 参考
828898
829-
- https://stackoverflow.com/questions/1906445/what-is-the-difference-between-jdk-and-jre
830-
- https://www.educba.com/oracle-vs-openjdk/
831-
- https://stackoverflow.com/questions/22358071/differences-between-oracle-jdk-and-openjdk
899+
- What is the difference between JDK and JRE?:https://stackoverflow.com/questions/1906445/what-is-the-difference-between-jdk-and-jre
900+
- Oracle vs OpenJDK:https://www.educba.com/oracle-vs-openjdk/
901+
- Differences between Oracle JDK and OpenJDK:https://stackoverflow.com/questions/22358071/differences-between-oracle-jdk-and-openjdk
902+
- 彻底弄懂Java的移位操作符:https://juejin.cn/post/6844904025880526861

docs/java/concurrent/aqs.md

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,7 @@ protected boolean isHeldExclusively()
116116

117117
`synchronized``ReentrantLock` 都是一次只允许一个线程访问某个资源,而`Semaphore`(信号量)可以用来控制同时访问特定资源的线程数量。
118118

119-
Semaphore 的使用简单,我们这里假设有 N(N>5) 个线程来获取 `Semaphore` 中的共享资源,下面的代码表示同一时刻 N 个线程中只有 5 个线程能获取到共享资源,其他线程都会阻塞,只有获取到贡献资源的线程才能执行。等到有线程释放了共享资源,其他阻塞的线程才能获取到。
119+
Semaphore 的使用简单,我们这里假设有 N(N>5) 个线程来获取 `Semaphore` 中的共享资源,下面的代码表示同一时刻 N 个线程中只有 5 个线程能获取到共享资源,其他线程都会阻塞,只有获取到共享资源的线程才能执行。等到有线程释放了共享资源,其他阻塞的线程才能获取到。
120120

121121
```java 
122122
// 初始共享资源数量

docs/java/concurrent/java-concurrent-questions-03.md

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -363,7 +363,7 @@ protected final boolean compareAndSetState(int expect, int update) {
363363

364364
`synchronized``ReentrantLock` 都是一次只允许一个线程访问某个资源,而`Semaphore`(信号量)可以用来控制同时访问特定资源的线程数量。
365365

366-
Semaphore 的使用简单,我们这里假设有 N(N>5) 个线程来获取 `Semaphore` 中的共享资源,下面的代码表示同一时刻 N 个线程中只有 5 个线程能获取到共享资源,其他线程都会阻塞,只有获取到贡献资源的线程才能执行。等到有线程释放了共享资源,其他阻塞的线程才能获取到。
366+
Semaphore 的使用简单,我们这里假设有 N(N>5) 个线程来获取 `Semaphore` 中的共享资源,下面的代码表示同一时刻 N 个线程中只有 5 个线程能获取到共享资源,其他线程都会阻塞,只有获取到共享资源的线程才能执行。等到有线程释放了共享资源,其他阻塞的线程才能获取到。
367367

368368
```java
369369
// 初始共享资源数量

0 commit comments

Comments
 (0)