更改一个模块中的模块变量,而原始模块中的变量不变
我正在使用 foo 模块,其中包含一些指示其行为的模块级变量。这些变量之一是 SANITIZE。
我希望能够以两种方式使用 foo,foo.parse() 和 foo2.parse(),其中唯一的区别是 foo 有 SANITIZE=false 而 foo2 有 SANITIZE=true
我想在不必复制的情况下执行此操作粘贴我的代码。例如
#foo module
SANITIZE='foo'
def parse():
print SANITIZE
#foo2 module
import foo
foo.SANITIZE='foo2'
#test script
import foo
import foo2
foo2.foo.parse() #i want this to print 'foo2'
foo.parse() #i want this to print 'foo'
但是,上面的示例将两次打印“foo2”。有没有办法实现这种行为?
谢谢!
I'm using the foo module which includes a few module level variables that dictate its behavior. One of these variables is SANITIZE.
I want to be able to use foo in two ways, foo.parse() and foo2.parse(), where the only difference is foo has SANITIZE=false and foo2 has SANITIZE=true
I want to do this while not having to copy paste my code. For example
#foo module
SANITIZE='foo'
def parse():
print SANITIZE
#foo2 module
import foo
foo.SANITIZE='foo2'
#test script
import foo
import foo2
foo2.foo.parse() #i want this to print 'foo2'
foo.parse() #i want this to print 'foo'
However, the above example will print 'foo2' both times. Is there a way to achieve this behavior?
Thanks!
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
如果这是您的代码,那么解决方案不是依赖于模块级变量,而是以其他方式来保持对象的状态。 Python 中的模块是“单例”——这意味着一旦被任何模块导入,在解释器范围内就只有一个版本——
例如,如果您使用类继承,您想要的行为正是您免费获得的 - 子类可以自定义一些父类,但无需重写全部父类。
因此,如果您将“foo”代码封装在一个类中 - 您甚至可能不需要编写一个需要实例的类,您就已经获得了您想要的功能:
当然,通过大量的内省和元编程, Python 允许,它将可以做你想要的事情 - 但这会使事情变得复杂,而且没有好处。模块变量与 C 代码中的全局变量具有大部分相同的缺点。
一种方法是在通过 foo2 访问时创建“foo”,以动态更改变量,调用“foo”中的函数并在退出时恢复之前的值。
对于要在模块“foo2”上的“foo”的属性访问上触发的任意代码,“foo”必须引用具有“property”属性的对象 -
因此,您编写的 exa 示例将同时运行-不安全的方式,顺便说一句,非常不安全的方式,如果 foo2 或多或少这样写:
这会在模块 foo2 中创建一个“foo”对象,当访问该对象时,会从原始“foo”模块中检索任何请求的属性。因此,“foo2.foo.parse”将获得“foo”解析函数 - 但是,感知这种方法中的“hackiness”量 - 以便能够恢复位于“foo”模块内部的原始值Python 解释器,从 foo2 获取函数后,该函数在返回时也会恢复值本身。这样做的唯一方法是修改函数,以便它在返回时运行附加代码 - 因此它是由上面的代码动态修饰的。
我认为这个例子清楚地表明,在这种情况下,模块级配置并不是可行的方法,尽管这是可能的。
编辑
OP 评论道:
回复:
阅读此描述可能听起来“不可行” - 但它比描述的更容易 -
下面是我的“foo2”模块,它执行上述操作:
此代码完全按照我的描述进行操作 - 它重新创建原始模块中的所有函数对象,重新绑定每个函数的全局字典。它是并发安全的。如果要克隆的模块确实指向本机代码类或函数(它们不是“FunctionType”),则存在一些极端情况。如果它大量使用多个类继承、元类,那么它可能有效,也可能无效。
我用一个简单的类对其进行了测试,效果很好:
输出
:
If this is your code, than the solution is not to depend in module level variables, but in some other way to keep state of the objects. Modules in Python are "singletons" - which means that once imported by any module, there is just one version of then, interpreter wide -
The behavior you want, for example, is exactly what you get, for free, if you use class inheritance - the child class can customize some, but no need to rewrite all, of the parent class.
So, if you as much as encapsulate your "foo" code inside a class - you may not even have to write a class that needs instances, you get the features you want already:
Of course, with the cheer amount of introspection and metaprogramming that Python allows, it would be possible to do something like what you want - but it would complicate matters, for no good. Module variables have most of the same disadvantages that global variables have in C code.
One way to do it is to make "foo" when accessed through foo2 to make the variable changes on the fly, call the function in "foo" and restore the previosu values on exit.
For arbitrary code to be triggered on attribute access of "foo" on the module "foo2", "foo" has to refer to an object with a "property" attribute --
So, the exa example you wrote would run, in a concurency-unsafe way, btw, very unsafe way, if foo2 is written more or less so:
This creates a "foo" object in module foo2 that when accessed, retrieves any requested attributes from the original "foo" module instead. So "foo2.foo.parse" will get the "foo" parse function - but, perceive the ammount of "hackiness" in this approach - in order to be able to restore the original value in the "foo" module that lives inside the Python interpreter, after the function was fetched from foo2, that function have too, upon returning, restore the values itself. The only way to do so is modifying the function so that it runs additional code when it returns - so it is decorated on the fly by the code above.
I think this example makes clear that having module level configurations is not the way to go in this case, although possible.
EDIT
The O.P. commented:
In reply to that:
Reading this description it may sound as "unfeasable" - but it is easier done than described -
Here follows my "foo2" module that does the above:
This code does exactly what I described - it recreates all function objects in the original module, rebindign the globals dict for each function. It is concurrency safe. There are some corner cases if the to-be clonned module does point to native code classes or functions (they are not of "FunctionType"). If it makes heavy usage of multiple class inheritance, metaclasses,- it miught work, or not.
I tested it with a simple class and it worked fine:
And
Output:
您的问题看起来与此处描述的非常相似: http ://effbot.org/pyfaq/how-do-i-share-global-variables-across-modules.htm
对于您的情况,也许您可以执行以下操作:
Your question looks very similar to what's described here: http://effbot.org/pyfaq/how-do-i-share-global-variables-across-modules.htm
For your situation perhaps you could do the following: