java與c#的區(qū)別、兩者有什么不同?
Java 的設(shè)計(jì)者是因?yàn)橛憛扖++的復(fù)雜,于是Java 非常簡(jiǎn)潔,GC 也讓內(nèi)存管理非常方便,C# 是看中了Java 的GC,和虛擬機(jī)技術(shù),希望把微軟的幾大語(yǔ)言集成到.NET 上來(lái)。 因此C#從語(yǔ)言上來(lái)講并不簡(jiǎn)單甚至可以算的上復(fù)雜。
兩種語(yǔ)言的設(shè)計(jì)思路也不一樣,Java 是編譯解釋語(yǔ)言,C#是編譯然后編譯運(yùn)行語(yǔ)言。Java 沒(méi)有委托,C# 有委托。Java 傾向于用Interface 實(shí)現(xiàn)委托的功能,而 在C# 中,Abstract Class 比Interface 發(fā)揮了更大功能。
Java 遵循camel 命名規(guī)則,C#遵循Pascal 命名規(guī)則。但是如今越來(lái)越多的Java 人開(kāi)始使用C#, 同時(shí)也把camel 命名規(guī)則帶到了C#,這可能會(huì)讓C#代碼越來(lái)越難讀。為什么當(dāng)初C#為什么不遵循camel 呢? 我看不出camel 命名規(guī)則有什么不好的。
一. 類(lèi)名.this 與內(nèi)部類(lèi)
在 java 中,經(jīng)??吹筋?lèi)似類(lèi)名.this 的用法,this 就是當(dāng)前對(duì)象實(shí)例,為什么前面會(huì)出現(xiàn)類(lèi)名呢?對(duì)此 C# 程序員會(huì)很納悶。
在 Java 中,很多地方使用到了內(nèi)部類(lèi),甚至可以在在內(nèi)部類(lèi)中訪(fǎng)問(wèn)外部類(lèi)中的成員,這個(gè)時(shí)候,在內(nèi)部類(lèi)中使用 this 的時(shí)候,就會(huì)出現(xiàn) this 到底是誰(shuí)的問(wèn)題,到底是表示內(nèi)部類(lèi)的當(dāng)前對(duì)象實(shí)例,還是外部類(lèi)的當(dāng)前對(duì)象實(shí)例問(wèn)題。
在 Java 中,通過(guò)在 this 的前面加上外部類(lèi)的類(lèi)名,表示在內(nèi)部類(lèi)中使用外部類(lèi)的當(dāng)前對(duì)象實(shí)例。
我們看下面的一個(gè)例子。
package core.sisheng;
// 外部類(lèi)定義
public class OuterClass {
// 內(nèi)部類(lèi)定義
private class InnerClass
{
// 內(nèi)部類(lèi)中沒(méi)有定義 id 成員,這里訪(fǎng)問(wèn)外部類(lèi)中的成員
public int getId(){ return OuterClass.this.id; }
public void setId(int id) { OuterClass.this.id = id;}
// 內(nèi)部類(lèi)中定義了 name 成員,直接訪(fǎng)問(wèn)內(nèi)部類(lèi)中的成員, 默認(rèn) this 訪(fǎng)問(wèn)當(dāng)前類(lèi)中的成員
private String name;
public String getName() { return this.name;}
// 可以在 this 的前面加上一個(gè)內(nèi)部類(lèi)的名稱(chēng)
public void setName(String name) { InnerClass.this.name = name;}
// 內(nèi)部類(lèi)中也可以訪(fǎng)問(wèn)外部類(lèi)中同名的成員,需要加上外部類(lèi)的名稱(chēng)
public String getOuterName() { return OuterClass.this.name;}
public void setOuterName(String name) { OuterClass.this.name = name;}
@Override
public String toString()
{
return "Id: " + this.getId() + ", Inner Name: " + this.getName() + ", Outer Name: " + this.getOuterName();
}
}
// 外部類(lèi)中定義的成員 id 和 name
private int id;
private String name;
private InnerClass innerInstance;
public OuterClass()
{
this.innerInstance = new InnerClass();
this.innerInstance.setId(20);
this.innerInstance.setName("Tom");
this.innerInstance.setOuterName("Alice");
}
public String toString()
{
return this.innerInstance.toString();
}
}
在C#中,類(lèi)區(qū)分為嵌套類(lèi)和非嵌套類(lèi),前者是聲明在其他數(shù)據(jù)類(lèi)型內(nèi)部的類(lèi)。后者是直接定義在某一個(gè)命名空間的類(lèi)。C# 中很少定義嵌套類(lèi)。
非內(nèi)嵌類(lèi)只允許使用public和internal的訪(fǎng)問(wèn)控制,而內(nèi)置類(lèi)則允許使用所有的五種訪(fǎng)問(wèn)控制符,private, protected , internal protected,public和internal。內(nèi)部類(lèi)也可以訪(fǎng)問(wèn)外部類(lèi)的所有方法,包括instance方法和private方法,但是需要顯式 的傳遞一個(gè)外部類(lèi)的實(shí)例。
C#中的內(nèi)部類(lèi)能夠使用外部類(lèi)定義的類(lèi)型和靜態(tài)方法,但是不能直接使用外部類(lèi)的實(shí)例方法,因此,不存在上面的問(wèn)題。
在C#中,外部類(lèi)對(duì)于內(nèi)部類(lèi)的作用更像是一個(gè)命名空間,只要訪(fǎng)問(wèn)控制允許,就可以使用下面的方法創(chuàng)建內(nèi)部類(lèi)對(duì)象實(shí)例。
OuterClass.InnerClass obj = new OuterClass.InnerClass();這個(gè)實(shí)例與外部類(lèi)的任何實(shí)例沒(méi)有任何直接的關(guān)系。類(lèi)似于Java中的靜態(tài)內(nèi)部類(lèi)。
二、類(lèi)名.class 與類(lèi)型
在 java 中還經(jīng)??吹筋?lèi)名.class 的用法,這個(gè)用法相當(dāng)于 C# 中的 typeof( 類(lèi)名 ),用來(lái)獲取類(lèi)型的類(lèi)型對(duì)象實(shí)例引用。
java中,每個(gè)class都有一個(gè)相應(yīng)的Class對(duì)象,當(dāng)編寫(xiě)好一個(gè)類(lèi),編譯完成后,在生成的.class文件中,就產(chǎn)生一個(gè)Class對(duì)象,用來(lái)表示這個(gè)類(lèi)的類(lèi)型信息。獲得Class實(shí)例的三種方式:
通過(guò)調(diào)用對(duì)象實(shí)例的 getClass() 方法獲取該對(duì)象的Class實(shí)例。
使用Class的靜態(tài)方法forName(),用類(lèi)的名字獲取一個(gè)Class實(shí)例。Class.forName(xxx.xx.xx) 返回的是一個(gè)類(lèi), 作用是要求JVM查找并加載指定的類(lèi),也就是說(shuō)JVM會(huì)執(zhí)行該類(lèi)的靜態(tài)代碼段。
類(lèi)名.calss的方式獲取Class實(shí)例,對(duì)基本數(shù)據(jù)類(lèi)型的封裝類(lèi),還可以采用.TYPE來(lái)獲取對(duì)應(yīng)的基本數(shù)據(jù)類(lèi)型的Class實(shí)例。
C# 中獲取類(lèi)型對(duì)象實(shí)例的方式更加簡(jiǎn)單明確一些。
通過(guò)調(diào)用數(shù)據(jù)實(shí)例的 GetType() 方法獲取,這個(gè)方法繼承自O(shè)bject,所以C#中任何對(duì)象都具有GetType()方法,x.GetType(),其中x為變量名。
typeof(x)中的x,必須是具體的類(lèi)名、類(lèi)型名稱(chēng)等,不可以是變量名稱(chēng)。
通過(guò) System.Type 的靜態(tài)方法 System.Type.GetType()。
三、匿名類(lèi)
在 java 中,匿名類(lèi)的使用也比較多,比如在 Android 中,實(shí)現(xiàn)按鈕的監(jiān)聽(tīng),經(jīng)常會(huì)看到類(lèi)似這樣的代碼。
this.listener0 = new OnClickListener() {
@Override
public void onClick(View arg0) {
Intent intent = new Intent( MainActivity.this, ActivityFrameLayout.class);
setTitle("FrameLayout");
startActivity( intent );
}
};
在這里,OnClickListenter 實(shí)際上是一個(gè)接口,接口能用來(lái)創(chuàng)建對(duì)象實(shí)例嗎?當(dāng)然不能。
所以,java 在這里自動(dòng)創(chuàng)建一個(gè)實(shí)現(xiàn)接口的匿名類(lèi),我們創(chuàng)建出來(lái)的實(shí)際上就是這個(gè)匿名類(lèi)的對(duì)象實(shí)例。
這樣做的好處就是我們沒(méi)有必須再定義一個(gè)只使用一次的類(lèi),然后再通過(guò)這個(gè)類(lèi)創(chuàng)建對(duì)象實(shí)例,簡(jiǎn)化了程序的開(kāi)發(fā)。
比如,我們有下面的一個(gè)接口。
public interface myInterface {
void onClick();
}
就可以通過(guò)接口創(chuàng)建一個(gè)實(shí)現(xiàn)接口的匿名類(lèi)的對(duì)象實(shí)例,然后使用這個(gè)對(duì)象實(shí)例。
myInterface instance = new myInterface(){
@Override
public void onClick()
{
System.out.println("Clicked!");
}
};
instance.onClick();在 C# 中,我們根本就不會(huì)使用這種形式,通過(guò)委托,可以非常簡(jiǎn)單地實(shí)現(xiàn)同樣的功能。
注意,java 中是沒(méi)有委托的。
如果我們輸出一下這個(gè)實(shí)例的類(lèi)型,你會(huì)看到這個(gè)匿名類(lèi)的實(shí)際類(lèi)型的。
System.out.println( instance.getClass());
// class core.sisheng.Study1$1四、屬性(Property)
屬性的概念對(duì)大家來(lái)說(shuō)應(yīng)該是很熟悉的,類(lèi)成員函數(shù)可以自由地訪(fǎng)問(wèn)本類(lèi)中的任何屬性成員。不過(guò)若要從一個(gè)類(lèi)中去訪(fǎng)問(wèn)另一個(gè)類(lèi)中的屬性,那就比較麻煩了,所以很多時(shí)候我們使用Getxxx和Setxxx方法,這樣看起來(lái)顯得極不自然,比如用Java或者C++,代碼是這樣的:
foo.setSize (getSize () + 1);
label.getFont().setBold (true);
但是,在C#中,這樣的方法被“屬性化”了。同樣的代碼,在C#就變成了:
foo.size++;
label.font.bold = true;
可以看出來(lái),C#顯然更容易閱讀和理解。我們從這個(gè)“屬性方法”的子程序代碼中,也可以看到類(lèi)似情況:
Java/C++:
public int getSize()
{
return size;
}
public void setSize (int value)
{
size = value;
}
C#:
public int Size
{
get{return size;}
set{size = value;}
}
為了區(qū)分這種屬性化的方法和類(lèi)的屬性成員,在C#中把屬性成員稱(chēng)作“域(field)”,而“屬性”則成為這種“屬性化的方法”專(zhuān)用的名詞。順便說(shuō)一句,其實(shí)這樣的屬性化方法在VB和DELPHI中是經(jīng)常碰到的,在VB中它也就叫屬性。另外,在C#中Get和Set必須成對(duì)出現(xiàn),一種屬性不能只有Get而沒(méi)有Set(在Java和C++中就可以只有Get或者只有Set),C#中這樣做的好處在于便于維護(hù),假如要對(duì)某種屬性進(jìn)行修改,就會(huì)同時(shí)注意Get和Set方法,同時(shí)修改,不會(huì)改了這個(gè)忘了那個(gè)。
五、對(duì)象索引機(jī)制(Indexer)
C#中引入了對(duì)象索引機(jī)制。說(shuō)得明白點(diǎn),對(duì)象索引其實(shí)就是對(duì)象數(shù)組。這里和上一節(jié)中的屬性聯(lián)系起來(lái)講一下,屬性需要隱藏Get和Set方法,而在索引機(jī)制中,各個(gè)對(duì)象的Get或者Set方法是暴露出來(lái)的。比如下面的例子就比較清楚地說(shuō)明了這一點(diǎn)。以上介紹C#和Java不同點(diǎn)
public class Skyscraper
{
Story[] stories;
public Story this [int index] {
get {
return stories [index];
}
set {
if (value != null) {
stories [index] = value;
}
}
}
...
}
以上介紹C#和JAVA得不同點(diǎn),希望對(duì)你了解C#和JAVA有所幫助。
相關(guān)文章
分享幾個(gè)快速提升工作效率的小工具(Listary等)
這篇文章主要介紹了分享幾個(gè)快速提升工作效率的小工具(Listary等),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04
HTTP中ETag語(yǔ)法及使用實(shí)戰(zhàn)詳解
這篇文章主要為大家介紹了HTTP中ETag語(yǔ)法及使用實(shí)戰(zhàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03
ffmpeg播放器實(shí)現(xiàn)詳解之框架搭建過(guò)程
這篇文章主要介紹了ffmpeg播放器實(shí)現(xiàn)詳解之框架搭建過(guò)程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07
密碼哈希函數(shù) Bcrypt的最大密碼長(zhǎng)度限制詳解
這篇文章主要介紹了密碼哈希函數(shù) Bcrypt的最大密碼長(zhǎng)度限制詳解的相關(guān)資料,需要的朋友可以參考下2017-03-03
淺談服務(wù)發(fā)現(xiàn)和負(fù)載均衡的來(lái)龍去脈
單機(jī)時(shí)代,傳統(tǒng)軟件大多是單體/巨石架構(gòu)(Monolithic)。大家往一個(gè)代碼倉(cāng)庫(kù)提交CODE,這會(huì)導(dǎo)致應(yīng)用膨脹,以及擴(kuò)展受限,無(wú)法按需伸縮等諸多問(wèn)題。單體架構(gòu)怎么解決多人合作的問(wèn)題?模塊化,按功能拆分,模塊之間定義編程接口(API)。本篇文章帶你詳細(xì)了解。2021-05-05
electron桌面應(yīng)用程序搭建及簡(jiǎn)單運(yùn)行
這篇文章主要介紹了electron桌面應(yīng)用程序搭建及運(yùn)行,需要的朋友可以參考下2022-12-12

