Python __setattr__ и __getattr__ для глобальной области видимости?
Предположим, мне нужно создать свой собственный небольшой DSL, который бы использовал Python для описания определенной структуры данных. Например, я хотел бы иметь возможность написать что-то вроде
f(x) = some_stuff(a,b,c)
и пусть Python вместо того, чтобы жаловаться на необъявленные идентификаторы или пытаться вызвать функцию some_stuff, преобразует ее в буквальное выражение для моего дальнейшего удобства.
Можно получить разумное приближение к этому, создав класс с правильно переопределенным __getattr__
а также __setattr__
методы и использовать его следующим образом:
e = Expression()
e.f[e.x] = e.some_stuff(e.a, e.b, e.c)
Было бы круто, хотя, если бы можно было избавиться от раздражающего "е". префиксы и, возможно, даже избежать использования []. Так что мне было интересно, возможно ли как-то временно "переопределить" глобальный поиск и присвоение имен? В связи с этим, может быть, есть хорошие пакеты для простого достижения такой "цитирующей" функциональности для выражений Python?
3 ответа
Я не уверен, что это хорошая идея, но я решил попробовать. Подвести итоги:
class PermissiveDict(dict):
default = None
def __getitem__(self, item):
try:
return dict.__getitem__(self, item)
except KeyError:
return self.default
def exec_with_default(code, default=None):
ns = PermissiveDict()
ns.default = default
exec code in ns
return ns
Возможно, вы захотите взглянуть на ast
или же parser
модули, включенные в Python для анализа, доступа и преобразования абстрактного синтаксического дерева (или дерева синтаксического анализа соответственно) входного кода. Насколько я знаю, математическая система Sage, написанная на Python, имеет подобный вид прекомпилятора.
В ответ на комментарий Вая вот одно забавное решение, которое я нашел. Прежде всего, чтобы еще раз объяснить, что он делает, предположим, что у вас есть следующий код:
definitions = Structure()
definitions.add_definition('f[x]', 'x*2')
definitions.add_definition('f[z]', 'some_function(z)')
definitions.add_definition('g.i', 'some_object[i].method(param=value)')
где добавление определений подразумевает разбор левой и правой сторон и выполнение других неприятных вещей. Теперь один (не обязательно хороший, но, безусловно, веселый) подход позволил бы написать приведенный выше код следующим образом:
@my_dsl
def definitions():
f[x] = x*2
f[z] = some_function(z)
g.i = some_object[i].method(param=value)
и Python сделает большую часть разбора под капотом. Идея основана на простом exec <code> in <environment>
Заявление, упомянутое Яном, с одним хакерским дополнением. А именно, байт-код функции должен быть слегка изменен, и все локальные операции доступа к переменным (LOAD_FAST) переключаются на доступ к переменным из среды (LOAD_NAME).
Это легче показать, чем объяснить: http://fouryears.eu/wp-content/uploads/pydsl/
Существуют различные приемы, которые вы можете сделать, чтобы сделать это практичным. Например, в коде, представленном по ссылке выше, вы не можете использовать встроенные функции и языковые конструкции, такие как for, и операторы if внутри функции @my_dsl. Однако вы можете заставить их работать, добавив больше поведения в класс Env.
Обновление Вот немного более подробное объяснение того же самого.