Python syntax for namedtuple

前端 未结 4 1115
情深已故
情深已故 2021-01-12 17:44

I see that the Python syntax for a namedtuple is:

Point = namedtuple(\'Point\', [\'x\', \'y\'])

Why isn\'t it simpler like so:



        
4条回答
  •  不知归路
    2021-01-12 18:28

    In general, objects don't know what variables they are assigned to:

    # Create three variables referring to an OrderedPair class
    
    tmp = namedtuple('OrderedPair', ['x','y'])  # create a new class with metadata
    Point = tmp                                 # assign the class to a variable
    Coordinate = tmp                            # assign the class to another var
    

    That's a problem for named tuples. We have to pass in the class name to the namedtuple() factory function so that the class can be given a useful name, docstring, and __repr__ all of which have the class name inside it.

    These reason it seems strange to you is that normal function and class definitions are handled differently. Python has special syntax for def and class that not only creates functions and classes, but it assigns their metadata (name and docstring) and assigns the result to a variable.

    Consider what def does:

    def square(x):
        'Return a value times itself'
        return x * x
    

    The keyword def takes care of several things for you (notice that the word "square" will be used twice):

    tmp = lambda x: x*x                         # create a function object
    tmp.__name__ = 'square'                     # assign its metadata
    tmp.__doc__ = 'Return a value times itself'
    square = tmp                                # assign the function to a variable
    

    The same is also true for classes. The class keyword takes care of multiple actions that would otherwise repeat the class name:

    class Dog(object):
        def bark(self):
            return 'Woof!'
    

    The underlying steps repeat the class name (notice that the word "Dog" is used twice):

    Dog = type('Dog', (object,), {'bark': lambda self: 'Woof'})
    

    Named tuples don't have the advantage of a special keyword like def or class so it has to do the first to steps itself. The final step of assigning to a variable belongs to you. If you think about it, the named tuple way is the norm in Python while def and class are the exception:

     survey_results = open('survey_results')      # is this really a duplication?
     company_db = sqlite3.connect('company.db')   # is this really a duplication?
     www_python_org = urllib.urlopen('http://www.python.org')
     radius = property(radius)
    

    You are not the first to notice this. PEP 359 that suggested we add a new keyword, make, that could allow any callable to gain the auto-assignment capabilities of def, class, and import.

    make   :
        
    

    would be translated into the assignment:

     = ("", , )
    

    In the end, Guido didn't like the "make" proposal because it caused more problems than it solved (after all, it only saves you from making a single variable assignment).

    Hope that helps you see why the class name is written twice. It isn't really duplication. The string form of the class name is used to assign metadata when the object is created, and the separate variable assignment just gives you a way to refer to that object. While they are usually the same name, they don't have to be :-)

提交回复
热议问题