【问题标题】:Difference between ByVal and ByRef?ByVal 和 ByRef 的区别?
【发布时间】:2011-06-21 23:25:37
【问题描述】:

有什么区别?我总是使用 ByVal,但是,我真的不知道什么时候应该什么时候不应该......

【问题讨论】:

  • 95% 以上的时间您都需要 ByVal,因此您可能一直在做出正确的选择 :) 但是 +1 表示主动了解它。
  • 前言:我不是VB程序员。如果它是 C 或 C++ 之类的东西,如果对象的复制成本很高,则传递 ByVal 可能会很昂贵。如果您知道不打算修改它,ByRef 可能会更快,并且函数的行为会相同。
  • 我在下面评论了同样的事情,但以防万一...... ByVal 创建对象的副本(值类型变量除外)。它创建对同一对象的新引用。你是对的,ByRef 可能更快(不必创建新的引用),但差异充其量是微不足道的。

标签: function vb.net-2010


【解决方案1】:

ByRef = 你给你的朋友你的学期论文(原件),他做了标记,可以退还给你。

ByVal = 你给他一份学期论文,他还给你他的修改,但你必须自己把它们放回原件。

尽可能简单。

为什么要使用 ByRef:
ByRef 会将指针传递给您正在传递的对象。如果您在同一个内存空间中,这意味着只传递“单词”而不是对象。您传递给它的方法可以在原始对象中进行更改,并且根本不需要将它们传递回来,因为它们在原始对象中。有助于加快大数据传递。您还可以使用 ByRef 来允许使用 SUB 而不是 FUNCTION(在 VB 中),因为它不需要传回对象。

为什么不使用 ByRef:
由于该方法可以访问原始方法,因此所做的任何更改都将是即时且永久的。如果该方法失败,则该对象可能已损坏。使用 ByVal 将制作一个副本,将整个副本传递给方法,然后该方法将处理信息并返回一个副本、报告信息或什么都不做。

【讨论】:

  • 我几乎完全不同意。使用 ByVal 不会复制对象。它会创建一个指向 same 对象的新引用。n 您不能对原始引用执行任何操作,但可以修改该对象。尝试将新引用设置为空并不会释放对象,因为先前的引用仍然存在(这使对象在 GC 眼中保持活动状态)
  • 我相信你是对的,但不是对的 :) 对于 By Val,新对象具有你正在调用的过程的范围,并且在过程的范围内是完全可编辑的。一旦处理离开过程,对象就会超出范围并被回收(并且无法使用)。因此,当使用 val 时,它确实会生成您传入的对象的完整副本。通过 ref 将指向的对象传递给对象,并且不会创建对象的额外副本。
  • @TomVandeStouwe:你错了。没有创建新对象。这很容易以多种不同的方式进行测试(从使用一个根本无法多次放入内存的巨大对象,到查看哈希码)。
【解决方案2】:

如果传入引用,当你修改方法中的值时,调用站点中的变量也会被修改。

如果你传递值,它就像在方法中创建另一个变量一样,所以即使你修改它,原始变量(在调用站点)的值也不会改变。

因此,实际上,您通常应该将变量作为值传递。仅在您有明确需要时作为参考传递。

【讨论】:

  • 我认为非常重要的是要注意,虽然原始对象在传递byval时无法更改,但它的子成员可以
  • 确实,Smudge202,这很重要……您可以改变对象成员(或者,换句话说,您可以改变对象)。你不能改变的是引用,这意味着你不能用新对象或 null 替换对象。
  • +1 用于澄清您的答案。另外,请参阅下面我的帖子中的编辑以了解 有趣的
【解决方案3】:

ByRef 就像第二个返回值。它将对对象的引用传递给函数而不是对象本身。如果您在函数中更改ByRef 参数的值,您将在函数结束后看到这些更改。如果这还不够清楚,read thisthis

【讨论】:

    【解决方案4】:

    我知道这个问题已经回答得差不多了,但我只想添加以下内容...

    您传递给函数的对象受 ByRef/ByVal 约束,但是,如果该对象包含对其他对象的引用,则它们可以被调用的方法修改,而不管 ByRef/ByVal。解释不好,我知道,请参阅下面的代码以获得更好的理解:

    Public Sub Test()
        Dim testCase As List(Of String) = GetNewList()
        ByRefChange1(testCase)
        'testCase = Nothing
        testCase = GetNewList()
    
        ByValChange1(testCase)
        'testCase is unchanged
        testCase = GetNewList()
    
        ByRefChange2(testCase)
        'testCase contains the element "ByRef Change 2"
        testCase = GetNewList()
    
        ByValChange2(testCase)
        'testCase contains the element "ByVal Change 2"
    
    End Sub
    
    Public Function GetNewList() As List(Of String)
        Dim result As List(Of String) = New List(Of String)
        result.Add("Value A")
        result.Add("Value B")
        result.Add("Value C")
        Return result
    End Function
    
    Public Sub ByRefChange1(ByRef aList As List(Of String))
        aList = Nothing
    End Sub
    
    Public Sub ByValChange1(ByVal aList As List(Of String))
        aList = Nothing
    End Sub
    
    Public Sub ByRefChange2(ByRef aList As List(Of String))
        aList.Add("ByRef Change 2")
    End Sub
    
    Public Sub ByValChange2(ByVal aList As List(Of String))
        aList.Add("ByVal Change 2")
    End Sub
    

    编辑:

    另外,考虑一下这个函数是否被调用:

    Public Sub ByValChange3(ByVal aList As List(Of String))
        aList.Add("ByVal Change 3")
        aList = New List(Of String)
        aList.Add("ByVal Change 4")
    End Sub
    

    在这种情况下发生的情况是“ByVal Change 3”被添加到调用者列表中,但是在您指定“aList = New List”时,您将新引用指向一个新对象,并变得分离从来电者列表中。既是常识,也可能有一天会被你发现,所以要记住一些事情。

    【讨论】:

      【解决方案5】:

      我希望这能回答你的问题

      Sub last_column_process()
      Dim last_column As Integer
      
      last_column = 234
      MsgBox last_column
      
      trying_byref x:=last_column
      MsgBox last_column
      
      trying_byval v:=last_column
      MsgBox last_column
      
      End Sub
      
      Sub trying_byref(ByRef x)
      x = 345
      End Sub
      
      Sub trying_byval(ByRef v)
      v = 555
      End Sub
      

      【讨论】:

        【解决方案6】:

        认为上一个示例中可能存在拼写错误: 最后一个子应该是“byval”而不是“byref”。 :)

        还在try_byval 中添加了一个msgbox 语句,以便您了解其含义。

        Sub begin()
        Dim last_column As Integer
        
        last_column = 234
        MsgBox "Begin:" & last_column
        
        trying_byref x:=last_column
        MsgBox "byref:" & last_column
        
        trying_byval v:=last_column
        MsgBox "byval:" & last_column
        End Sub
        
        Sub trying_byref(ByRef x)
        x = 111
        End Sub
        
        Sub trying_byval(ByVal v)  '<--not ByRef, that was in sub trying_byref.
        v = 222
        MsgBox "In Here:" & v
        End Sub
        

        【讨论】:

          【解决方案7】:

          ByRef,一个值有2个地址

          所以如果 x=80(80 是值,x 是地址,那么例如变量 y 也可以是 80,因此 80 可以被 x 和 y 访问)

          【讨论】:

            【解决方案8】:

            @Tom 和@kelloti 的回答很有帮助。下面是一个代码示例来进一步说明:

            Private Function ValMessage(ByVal SomeMessage As String)
                SomeMessage = "Val Val Val" ' <-- this variable modification doesn't persist after the function finishes execution
                ValMessage = "Some Return Value"
            End Function
            
            Private Function RefMessage(ByRef SomeMessage As String)
                SomeMessage = "Ref Ref Ref" ' <-- this variable modification persists even after the function finishes execution
                RefMessage = "Some Return Value"
            End Function
            
            Private Sub DoStuff()
                Dim OriginalMessage As String
                Dim OtherMessage As String
                Dim AnotherMessage As String
            
                OriginalMessage = "Original"
                MsgBox ("ORIGINAL: " & OriginalMessage) '--> "Original"
            
                OtherMessage = ValMessage(OriginalMessage)
                MsgBox ("ORIGINAL: " & OriginalMessage) '--> "Original"
            
                AnotherMessage = RefMessage(OriginalMessage)
                MsgBox ("ORIGINAL: " & OriginalMessage) '--> "Ref Ref Ref" <--- this is the difference when you pass a paramter by reference
            End Sub
            

            【讨论】:

              【解决方案9】:

              我会尽量用简单的词来解释区别。

              • 按值传递参数使其仅作为输入参数。这是最安全的方式,因此在 95% 的情况下默认使用。

              • 通过引用传递参数使其既是输入参数又是输出参数。可以在函数内部更改输出参数,这会产生很少使用的副作用。

              【讨论】:

                猜你喜欢
                • 2013-03-16
                • 2012-01-21
                • 1970-01-01
                • 2020-06-30
                • 2015-02-15
                • 1970-01-01
                • 1970-01-01
                • 2018-03-11
                • 1970-01-01
                相关资源
                最近更新 更多