运算符拦截

New in version 2.6.

为了性能最大化, Jinja2 会让运算符直接条用类型特定的回调方法。这意味着,通过重载 Environment.call() 来拦截是不可能的。此外,由于运算符的工作方式,把运算符转换为特殊方法不总是直接可行的。比如为了分类,至少一个特殊方法存在。

在 Jinja 2.6 中,开始支持显式的运算符拦截。必要时也可以用于自定义的特定运算符。为了拦截运算符,需要覆写SandboxedEnvironment.intercepted_binops 属性。当需要拦截的运算符被添加到这个集合, Jinja2 会生成调用SandboxedEnvironment.call_binop() 函数的字节码。对于一元运算符,必须替代地使用 unary 属性和方法。

SandboxedEnvironment.call_binop 的默认实现会使用SandboxedEnvironment.binop_table 来把运算符标号翻译成执行默认运算符行为的回调。

这个例子展示了幂( ** )操作符可以在 Jinja2 中禁用:

  1. from jinja2.sandbox import SandboxedEnvironment
  2. class MyEnvironment(SandboxedEnvironment):
  3. intercepted_binops = frozenset(['**'])
  4. def call_binop(self, context, operator, left, right):
  5. if operator == '**':
  6. return self.undefined('the power operator is unavailable')
  7. return SandboxedEnvironment.call_binop(self, context,
  8. operator, left, right)

确保始终调入 super 方法,即使你不拦截这个调用。 Jinja2 内部会调用这个方法来对表达式求值。