C#中ref和out的区别

本文详细解析了C#中的Ref与Out关键字的作用及区别,重点介绍了它们在值类型和引用类型上的不同表现,并通过具体示例展示了如何正确使用这两个关键字。

看了很多.按照 自己明白解释一下

 

.NET Ref和Out关键字

 

对于值类型。

如果不使用ref /out则传递的只是这些值的COPY,

使用了Ref和Out的效果就几乎和C中使用了指针变量一样。(传递的就是原值),它能够让你直接对原数进行操作,而不是对那个原数的Copy进行操作

 

对于引用类型:

如果不使用ref /out,因为传递的是引用类型的地址值,则将传递引用类型的地址值的一个COPY(--针对地址值的角度还是值类型传递),实际上就是新开一个不同的内存变量来存储这个地址值的拷贝

 

而使用ref /out,传递的还是引用类型的地址值,但是传递的就不是一个新变量来存拷贝,而是就是传原来的那个应用类型的地址值

 /////////////////注意list的处理////////不用关心,异步,代理委托这些;可以和使用同步方法一样理解;只是正好遇到了这个例////////////////////////////////////////////////子所以拿来用;

 

public delegate string DelegateWithParameters(string param1, int param2, ArrayList list);

private void CallFooWithParameters()
{
    // create the paramets to pass to the function
    string strParam1 = "Param1";
    int intValue = 100;
    ArrayList list = new ArrayList();
    list.Add("Item1");

    // create the delegate
    DelegateWithParameters delFoo =
        new DelegateWithParameters(FooWithParameters);

    // call  FooWithParameters(string param1,int param2, ArrayList list)

   // 实际上list的引用地址作为值类型传递,进入方法时,用一个新内存变量存储这个引用的地址

   //所以在函数内部多该新内存变量地址修改.只是让该新内存变量指向了另外一个list
    IAsyncResult tag =
        delFoo.BeginInvoke(strParam1, intValue, list, null, null);
   
    // normally control is returned right away,
    // so you can do other work here...

    // calling end invoke to get the return value
    string strResult = delFoo.EndInvoke(tag);

    // write down the parameters:
    Trace.WriteLine("param1: " + strParam1);
    Trace.WriteLine("param2: " + intValue);
    Trace.WriteLine("ArrayList count: " + list.Count);
}

 

private string FooWithParameters(string param1,
        int param2, ArrayList list)
{
    // lets modify the data!
    param1 = "Modify Value for param1";
    param2 = 200;
    list = new ArrayList();
     //list是传进来的那个地址的拷贝;所以只是把这个新的拷贝指向了一个新的new ArralyList()对象

    //所以没有改变原来的值
    return "Thank you for reading this article";
}

 

 

///////console output

param1: Param1
param2: 100
ArrayList count: 1

 

 

 

没有ref时,用个新的变量地址为4000的存储list的地址

 

 

 

 

 

而使用Ref out方式的/////////////////////////

public delegate string DelegateWithOutAndRefParameters(string param1,
                out int param2, ref ArrayList list);

private void CallFooWithOutAndRefParameters()
{
    // create the paramets to pass to the function
    string strParam1 = "Param1";
    int intValue = 100;
    ArrayList list = new ArrayList();
    list.Add("Item1");

    // create the delegate
    DelegateWithOutAndRefParameters delFoo =
      new DelegateWithOutAndRefParameters(FooWithOutAndRefParameters);

    // call the beginInvoke function!
    IAsyncResult tag =
        delFoo.BeginInvoke(strParam1,
            out intValue,
            ref list,
            null, null);

     //使用ref,则在进入方法时,不开新的内存变量传的就是原来list的地址的变量

//所以在函数里,可以修改

    // normally control is returned right away,
    // so you can do other work here...

    // calling end invoke notice that intValue and list are passed
    // as arguments because they might be updated within the function.
    string strResult =
        delFoo.EndInvoke(out intValue, ref list, tag);

    // write down the parameters:
    Trace.WriteLine("param1: " + strParam1);
    Trace.WriteLine("param2: " + intValue);
    Trace.WriteLine("ArrayList count: " + list.Count);
    Trace.WriteLine("return value: " + strResult);
}

private string FooWithOutAndRefParameters(string param1,
        out int param2, ref ArrayList list)
{
    // lets modify the data!
    param1 = "Modify Value for param1";
    param2 = 200;
    list = new ArrayList();
    //
    return "Thank you for reading this article";
}

 

param1: Param1
param2: 200
ArrayList count: 0
return value: Thank you for reading this article

 

 

 

 

 

 

 

 

 用ref的方法:  还是原来的变量地址1000

 

------------------------------------------------

由以上可以看到他们的区别;但又会有个问题了,其实我们平时对使用引用类型,在方法中带ref/out 还是不使用ref/out比如list;对它进行操作,其实都是都可以直接改变原来的list内部的值,为什么呢?

 

因为其实我们平时使用,都是把引用类型传进来,对他进行操作.比如list添加和删除元素.

而但如果对传进来的这个引用地址进行操作改变,则会有区别了

 

   方法中: list = new ArrayList();

对list这个引用地址:

    1.不使用ref/out修饰符:实际是一个新的用来存储地址的内存变量(例子中是4000),对它进行重新分配,并不影响原来的list值,因为原来的list引用地址是1000

   2.如果使用ref/out修饰符,在方法内,因为生成的引用变量还是1000,所以当给list设置新内容,则会对原来的有影响.

 

 

 以上示例引用

http://www.codeproject.com/KB/cs/AsyncMethodInvocation.aspx

这是个分析异步操作很多细节的地方

---------------------------------------------------------

 

 

 

而 Ref和Out关键字的区别:

 

使用Ref型参数时,传入的参数必须先被初始化。而Out则不需要,对Out而言,就必须在方法中对其完成初始化
 

 

首先:两者都是按地址传递的,使用后都将改变原来的数值。很多人在论坛上解释说out是按数值传递,是错误的。简单的测试后可以知道out使用也能改变数值的,所以肯定是按照地址传递的。
其次:ref可以把参数的数值传递进函数,但是out是要把参数清空,就是说你无法把一个数值从out传递进去的,out进去后,参数的数值为空,所以你必须初始化一次。这个就是两个的区别,或者说就像有的网友说的,ref是有进有出,out是只出不进。经典!!!

 

Out更适合用在需要Return多个返回值的地方,而Ref则用在需要被调用的方法修改调用者的引用的时候。

 

<think>好的,我现在需要回答用户关于C#refout关键字的区别及用法的问题。首先,我需要回顾一下已有的引用内容,确保准确理解两者的不同点。根据引用[1],refout都可以用于方法参数,允许方法修改传入的变量。区别在于ref要求参数在传入前必须初始化,而out不需要。此外,在方法内部,out参数必须在返回前赋值,而ref参数可以不用重新赋值。引用[2]补充说,ref是“有进有出”,即参数值可以传入传出,而out只能传出,不能传入,必须在方法内赋值。接下来,我需要整理这些信息,结构清晰地呈现给用户。首先介绍两者的基本概念,然后列出主要区别,如初始化要求、方法内部处理等。接着给出代码示例,帮助用户直观理解。最后,总结使用场景,说明何时选择refout。需要注意使用正确的LaTeX格式,行内公式用$...$,独立公式单独成段。同时生成相关问题,比如性能优化、应用场景等。确保回答符合用户要求,结构清晰,真实可靠。现在开始组织内容,分步骤说明区别用法,并附上示例代码。最后总结要点,并生成相关问题。</think>### C#中`ref`与`out`关键字的区别及用法 #### 1. 核心概念 `ref``out`都是C#中用于方法参数传递的关键字,允许方法修改传入的变量值,实现类似指针的效果。两者的核心区别在于**参数初始化方法内部处理**。 #### 2. 主要区别 | 特性 | `ref`关键字 | `out`关键字 | |--------------------|----------------------------------|--------------------------------| | **初始化要求** | 参数传入前必须初始化 | 参数传入前无需初始化 | | **方法内部赋值要求**| 方法内可不重新赋值 | 方法内必须赋值后才能使用 | | **数据传递方向** | 双向(传入值可被读取修改) | 单向(仅用于输出结果) | #### 3. 代码示例 ```csharp using System; class Program { static void Main() { // ref示例:必须初始化 int a = 10; ModifyWithRef(ref a); Console.WriteLine($"ref结果: {a}"); // 输出11 // out示例:无需初始化 ModifyWithOut(out int b); Console.WriteLine($"out结果: {b}"); // 输出20 } static void ModifyWithRef(ref int x) { x++; // 可直接使用原始值 } static void ModifyWithOut(out int y) { y = 20; // 必须赋值后才能使用 } } ``` #### 4. 使用场景 - **`ref`适用**:需要**保留参数原始值**并可能修改的场景,如交换变量: ```csharp void Swap(ref int x, ref int y) { int temp = x; x = y; y = temp; } ``` - **`out`适用**:需要**返回多个值**的场景,如解析数据: ```csharp bool TryParse(string input, out int result) { return int.TryParse(input, out result); } ``` #### 5. 注意事项 - `ref`参数的行为类似于“按引用传递”,而`out`更强调“输出参数”[^1][^2]。 - 在异步方法或迭代器中,`ref``out`参数的使用会受到限制。 ---
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值