C# 中定義方法
當(dāng)定義一個(gè)方法時(shí),從根本上說是在聲明它的結(jié)構(gòu)的元素。在 C# 中,定義方法的語法如下:
<Access Specifier> <Return Type> <Method Name>(Parameter List)
{
Method Body
}
下面是方法的各個(gè)元素:
- Access Specifier:訪問修飾符,這個(gè)決定了變量或方法對(duì)于另一個(gè)類的可見性。
- Return type:返回類型,一個(gè)方法可以返回一個(gè)值。返回類型是方法返回的值的數(shù)據(jù)類型。如果方法不返回任何值,則返回類型為 void。
- Method name:方法名稱,是一個(gè)唯一的標(biāo)識(shí)符,且是大小寫敏感的。它不能與類中聲明的其他標(biāo)識(shí)符相同。
- Parameter list:參數(shù)列表,使用圓括號(hào)括起來,該參數(shù)是用來傳遞和接收方法的數(shù)據(jù)。參數(shù)列表是指方法的參數(shù)類型、順序和數(shù)量。參數(shù)是可選的,也就是說,一個(gè)方法可能不包含參數(shù)。
- Method body:方法主體,包含了完成任務(wù)所需的指令集。
按引用傳遞參數(shù)
引用參數(shù)是一個(gè)對(duì)變量的內(nèi)存位置的引用。當(dāng)按引用傳遞參數(shù)時(shí),與值參數(shù)不同的是,它不會(huì)為這些參數(shù)創(chuàng)建一個(gè)新的存儲(chǔ)位置。引用參數(shù)表示與提供給方法的實(shí)際參數(shù)具有相同的內(nèi)存位置。
在 C# 中,使用 ref 關(guān)鍵字聲明引用參數(shù)。下面的實(shí)例演示了這點(diǎn):
using System;
namespace CalculatorApplicatio
{
class NumberManipulator
{
public void swap( int a, int b)
{
int temp;
temp = a;
a = b;
b = temp;
}
public void getValue(out int x)
{
int temp = 5;
x = temp;
}
static void Main(string[] args)
{
int a = 100;
int b = 200;
Console.WriteLine("在交換之前的a:{0}",a);
Console.WriteLine("在交換之前的b:{0}",b);
NumberManipulator n = new NumberManipulator();
n.swap( a, b);
Console.WriteLine("在交換之后的a:{0}",a);
Console.WriteLine("在交換之后的b:{0}",b);
Console.WriteLine("在方法調(diào)用之前的a:{0}",a);
n.getValue(out a);
Console.WriteLine("在方法調(diào)用之后的a:{0}", a);
Console.ReadLine();
}
}
}
在交換之前,a 的值:100
在交換之前,b 的值:200
在交換之后,a 的值:200
在交換之后,b 的值:100
結(jié)果表明,swap 函數(shù)內(nèi)的值改變了,且這個(gè)改變可以在 Main 函數(shù)中反映出來。
按輸出傳遞參數(shù)
return 語句可用于只從函數(shù)中返回一個(gè)值。但是,可以使用 輸出參數(shù) 來從函數(shù)中返回兩個(gè)值。輸出參數(shù)會(huì)把方法輸出的數(shù)據(jù)賦給自己,其他方面與引用參數(shù)相似(out關(guān)鍵字)
以下代碼實(shí)現(xiàn)效果:
using System;
namespace CalculatorApplication
{
class NumberManipulator
{
public void getValues( out int x, out int y)
{
Console.WriteLine("請(qǐng)輸入第一個(gè)值: ");
x = Convert.ToInt32(Console.ReadLine());
Console.WriteLine("請(qǐng)輸入第二個(gè)值: ");
y = Convert.ToInt32(Console.ReadLine());
}
static void Main(string[] args)
{
NumberManipulator n = new NumberManipulator();
/* 局部變量定義 */
int a, b;
/* 調(diào)用函數(shù)來獲取值 */
n.getValues( out a, out b);
Console.WriteLine("在方法調(diào)用之后,a 的值: {0}", a);
Console.WriteLine("在方法調(diào)用之后,b 的值: {0}", b);
Console.ReadLine();
}
}
}
一個(gè)用關(guān)鍵字 ref 標(biāo)示,一個(gè)用 out 標(biāo)示。
牽扯到數(shù)據(jù)是引用類型還是值類型。
一般用這兩個(gè)關(guān)鍵字你是想調(diào)用一個(gè)函數(shù)將某個(gè)值類型的數(shù)據(jù)通過一個(gè)函數(shù)后進(jìn)行更改。傳 out 定義的參數(shù)進(jìn)去的時(shí)候這個(gè)參數(shù)在函數(shù)內(nèi)部必須初始化。否則是不能進(jìn)行編譯的。ref 和 out 都是傳遞數(shù)據(jù)的地址,正因?yàn)閭髁说刂?,才能?duì)源數(shù)據(jù)進(jìn)行修改。
一般情況下不加 ref 或者 out 的時(shí)候,傳值類型的數(shù)據(jù)進(jìn)去實(shí)際上傳進(jìn)去的是源數(shù)據(jù)的一個(gè)副本,也就是在內(nèi)存中新開辟了一塊空間,這里面存的值是與源數(shù)據(jù)相等的,這也就是為什么在傳值類型數(shù)據(jù)的時(shí)候你如果不用 return 是無法修改原值的原因。但是你如果用了 ref,或者 out,這一切問題都解決了,因?yàn)樗麄儌鞯氖堑刂贰?/p>
out 比起 ref 來說,還有一個(gè)用法就是可以作為多返回值來用,都知道函數(shù)只能有一個(gè)返回值,C#里,如果你想讓一個(gè)函數(shù)有多個(gè)返回值,那么OUT能很容易解決。
法中參數(shù)的類型有三種
in型參數(shù)
int 型參數(shù)通過值傳遞的方式將數(shù)值傳入方法中,即我們?cè)贘ava中常見的方法。
ref型參數(shù)
該種類型的參數(shù)傳遞變量地址給方法(引用傳遞),傳遞前變量必須初始化。
該類型與out型的區(qū)別在與:
1).ref 型傳遞變量前,變量必須初始化,否則編譯器會(huì)報(bào)錯(cuò), 而 out 型則不需要初始化
2).ref 型傳遞變量,數(shù)值可以傳入方法中,而 out 型無法將數(shù)據(jù)傳入方法中。換而言之,ref 型有進(jìn)有出,out 型只出不進(jìn)。
out 型參數(shù)
與 ref 型類似,僅用于傳回結(jié)果。
注意:
1). out型數(shù)據(jù)在方法中必須要賦值,否則編譯器會(huì)報(bào)錯(cuò)。
eg:如下圖若將代碼中的sum1方法的方法體
改為 a+=b; 則編譯器會(huì)報(bào)錯(cuò)。原因:out 型只出不進(jìn),在沒給 a 賦值前是不能使用的
改為 b+=b+2; 編譯器也會(huì)報(bào)錯(cuò)。原因:out 型數(shù)據(jù)在方法中必須要賦值。
2). 重載方法時(shí)若兩個(gè)方法的區(qū)別僅限于一個(gè)參數(shù)類型為ref 另一個(gè)方法中為out,編譯器會(huì)報(bào)錯(cuò)
eg:若將下面的代碼中將方法名 vsum1 改為 sum(或者將方法名 sum 改為 sum1),編譯器會(huì)報(bào)錯(cuò)。