40

Suppose I have a python function that takes two arguments, but I want the second arg to be optional, with the default being whatever was passed as the first argument. So, I want to do something like this:

def myfunc(arg1, arg2=arg1):
    print (arg1, arg2)

Except that doesn't work. The only workaround I can think of is this:

def myfunc(arg1, arg2=None):
    if arg2 is None:
        arg2 = arg1
    print (arg1, arg2)

Is there a better way to do this?

Ryan C. Thompson
  • 40,856
  • 28
  • 97
  • 159
  • I'm just rethinking this, it might be possible to achieve with a decorator. It's going to be unclear what it does at first glance so probably a bad idea, but it's an idea. – moinudin Jan 01 '11 at 19:39
  • Not directly in the function header line, no. But inside the function body, by testing `if argn is None: argn = `, yes. – smci Jul 18 '19 at 02:09

3 Answers3

30

As @Ignacio says, you can't do this. In your latter example, you might have a situation where None is a valid value for arg2. If this is the case, you can use a sentinel value:

sentinel = object()
def myfunc(arg1, arg2=sentinel):
    if arg2 is sentinel:
        arg2 = arg1
    print (arg1, arg2)

myfunc("foo")           # Prints 'foo foo'
myfunc("foo", None)     # Prints 'foo None'
martineau
  • 119,623
  • 25
  • 170
  • 301
moinudin
  • 134,091
  • 45
  • 190
  • 216
  • 5
    probably better to just default arg2 to None and check that instead of bothering with a custom sentinel object unless you need to handle user provided None different from default – user234461 Mar 07 '18 at 15:31
8

Not really. The other argument names don't exist until the function is actually entered.

Ignacio Vazquez-Abrams
  • 776,304
  • 153
  • 1,341
  • 1,358
4

The second way is fine in my opinion: it is exactly clear what you do, and for all those who will come after you and read the code. While there is a slight overhead in documenting the default behavior, the use of 'None' or any other to make your function generate a default value is very common. Putting logic inside the function calls would certainly not be commendable in my opinion, it could become very complex very soon. So just leave it in the function body, where all the code is.

Inca
  • 1,891
  • 14
  • 15