Як передати змінну магічній функції "run" в IPython


82

Я хочу зробити щось на зразок наступного:

In[1]: name = 'long_name_to_type_every_now_and_then.py'

In[2]: %run name

але це насправді намагається запустити 'name.py', що я не хочу робити.

Чи існує загальний спосіб перетворити змінні на рядки?

Щось на зразок наступного:

In[3]: %run %name%

Оскільки команда еквівалентна $ python file args, то, я думаю, це неможливо.
Ashwini Chaudhary

Відповіді:


133

IPython розширює змінні за допомогою $nameстилю 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']

5
Просто хотів зазначити, що це також дуже добре працює, коли просто запускаються довільні команди оболонки з використанням знака оклику (!) - див. Офіційні документи - ipython.org/ipython-doc/rel-0.10.2/html/interactive/…
yoniLavi

@minrk Просто просте запитання, в чому різниця між змінною та виразом. Чи є змінна словника виразом? Я запитую, тому що за вашим прикладом мені довелося використовувати вигадливе форматування рядків для розширення словника, але не вдалося, коли я використав замість цього $. Я думав, що списки, словники тощо - це всі типи змінних.
Стівен Джейкоб,

Я спробував це на %%htmlклітинній магії, і це не спрацювало. Чи існує інший синтаксис для клітинної магії на відміну від лінійної?
MROB

Починаючи з версії 7.3, деякі magicsвідмовитися від цього розширення time, timeit, debugі prun. github.com/ipython/ipython/pull/11516
hpaulj

11

Використовуйте, get_ipython()щоб отримати посилання на поточну InteractiveShell , а потім викликайте magic()метод:

In [1]: ipy = get_ipython()

In [2]: ipy.magic("run foo.py")
ERROR: File `u'foo.py'` not found.

Редагувати Дивіться відповідь minrk - це набагато кращий спосіб це зробити.


1
Хоча це не найкращий спосіб це зробити, він відповів на ще одне моє запитання: як виконувати магічні команди ipython у консолі python (як, під час використання шпигуна)
stuppie

1

Здається, це неможливо завдяки вбудованій %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'

Так, це величезний і, мабуть, марнотратний хак. У цьому ключі мені цікаво, чи є спосіб, щоб ім’я, прив’язане до об’єкта Макрос, використовувалось як дійсне ім’я макросу. Будемо розбиратися в цьому.


Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.