How can you print a variable name in python? [duplicate]
Asked Answered
P

8

137

Say I have a variable named choice it is equal to 2. How would I access the name of the variable? Something equivalent to

In [53]: namestr(choice)
Out[53]: 'choice'

for use in making a dictionary. There's a good way to do this and I'm just missing it.

EDIT:

The reason to do this is thus. I am running some data analysis stuff where I call the program with multiple parameters that I would like to tweak, or not tweak, at runtime. I read in the parameters I used in the last run from a .config file formated as

filename
no_sig_resonance.dat

mass_peak
700

choice
1,2,3

When prompted for values, the previously used is displayed and an empty string input will use the previously used value.

My question comes about because when it comes to writing the dictionary that these values have been scanned into. If a parameter is needed I run get_param which accesses the file and finds the parameter.

I think I will avoid the problem all together by reading the .config file once and producing a dictionary from that. I avoided that originally for... reasons I no longer remember. Perfect situation to update my code!

Plonk answered 26/2, 2009 at 22:26 Comment(8)
python.net/~goodger/projects/pycon/2007/idiomatic/…Paravane
The link provides some pictures that in 2 minutes will teach you what are names in Python.Paravane
@dF: This question is different. It is not about enums.Paravane
@Sebastian, I've had the Idiomatic Python on my toolbar for a long time. Those pictures taught me how Python names variables the first time I read it. I understand how it does it and how it may be difficult to do what I ask because of it, but is there a good way to do it I ask.Plonk
ConfigParser (from @Alabaster Codify's answer) allows you to produce a dictionary from a config file (in .ini format) in one line.Paravane
also related: #933318Belly
There is little worse on Stack Overflow than people questioning the existence of a question itself.Selfimportant
Finally, a perfectly valid solution with Python 3.8 f-string = syntax : https://mcmap.net/q/37538/-print-a-variable-39-s-name-and-valueTomika
G
155

If you insist, here is some horrible inspect-based solution.

import inspect, re

def varname(p):
  for line in inspect.getframeinfo(inspect.currentframe().f_back)[3]:
    m = re.search(r'\bvarname\s*\(\s*([A-Za-z_][A-Za-z0-9_]*)\s*\)', line)
    if m:
      return m.group(1)

if __name__ == '__main__':
  spam = 42
  print varname(spam)

I hope it will inspire you to reevaluate the problem you have and look for another approach.

Gilligan answered 26/2, 2009 at 23:5 Comment(6)
There is related use case for inspect.currentframe().f_back.f_localsParavane
Sometimes a night of debugging makes us do things we're not proud of.Fujimoto
upvoted for literal LOLEdlyn
I like how Jake's comment comes 5 years after the question and yet it still has 70+ up votes of people that, like me, reluctantly searched how to do this shameless thing in a moment of desperationEllie
It doesn't work with list comprehension a,b=1,2; print([varname(x) for x in [a,b]]) gives ['x', 'x']Washedup
Yes it does. When it is passed to the function varname the variable name is indeed x.Karolkarola
P
118

To answer your original question:

def namestr(obj, namespace):
    return [name for name in namespace if namespace[name] is obj]

Example:

>>> a = 'some var'
>>> namestr(a, globals())
['a']

As @rbright already pointed out whatever you do there are probably better ways to do it.

Paravane answered 26/2, 2009 at 23:14 Comment(9)
This fails if there is more than one variable name referring to the same value.Internalize
@Greg Hewgill: You might have noticed that namestr returns list. It is a hint that there could be more than one name.Paravane
@Greg Hewgill: It doesn't matter in this case (due to strings are immutable) but generally there is a difference between object's value and its identity. Therefore it is better to say that a name refers to an object (not its value). Mutable objects may change their values, but not identity id().Paravane
Ok, I seem to have missed the list. Fair enough, but integers don't always share representation, see: #306813Internalize
I'd say most integers don't share their representations.Paravane
I like the flow of your answer code, but I may be working with integers that may be repeated several times in the code... sighPlonk
@J.F.Sebastian I'd say most integers don't share their representations I'm pretty sure there is the same amount of each of these two kinds :) Both sets are countable.Fakery
@Piotr Dobrogost: "share their representions" refers to CPython optimization that caches small integers therefore one of the sets is not countable it is finite.Paravane
I like this solution the best, but did not like how '_' or '_70' would also show up. Unless you are searching fro private objects this provides a bit cleaner output. ``` def var_name(obj, namespace=globals()): return [name for name in namespace if namespace[name] is obj and name[0] != '_'][0] ```Deborahdeborath
A
18

If you are trying to do this, it means you are doing something wrong. Consider using a dict instead.

def show_val(vals, name):
    print "Name:", name, "val:", vals[name]

vals = {'a': 1, 'b': 2}
show_val(vals, 'b')

Output:

Name: b val: 2
Acquainted answered 26/2, 2009 at 22:52 Comment(3)
+1: The original question is not sensible. It's a simple dictionary. Or -- perhaps -- use a different language.Graehme
Lets say I have choice = 'He chose 3!'. Can I add this to a dict without typing out "choice" when specifying the key?Plonk
key = "choice"; print vals[key]Acquainted
K
11

Rather than ask for details to a specific solution, I recommend describing the problem you face; I think you'll get better answers. I say this since there's almost certainly a better way to do whatever it is you're trying to do. Accessing variable names in this way is not commonly needed to solve problems in any language.

That said, all of your variable names are already in dictionaries which are accessible through the built-in functions locals and globals. Use the correct one for the scope you are inspecting.

One of the few common idioms for inspecting these dictionaries is for easy string interpolation:

>>> first = 'John'
>>> last = 'Doe'
>>> print '%(first)s %(last)s' % globals()
John Doe

This sort of thing tends to be a bit more readable than the alternatives even though it requires inspecting variables by name.

Karlene answered 26/2, 2009 at 22:49 Comment(2)
I want to try to and print 'first' and 'last', the values that I call the references, not what they are equal to.Plonk
I know, I was just letting you know one reason people commonly look at those dictionaries at all. Usually you shouldn't need to. Unless you want to give us more detail, just dig around in them for what you're looking for. E.g., globals().keys()Karlene
P
10

You can't, as there are no variables in Python but only names.

For example:

> a = [1,2,3]
> b = a
> a is b
True

Which of those two is now the correct variable? There's no difference between a and b.

There's been a similar question before.

Periphery answered 26/2, 2009 at 22:31 Comment(7)
The difference between a and b is the name.Blen
gs is right. a and b refer to the same object, so a function receiving this object has no way to know which name was used to refer to the object in the function call.Pallas
He's kinda right, in that there is no way for you to write a function in Python to differentiate. Still, there's no reason there couldn't be a built-in function that accomplishes this.Blen
dis.dis() returns some names (not always).Paravane
Therefore It is sometimes possible to distinguish between a and b even they're referring to the same object. Python scopes are static. Namespaces are dynamic.Paravane
It means that e.g. when a function returns there might be no local namespace of that function left but you still may inspect the function code object to find what names are used inside.Paravane
"There are no variables in Python" Silly misinformation. You mean, "variables in Python work differently than in C."Regeneration
J
6

Will something like this work for you?

>>> def namestr(**kwargs):
...     for k,v in kwargs.items():
...       print "%s = %s" % (k, repr(v))
...
>>> namestr(a=1, b=2)
a = 1
b = 2

And in your example:

>>> choice = {'key': 24; 'data': None}
>>> namestr(choice=choice)
choice = {'data': None, 'key': 24}
>>> printvars(**globals())
__builtins__ = <module '__builtin__' (built-in)>
__name__ = '__main__'
__doc__ = None
namestr = <function namestr at 0xb7d8ec34>
choice = {'data': None, 'key': 24}
Jojo answered 26/2, 2009 at 23:1 Comment(0)
J
5

With eager evaluation, variables essentially turn into their values any time you look at them (to paraphrase). That said, Python does have built-in namespaces. For example, locals() will return a dictionary mapping a function's variables' names to their values, and globals() does the same for a module. Thus:

for name, value in globals().items():
    if value is unknown_variable:
        ... do something with name

Note that you don't need to import anything to be able to access locals() and globals().

Also, if there are multiple aliases for a value, iterating through a namespace only finds the first one.

Janettjanetta answered 26/2, 2009 at 22:45 Comment(2)
This fails if there is more than one variable name with the same value.Internalize
Not much you can do about that, although I'll add a warning. ;)Janettjanetta
G
3

For the revised question of how to read in configuration parameters, I'd strongly recommend saving yourself some time and effort and use ConfigParser or (my preferred tool) ConfigObj.

They can do everything you need, they're easy to use, and someone else has already worried about how to get them to work properly!

Goldin answered 26/2, 2009 at 23:39 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.