Thursday, March 03, 2011

正确理解 C# 中的 ref 关键字 (ZT)

http://www.cnblogs.com/Kellin/archive/2007/09/02/879084.html
 
Summary:
 
很容易理解,如果仅仅是改变传入对象的成员字段,使用和不使用ref是一样的。
而当函数内使用了new重新创建对象,而又想将这种变化带出函数,就需要使用ref了
 
// ----------------------------------------
// MyClass definition
public class MyClass
{
public int Value;
}


// ----------------------------------------
// Tester methods
public static void TestRef(ref MyClass m1)
{
// 这里的 m1 也就相当于大家所说的指向指针的指针:
// m1 指向 Main 中的 m,而 m 则指向那个实际的 MyClass 的实例
// 相当于 C++ 中的
// MyClass** m1 = &m;
// (*m1)->Value = 10;
//
m1.Value = 10;
}

public static void TestNoRef(MyClass m1)
{
// 这里是一个普通的传递引用类型的例子。相当于:
// MyClass m1 = m;
// m1.Value = 20;
//
// m1 复制了 m 中的内容,也就是说现在 m1 也和 m 一样,指向了 m 指向的实例
// 所以这里对 m1 的修改也会影响到 Main 中的 m。
//
m1.Value = 20;
}

public static void TestCreateRef(ref MyClass m1)
{
// 这里的 m1 也是一个指向引用的引用:
// m1 指向 Main 中的 m,而 m 则指向那个实际的 MyClass 的实例
// 相当于 C++ 中的
// MyClass** m1 = &m;
// *m1 = new MyClass();
// (*m1)->Value = 100;
// 在上面的 *m1 = new MyClass() 这个调用的时候,实际上只是将 Main 中 m 的值(引用)给修改了,
// 也就是说现在 m1 指向 Main 中的 m,而 m 现在则指向了这个新生成的实例。
// 所以这里 m1.Value = 100 是会影响到 Main 中的结果的
//
m1 = new MyClass();
m1.Value = 100;
}

public static void TestCreateNoRef(MyClass m1)
{
// 在这个方法里面,我们新申明了一个 MyClass 的实例,而让 m1 指向了这个实例
// 这时候,实际上将 m1 的值修改了, m1 和 Main 中的 m 各自指向不同的实例
// 所以对 m1 做的任何修改都不会影响到 m 了
//
m1 = new MyClass();
m1.Value = 200;
}

public static void Main()
{
MyClass m = new MyClass();
m.Value = 1;

TestRef(ref m);
Console.WriteLine(m.Value);

TestNoRef(m);
Console.WriteLine(m.Value);

TestCreateRef(ref m);
Console.WriteLine(m.Value);

TestCreateNoRef(m);
Console.WriteLine(m.Value);
}

 


 


Final Result:


10


20


100


100

0 Comments:

Post a Comment

<< Home