我想做如下的事情:
In[1]: name = 'long_name_to_type_every_now_and_then.py'
In[2]: %run name
但这实际上试图运行'name.py'
,这不是我想要做的。
有没有将变量转换为字符串的通用方法?
类似于以下内容:
In[3]: %run %name%
我想做如下的事情:
In[1]: name = 'long_name_to_type_every_now_and_then.py'
In[2]: %run name
但这实际上试图运行'name.py'
,这不是我想要做的。
有没有将变量转换为字符串的通用方法?
类似于以下内容:
In[3]: %run %name%
$name
IPython 使用, bash 样式扩展变量。这适用于所有魔法,而不仅仅是%run
。
所以你会这样做:
In [1]: filename = "myscript.py"
In [2]: %run $filename
['myscript.py']
myscript.py 包含:
import sys
print(sys.argv)
通过 Python 的花哨字符串格式,您甚至可以将表达式放入{}
:
In [3]: args = ["arg1", "arg2"]
In [4]: %run $filename {args[0]} {args[1][-2:]}
['myscript.py', 'arg1', 'g2']
用于get_ipython()
获取对当前InteractiveShell的引用,然后调用该magic()
方法:
In [1]: ipy = get_ipython()
In [2]: ipy.magic("run foo.py")
ERROR: File `u'foo.py'` not found.
编辑查看minrk 的答案——这是一种更好的方法。
如果参数中可能有空格,例如文件名,最好使用这个:
%run "$filename"
使用内置的%run
魔法功能,这似乎是不可能的。不过,你的问题让我陷入了困境,我想看看做类似的事情是多么容易。最后,花费所有这些精力来创建另一个仅使用execfile()
. 也许这对某人、某处有一些用处。
# custom_magics.py
from IPython.core.magic import register_line_magic, magics_class, line_magic, Magics
@magics_class
class StatefulMagics(Magics):
def __init__(self, shell, data):
super(StatefulMagics, self).__init__(shell)
self.namespace = data
@line_magic
def my_run(self, line):
if line[0] != "%":
return "Not a variable in namespace"
else:
filename = self.namespace[line[1:]].split('.')[0]
filename += ".py"
execfile(filename)
return line
class Macro(object):
def __init__(self, name, value):
self.name = name
self._value = value
ip = get_ipython()
magics = StatefulMagics(ip, {name: value})
ip.register_magics(magics)
def value(self):
return self._value
def __repr__(self):
return self.name
使用这对类,(并给定一个 python 脚本tester.py
),可以创建和使用带有新创建的“my_run”魔术函数的“宏”变量,如下所示:
In [1]: from custom_magics import Macro
In [2]: Macro("somename", "tester.py")
Out[2]: somename
In [3]: %my_run %somename
I'm the test file and I'm running!
Out[3]: u'%somename'
是的,这是一个巨大且可能是浪费的黑客攻击。在这种情况下,我想知道是否有办法将绑定到宏对象的名称用作宏的实际名称。会调查的。
如果“变量”是指用户命名空间中可用的变量,那么我建议使用get_ipython().ev(varname)
(docs)在函数中获取变量:
from __future__ import print_function
import argparse
import shlex
from IPython import get_ipython
from IPython.core.magic import register_cell_magic
@register_cell_magic
def magical(line, cell):
parser = argparse.ArgumentParser()
parser.add_argument("varname")
# using shlex to split the line in a sys.argv-like format
args = parser.parse_args(shlex.split(line))
ipython = get_ipython()
v = ipython.ev(args.varname)
# do whatever you want here
print(v)
上面的这个模板对我很有帮助——我用它来制作一个 jinja2 单元魔法,每当单元运行时它也会打印到文件中。