# 邏輯運算符
## 邏輯運算符概念
- 有時候,我們需要在多個條件同時成立的時候才能執行某段代碼,比如:用戶只有同時輸入了QQ和密碼,才能執行登錄代碼,如果只輸入了QQ或者只輸入了密碼,就不能執行登錄代碼。這種情況下,我們就要借助于C語言提供的邏輯運算符。
- C語言中??提供了三種邏輯運算符:
+ &&(與運算)
+ ||(或運算)
+ !(非運算)
> + 邏輯運算的結果只有2個:“真”為1,“假”為0
## 邏輯與
- 格式:
+ “條件A && 條件B”
- 運算結果:
+ 只有當條件A和條件B都成立時,結果才為1,也就是“真”;其余情況的結果都為0,也就是“假”。因此,條件A或條件B只要有一個不成立,結果都為0,也就是“假”
+ 口訣:一假則假
- 邏輯與運算過程
+ 總是先判斷條件A是否成立
+ 如果條件A成立,接著再判斷條件B是否成立:如果條件B成立,“條件A && 條件B”的結果就為1,即“真”,如果條件B不成立,結果就為0,即“假”
+ 如果條件A不成立,就不會再去判斷條件B是否成立:因為條件A已經不成立了,不管條件B如何,“條件A && 條件B”的結果肯定是0,也就是“假”
- 運算過程示例:
+ 邏輯與的結合方向是**“從左至右”**。比如表達式 (a>3) && (a<5)
+ 若a的值是4:先判斷a>3,成立;再判斷a<5,也成立。因此結果為1
+ 若a的值是2:先判斷a>3,不成立,停止判斷。因此結果為0
+ 因此,如果a的值在(3, 5)這個范圍內,結果就為1;否則,結果就為0
- 使用注意
+ 若想判斷a的值是否在(3,5)范圍內,千萬不能寫成 `3<a<5`因為關系運算符的結合方向為“從左往右”。
* 例如a為2,它會先算3<a,也就是3<2,條件不成立,結果為0。
* 再與5比較,即0<5,條件成立,結果為1。
* 因此`3<a<5`的結果為1,條件成立,也就是說當a的值為2時,a的值是在(3,5)范圍內的。
* 這明顯是不對的。正確的判斷方法是:(a>3) && (a<5)
+ C語言規定:任何非0值都為“真”,只有0才為“假”。因此邏輯與也適用于數值。比如 5 && 4的結果是1,為“真”;-6 && 0的結果是0,為“假”
## 邏輯或
- 格式
+ “條件A || 條件B”
- 運算結果
+ 當條件A或條件B只要有一個成立時(也包括條件A和條件B都成立),結果就為1,也就是“真”;只有當條件A和條件B都不成立時,結果才為0,也就是“假”。
+ 口訣:一真為真
- 邏輯或運算過程
+ 總是先判斷條件A是否成立
+ 如果條件A成立,就不會再去判斷條件B是否成立:因為條件A已經成立了,不管條件B如何,“條件A || 條件B”的結果肯定是1,也就是“真”
+ 如果條件A不成立,接著再判斷條件B是否成立:如果條件B成立,“條件A||條件B”的結果就為1,即“真”,如果條件B不成立,結果就為0,即“假”
- 運算過程示例
+ 邏輯或的結合方向是**“從左至右”**。比如表達式 (a<3) || (a>5)
+ 若a的值是4:先判斷a<3,不成立;再判斷a>5,也不成立。因此結果為0
+ 若a的值是2:先判斷a<3,成立,停止判斷。因此結果為1
+ 因此,如果a的值在(-∞, 3)或者(5, +∞)范圍內,結果就為1;否則,結果就為0
- 使用注意
+ C語言規定:任何非0值都為“真”,只有0才為“假”。因此邏輯或也適用于數值。比如 5 || 4的結果是1,為“真”;-6 || 0的結果是1,為“真”;0 || 0的結果是0,為“假”
## 邏輯非
- 格式:
+ “! 條件A”
- 運算結果:
+ 其實就是對條件A進行取反:若條件A成立,結果就為0,即“假”;若條件A不成立,結果就為1,即“真”。也就是說:真的變假,假的變真。
+ 口訣:真變假,假變真
- 運算過程示例:
+ 邏輯非的結合方向是**“從右至左”**。比如表達式 ! (a>5)
+ 若a的值是6:先判斷a>5,成立,再取反之后的結果為0
+ 若a的值是2:先判斷a>3,不成立,再取反之后的結果為1
+ 因此,如果a的值大于5,結果就為0;否則,結果就為1
- 使用注意:
+ 可以多次連續使用邏輯非運算符:!(4>2)結果為0,是“假”,!!(4>2)結果為1,是“真”,!!!(4>2)結果為0,是“假”
+C語言規定:任何非0值都為“真”,只有0才為“假”。因此,對非0值進行邏輯非!運算的結果都是0,對0值進行邏輯非!運算的結果為1。!5、!6.7、!-9的結果都為0,!0的結果為1
# 邏輯運算符結合性和優先級
## 優先級
```
!(非) // 注意
↑
算術運算符
↑
關系運算符
↑
&&和|| // 注意
↑
賦值運算符
```
- 邏輯非優先級
```
int result = !0 + 1;
printf("result = %d", result); // 輸出結果: 2
```
>先計算!0等于1
>然后再用計算結果加上1
- 算術運算符與邏輯運算符優先級
```
int result = 3 + 3 && 0 + 1; // (3 + 3) && (0 + 1);
printf("result = %d", result); // 輸出結果: 1
```
>先計算3+3和0+1等到6和1
>然后6&&1
- 關系運算符與邏輯運算符優先級
```
int result = 3>5 || 2<4 && 6<1;
// (3>5) || (2<4) && (6<1)
// 0 || 1 && 0
// 1 && 0
printf("result = %d", result); // 輸出結果: 0
```
>先計算3>5,2<4,6<1得到結果0,1,0
>再進行邏輯運算0||1得到結果1
>再進行邏輯運算1&&0得到結果0
- 練習
```
3+2<5||6>3
4>3 && !-5>2
```
## 短路問題
- 與短路:`&&` 只要第一個條件表達為假那么后面的條件表達就不參與運算了
```
int a = 10;
int b = 20;
int result = (a > 19) && (b++ > 6);
printf("a = %d,b = %d, result = %d\n", a, b, result); // 輸出結果:a = 10,b = 20, result = 0
```
- 或短路:`||` 只要第一個條件表達式為真那么后面的條件表達式就不參與運算了
```
int a = 10;
int b = 20;
int result = (a > 19) || (b++ > 6);
printf("a = %d,b = %d, result = %d\n", a, b, result); // 輸出結果:a = 10,b = 21, result = 1
```