C# 常見操作符整理
常見的操作符
賦值操作符:=
復(fù)合賦值運算符:+=,-=,*=,/=,%=
算數(shù)運算符:+,-,*,/,%(ß二元運算符),++,--(ß一元運算符)
條件運算符:>,<,>=,<=,!=,==
邏輯運算符:&&,||,|,!
幾乎所有的操作符都只能操作基本數(shù)據(jù)類型,但是”=”,”==”和”!=”,這些操作符能操作所有的對象。除此以外String類支持”+”和”+=”。
優(yōu)先級
當(dāng)一個表達(dá)式中存在多個操作符時,操作符的優(yōu)先級便決定了表達(dá)式中各部分的運算順序。Java中操作符有自己的一套計算順序,牢記順序可避免程序出現(xiàn)運算錯誤。最簡單的就是先乘除后加減
操作符優(yōu)先級
postfix operators |
[] . (params) expr++ expr-- |
unary operators |
++expr --expr +expr -expr ~ ! |
creation or cast |
new (type)expr |
multiplicative |
* / % |
additive |
+ - |
shift |
<< >> >>> |
relational |
< > <= >= instanceof |
equality |
== != |
bitwise AND |
& |
bitwise exclusive OR |
^ |
bitwise inclusive OR |
| |
logical AND |
&& |
logical OR |
|| |
conditional |
? : |
assignment |
= += -= *= /= %= &= ^= |= <<= >>= >>>= |
注:上面的操作數(shù)優(yōu)先級從上到下依次降低,同一個單元格內(nèi)的優(yōu)先級相同
我們在編程的是候難免會忘記操作符的順序,所以應(yīng)該用括號明確規(guī)定運算順序。
public class Precedence{
public static void main(String [] args){
int x=1,y=2,z=3;
int a=x + y – 2 / 2 + z;
int b=x + ( y – 2 ) / ( 2 + z );
System.out.println(“a =” + a + “b=”+b);
}
}
輸出
a=5 b=1
輸出語句中的+在這種上下文環(huán)境中意味著“字符串連接”,并且如果必要,他還要執(zhí)行“字符串轉(zhuǎn)換”。當(dāng)編譯器觀察到一個String后面進(jìn)梗著一個”+”,而這個”+”的后面又緊跟著一個非String類型的元素時,就會嘗試著將這個非String類型的元素轉(zhuǎn)換成String類型。
賦值
賦值操作符”=”,他的含義是:取右邊的值(右值),然后復(fù)制給左邊(左值)。右值可以是任何常數(shù),變量或者表達(dá)式,或者是任意可以產(chǎn)生值的方法。但左邊必須是一個明確的以命名的變量。也就是說,必須有一個屋里空間可以儲存等號右邊的值。比如可以將一個常數(shù)賦值給一個變量
a = 4
但是不能把任何東西賦值給一個常數(shù),常數(shù)不能作為左值 4 = a不成立。
基本類型數(shù)據(jù)儲存了實際的數(shù)值,而并非指向一個對象的引用,所以在為其賦值的時候,是直接講一個地方的內(nèi)容復(fù)制到了另一個地方。比如基本數(shù)據(jù)類型使用a = b,實際的含義是將b中的內(nèi)容復(fù)制給a,如果之后給a從新賦值b不會受到影響。但是并不是所有的賦值都會達(dá)到這種預(yù)期的效果。
為對象做賦值操作的時候,其實我們操作的是對象的引用,所以若我們將一個對象賦值給另一個對象實際上是將”引用”從一個地方復(fù)制到另一個地方,這就意味著我們在就該其中一個對象值的同時,另一個也會跟這發(fā)生變化。
class Tank{
int level;
}
public class Assignment{
public static void main(String [] args){
Tank t1=new Tank();
Tank t2=new Tank();
t1.level=9;
t2.level=47;
System.out.print(“1: t1.level” +t1.level+”,t2.level”+t2.level);
t1 = t2;
System.out.print(“2: t1.level” +t1.level+”,t2.level”+t2.level);
t1.level=27;
System.out.print(“3: t1.level” +t1.level+”,t2.level”+t2.level);
}
}
輸出
1:t1.level: 9,t2.level:47;
2:t1.level: 47,t2.level:47;
3:t1.level:27,t2.level:27;
在這個例子中就發(fā)生了操作引用的問題,我們在修改t1的同時t2也被修改了,而在大多情況下我們是希望t1與t2可以相互獨立的,但是由于賦值操作的是一個對象的引用,所以這里t1和t2包含的是相同的引用,它們指向相同的對象(原本t1包含的對對象的引用,是一個指向值為9的對象。在t1賦值的時候,這個引用被覆蓋,也就是丟失了;而那個不再被引用的對象會由”垃圾回收器”自動清理。
這種特殊的現(xiàn)象通常稱作”別名現(xiàn)象”,是Java操作對象的一種基本方式。在這個例子中如果想避免別名現(xiàn)在,我們應(yīng)該直接對對象的值進(jìn)行操作:
t1.level=t2.level;
這樣寫便可以保持兩個對象的獨立,但是直接操作對象的域很容易導(dǎo)致混亂,并且,違背了良好的面向?qū)ο蟪绦蛟O(shè)計的原則。
下面的例子介紹了方法調(diào)用中的別名問題
class Letter{
char c;
}
public class PassObject{
static void f(Letter y){
y.c='z';
}
public static void main(String [] args){
Letter x=new Letter();
x.c='a';
System.out.print(“1: x.c ” + x.c);
f(x);
System.out.print(“2: x.c” + x.c);
}
}
輸出
1: x.c: a
2: x.c: z
我們在使用f方法的時候似乎覺得它的作用域內(nèi)復(fù)制其參數(shù)Letter y的一個副本;但實際上只是傳遞了一個引用。所以代碼行
y.c='z';
實際改變的是f()之外的對象。
其他賦值操作符
Operator |
Use |
Equivalent to |
+= |
op1 += op2 |
op1 = op1 + op2 |
-= |
op1 -= op2 |
op1 = op1 - op2 |
*= |
op1 *= op2 |
op1 = op1 * op2 |
/= |
op1 /= op2 |
op1 = op1 / op2 |
%= |
op1 %= op2 |
op1 = op1 % op2 |
&= |
op1 &= op2 |
op1 = op1 & op2 |
|= |
op1 |= op2 |
op1 = op1 | op2 |
^= |
op1 ^= op2 |
op1 = op1 ^ op2 |
<<= |
op1 <<= op2 |
op1 = op1 << op2 |
>>= |
op1 >>= op2 |
op1 = op1 >> op2 |
>>>= |
op1 >>>= op2 |
op1 = op1 >>> op2 |
相關(guān)文章
在c#中使用servicestackredis操作redis的實例代碼
本篇文章主要介紹了在c#中使用servicestackredis操作redis的實例代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06C#實現(xiàn)微信跳一跳小游戲的自動跳躍助手開發(fā)實戰(zhàn)
前段時間微信更新了新版本后,帶來的一款H5小游戲“跳一跳”在各朋友圈里又火了起來,類似以前的“打飛機(jī)”游戲,這游戲玩法簡單,但加上了積分排名功能后,卻成了“裝逼”的地方,于是很多人花錢花時間的刷積分搶排名2018-01-01C#調(diào)用攝像頭實現(xiàn)拍照功能的示例代碼
這篇文章主要介紹了C#調(diào)用攝像頭實現(xiàn)拍照功能的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09C#程序中session值的保存方法以及轉(zhuǎn)為字符串的方法總結(jié)
這篇文章主要介紹了C#程序中session值的保存方法以及轉(zhuǎn)為字符串的方法總結(jié),經(jīng)常被用于ASP.NET網(wǎng)絡(luò)編程項目中,需要的朋友可以參考下2016-04-04