Is there a way to define a function to be global from within a class( or from within another function, as matter of fact)? Something similar to defining a global variable.
4 Answers
Functions are added to the current namespace like any other name would be added. That means you can use the global keyword inside a function or method:
def create_global_function():
global foo
def foo(): return 'bar'
The same applies to a class body or method:
class ClassWithGlobalFunction:
global spam
def spam(): return 'eggs'
def method(self):
global monty
def monty(): return 'python'
with the difference that spam will be defined immediately as top-level class bodies are executed on import.
Like all uses of global you probably want to rethink the problem and find another way to solve it. You could return the function so created instead, for example.
Demo:
>>> def create_global_function():
... global foo
... def foo(): return 'bar'
...
>>> foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'foo' is not defined
>>> create_global_function()
>>> foo
<function foo at 0x102a0c7d0>
>>> foo()
'bar'
>>> class ClassWithGlobalFunction:
... global spam
... def spam(): return 'eggs'
... def method(self):
... global monty
... def monty(): return 'python'
...
>>> spam
<function spam at 0x102a0cb18>
>>> spam()
'eggs'
>>> monty
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'monty' is not defined
>>> ClassWithGlobalFunction().method()
>>> monty()
'python'
3 Comments
class definition body, but I'd be rather pushed as to why you'd want to do that.You can use global to declare a global function from within a class. The problem with doing that is you can not use it with a class scope so might as well declare it outside the class.
class X:
global d
def d():
print 'I might be defined in a class, but I\'m global'
>> X.d
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'X' object has no attribute 'd'
>> d()
I might be defined in a class, but I'm global
2 Comments
I found a case where global does not have the desired effect: inside .pdbrc file. If you define functions in .pdbrc they will only be available from the stack frame from which pdb.set_trace() was called.
However, you can add a function globally, so that it will be available in all stack frames, by using an alternative syntax:
def cow(): print("I'm a cow")
globals()['cow']=cow
I've tested that it also works in place of the global keyword in at least the simplest case:
def fish():
def cow():
print("I'm a cow")
globals()['cow']=cow
Despite being more verbose, I thought it was worth sharing this alternative syntax. I have not tested it extensively so I can't comment on its limitations vs using the global keyword.
Comments
The global trick doesn't seem to work with python 3.12 or later (and likely earlier)
This does, setting the name in the builtins module directly with sys.modules:
sys.modules['builtins'].name=name
e.g.
import sys
def cow():
print("I'm a cow")
sys.modules['builtins'].cow = cow
The builtins module has all the identifiers that are globally available in Python.
This can be done even more neatly using decorators. After reading the excellent essay at How do I make function decorators and chain them together? I also provide this tested solution:
def make_builtin(func):
def builtin_func(*args, **kwargs):
func(*args, **kwargs)
import sys
sys.modules['builtins'].__dict__[func.__name__]=func
return builtin_func
@make_builtin
def cow(): print("I'm a cow")
Be warned though, just because it is easy, doesn't make it a good idea.
Why not do it nicely, with functools.wrap https://docs.python.org/3/library/functools.html?
def make_builtin(func):
import functools
@functools.wraps(func)
def builtin_func(*args, **kwargs):
func(*args, **kwargs)
import sys
sys.modules['builtins'].__dict__[func.__name__]=func
return builtin_func
But I don't like import functools making functools available to the wrapper function when it would not otherwise have been, so maybe a sly loader is needed
5 Comments
fish(), after import p I must do p.cow() and globals()['cow'] is erroned. I must do from p import * to be able to do cow() and globals()['cow']. Sorry, when I did the test it was entering the definition and call of fish directly in the python shell, not through an importsys.modules['builtins'] is a very bad idea !
bytearray()for optimization purposes, because all your values are assumed to be within 0 and 255. Now let's say that you want to support values above 255 in some cases, but you want to get maximum performance for cases when values are guaranteed to be bounded to 0..255. One way to do that is atmy_setup_function()to redefinebytearraydynamically to alist()if needed.