zoukankan      html  css  js  c++  java
  • Python如何通过引用传递变量?

    """
    # 1 Python的函数参数传递
    这里记住的是类型是属于对象的,而不是变量。
    而对象有两种,“可更改”(mutable)与“不可更改”(immutable)对象。在python中,strings, tuples, 
    和numbers是不可更改的对象,而 list, dict, set 等则是可以修改的对象。(这就是这个问题的重点)
    当一个引用传递给函数的时候,函数自动复制一份引用,这个函数里的引用和外边的引用没有半毛关系了.
    所以第一个例子里函数把引用指向了一个不可变对象,当函数返回的时候,外面的引用没半毛感觉.而第二个例子就不一样了,
    函数内的引用指向的是可变对象,对它的操作就和定位了指针地址一样,在内存里进行修改.
    """
    # a = 1
    # def fun(a):
    #     print( "func_in",id(a))   # func_in 1604579120
    #     a = 2
    #     print( "re-point",id(a), id(2))   # re-point 1604579152 1604579152
    #
    # print( "func_out",id(a), id(1))  # func_out 1604579120 1604579120
    # fun(a)
    # print (a)  # 1
    
    """
    输出
    func_out 1604579120 1604579120
    func_in 1604579120
    re-point 1604579152 1604579152
    1
    
    Process finished with exit code 0
    """
    # 所有的变量都可以理解是内存中一个对象的“引用”
    # 可以看到,在执行完a = 2之后,a引用中保存的值,即内存地址发生变化,由原来1对象的所在的地址变成了2这个实体对象的内存地址。
    """
    参数通过赋值传递。这背后的理由是双重的:
    
    传递的参数实际上是一个参考的一个对象(但参考通过值传递)
    一些数据类型是可变的,但其他数据类型则不可变
    所以:
    如果你将一个可变对象传递给一个方法,那么该方法会获得对同一个对象的引用,你可以将它改变,但是如果你在方法中重新引用引用,那么外部范围将对它一无所知,之后你完成后,外部引用仍将指向原始对象。
    如果将不可变对象传递给方法,则仍然无法重新绑定外部引用,甚至无法改变对象。
    为了更清楚,让我们举一些例子。
    """
    """
    列表 - 可变类型
    让我们尝试修改传递给方法的列表:
    """
    # def try_to_change_list_contents(the_list):
    #     print('got', the_list)
    #     the_list.append('four')
    #     print('changed to', the_list)
    #
    # outer_list = ['one', 'two', 'three']
    #
    # print('before, outer_list =', outer_list)
    # try_to_change_list_contents(outer_list)
    # print('after, outer_list =', outer_list)
    
    """
    输出
    before, outer_list = ['one', 'two', 'three']
    got ['one', 'two', 'three']
    changed to ['one', 'two', 'three', 'four']
    after, outer_list = ['one', 'two', 'three', 'four']
    
    """
    
    # 由于传入的参数是outer_list对它的引用,而不是它的副本,我们可以使用变异列表方法来更改它并使更改反映在外部作用域中。
    """
    现在让我们看看当我们尝试更改作为参数传入的引用时会发生什么:
    """
    # def try_to_change_list_reference(the_list):
    #     print('got', the_list)
    #     the_list = ['and', 'we', 'can', 'not', 'lie']
    #     print('set to', the_list)
    #
    # outer_list = ['we', 'like', 'proper', 'English']
    #
    # print('before, outer_list =', outer_list)
    # try_to_change_list_reference(outer_list)
    # print('after, outer_list =', outer_list)
    
    """
    输出
    before, outer_list = ['we', 'like', 'proper', 'English']
    got ['we', 'like', 'proper', 'English']
    set to ['and', 'we', 'can', 'not', 'lie']
    after, outer_list = ['we', 'like', 'proper', 'English']
    """
    # 由于the_list参数是按值传递的,因此为其分配新列表不会影响方法外部的代码。这the_list是outer_list引用的副本,我们the_list指向了一个新列表,但没有办法改变outer_list指向的位置
    """
    字符串 - 不可变类型
    它是不可变的,所以我们无法改变字符串的内容
    现在,让我们尝试更改引用
    """
    
    # def try_to_change_string_reference(the_string):
    #     print('got', the_string)
    #     the_string = 'In a kingdom by the sea'
    #     print('set to', the_string)
    #
    # outer_string = 'It was many and many a year ago'
    #
    # print('before, outer_string =', outer_string)
    # try_to_change_string_reference(outer_string)
    # print('after, outer_string =', outer_string)
    #
    
    """
    输出
    before, outer_string = It was many and many a year ago
    got It was many and many a year ago
    set to In a kingdom by the sea
    after, outer_string = It was many and many a year ago
    """
    # 同样,由于the_string参数是通过值传递的,因此为其分配新字符串不会影响方法外部的代码。这the_string是一个outer_string引用的副本,我们the_string指向一个新的字符串,但没有办法改变outer_string指向的位置。
    a = 1
    a = 2
    
    """
    您认为这a是存储值的内存位置1,然后更新以存储该值2。这不是Python中的工作方式。
    相反,a作为对具有该值的对象的引用开始1,然后被重新分配为具有该值的对象的引用2。
    这两个对象可能会继续共存,即使a不再引用第一个对象; 实际上,它们可能被程序中的任何其他引用共享。
    当您使用参数调用函数时,会创建一个引用传入对象的新引用。这与函数调用中使用的引用是分开的,因此无法更新该引用并使其引用新对象。在你的例子中:
    """
    # def __init__(self):
    #     self.variable = 'Original'
    #     self.Change(self.variable)
    #
    # def Change(self, var):
    #     var = 'Changed'
    
    
    """
    self.variable是对字符串对象的引用'Original'。
    当您调用时Change,创建var对该对象的第二个引用。
    在函数内部,您将引用重新分配var给不同的字符串对象'Changed',但引用self.variable是独立的,不会更改。
    解决这个问题的唯一方法是传递一个可变对象。因为两个引用都引用同一个对象,所以对象的任何更改都会反映在两个位置。
    """
    # def __init__(self):
    #     self.variable = ['Original']
    #     self.Change(self.variable)
    #
    # def Change(self, var):
    #     var[0] = 'Changed'
    
    
    """
    对上面的评论:
    它既不是按值传递,也不是按引用传递 - 它是逐个调用的。请见Fredrik Lundh:
    http://effbot.org/zone/call-by-object.htm
    这是一个重要的引用:
    “......变量[名称] 不是对象;它们不能用其他变量表示或由对象引用。”
    在您的示例中,Change调用方法时- 为其创建名称空间 ; 并var成为该命名空间中字符串对象的名称'Original'。
    然后,该对象在两个名称空间中具有名称。接下来,var = 'Changed'绑定var到一个新的字符串对象,因此该方法的命名空间忘记了'Original'。
    最后,忘记了该命名空间,并将字符串'Changed'与它一起使用。
    """
    # x = [ 2, 4, 4, 5, 5 ]
    # print (x)  # 2, 4, 4, 5, 5
    #
    # def go( li ) :
    #   li = [ 5, 6, 7, 8 ]  # re-assigning what li POINTS TO, does not
    #   # change the value of the ORIGINAL variable x
    #   print("go_li",li)
    # go( x )
    #
    # print (x)  # 2, 4, 4, 5, 5  [ STILL! ]
    #
    #
    # print( 'press any key to continue' )
    
    
    """
    有趣的总结
    事实是,整个参考/值概念将不适合python。Python没有变量的“价值”。Python只有引用对象的对象和名称。
    
    因此,当您调用函数并在括号内放置“名称”时,如下所示:
    
    def func(x): # defines a function that takes an argument
        ... # do something here
    
    func(myname) # calling the function
    myname传递指向的实际对象,而不是名称 myname 本身。在函数内部,给出了另一个name(x)来引用传递的同一个对象。
    
    您可以修改函数内部的对象(如果它是可变的),但您无法更改外部名称指向的对象。就像你做的那样
    
    anothername = myname
    因此,我可以回答你的问题:
    
    它是“按值传递”,但所有值都只是对象的引用。
    
    """
    # a=1
    # print("a",id(a))
    # def nuw_1(data):
    #     print("data",id(data))
    #     data+=1
    #
    #     print("data",id(data))
    #     return data
    #
    # a=nuw_1(a)
    # print("a",id(a))
    
    
    a=[]
    print("a",id(a))
    def nuw_1(data):
        print("data",data,id(data))
        data=1
        print("data",data,id(data))
    
    nuw_1(a)
    print("a",a,id(a))
    
    """
    输出

      a 2072004795720
      data [] 2072004795720
      data 1 1604579120
      a [] 2072004795720

    """
    
    a=[]
    print("a",id(a))
    def nuw_1(data):
        print("data",data,id(data))
        data.append(1)
        print("data",data,id(data))
    
    nuw_1(a)
    print("a",a,id(a))
    """

    输出:

      a 2072004793288
      data [] 2072004793288
      data [1] 2072004793288
      a [1] 2072004793288

    """
  • 相关阅读:
    有一天,我们能这样相爱吗?
    端午节来源六说
    一个ini类代替缓存使用
    创意生活可爱香皂
    漂亮的韩国发饰
    Oracle中PL/SQL单行函数和组函数详解
    真正爱你的女人是这样的
    执子之手,与子偕老。你同意么?
    男人如茶
    Oracle SQL 內置函數大全
  • 原文地址:https://www.cnblogs.com/Xingtxx/p/11044255.html
Copyright © 2011-2022 走看看