资讯专栏INFORMATION COLUMN

web.py源码分析: 模板(2)

figofuture / 3110人阅读

摘要:上一篇文章源码分析模板说明了的模板的大致工作原理。本文重点讲述模板支持的语法是如何转换生成函数的。模板的名称统一是。模板代码断行模板内容函数内容从结果来看,模板中的断行只是为了不再结果中插入一个多余的换行符而已。

上一篇文章web.py源码分析: 模板(1)说明了web.py的模板的大致工作原理。本文重点讲述web.py模板支持的语法是如何转换生成__template__函数的。

web.py模板语法和__template__()函数的对应关系

本章会列出模板内容以及转换之后的__template__()函数的内容,以及必要的文字说明。模板的名称统一是hello.html

纯字符串

模板内容

</>复制代码

  1. hello, world

函数内容

</>复制代码

  1. def __template__():
  2. __lineoffset__ = -5
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([u"hello, world
  6. "])
  7. return self
def with

模板内容

</>复制代码

  1. $def with (name, value=[], *args, **kargs)
  2. hello, $name

函数内容

</>复制代码

  1. def __template__ (name, value=[], *args, **kargs):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([u"hello, ", escape_(name, True), u"
  6. "])
  7. return self

从生成的函数可以看出,def with语法所生成的就是__template__()函数的参数列表。

表达式替换

模板内容

</>复制代码

  1. $def with (name, value)
  2. $name
  3. ${name + value}
  4. $(name + value)ing.
  5. $name[value].function()

函数内容

</>复制代码

  1. def __template__ (name, value):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([escape_(name, True), u"
  6. "])
  7. extend_([escape_((name + value), True), u"
  8. "])
  9. extend_([escape_((name + value), True), u"ing.
  10. "])
  11. extend_([escape_(name[value].function(), True), u"
  12. "])
  13. return self

表达式的替换就是执行表达式(表达式对应的代码),得到的结果添加到TemplateResult实例中。

赋值

模板内容

</>复制代码

  1. $def with (name, func)
  2. $ name1 = name
  3. $ name2 = func()

函数内容

</>复制代码

  1. def __template__ (name, func):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. name1 = name
  6. name2 = func()
  7. return self

其实,转换后就是Python代码里的赋值语句。

内容过滤

模板内容

</>复制代码

  1. $def with (name)
  2. $name
  3. $:name

函数内容

</>复制代码

  1. def __template__ (name):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([escape_(name, True), u"
  6. "])
  7. extend_([escape_(name, False), u"
  8. "])
  9. return self

从生成的代码来看,是否使用过滤语法的区别就是传入excape_函数的第二个参数,这个函数其实只是一个字符串处理函数,后续再说。

模板代码断行(newline suppression)

模板内容

</>复制代码

  1. $def with (name, func)
  2. $name
  3. hello,
  4. $name
  5. !
  6. $func(1, 2, 3, 4, 5)

函数内容

</>复制代码

  1. def __template__ (name, func):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([escape_(name, True), u"
  6. "])
  7. extend_([u"hello, "])
  8. extend_([escape_(name, True), u" "])
  9. extend_([u"!
  10. "])
  11. extend_([escape_(func(1, 2, 3, 4, 5), True), u"
  12. "])
  13. return self

从结果来看,模板中的断行只是为了不再结果中插入一个多余的换行符而已。另外,一个表达式的中间是不支持断行的,就比如在模板中不能把func函数的参数列表写成两行。

$符号

模板内容

</>复制代码

  1. $$

函数内容

</>复制代码

  1. def __template__():
  2. __lineoffset__ = -5
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([u"$", u"
  6. "])
  7. return self
注释

模板内容

</>复制代码

  1. $# comment line
  2. hello, world.

函数内容

</>复制代码

  1. def __template__():
  2. __lineoffset__ = -5
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([u"
  6. "])
  7. extend_([u"hello, world.
  8. "])
  9. return self

模板中注释的行在生成的函数中只有一个换行符。

控制结构 for循环

模板内容

</>复制代码

  1. $for i in range(10):
  2. I like $i

函数内容

</>复制代码

  1. def __template__():
  2. __lineoffset__ = -5
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. for i in loop.setup(range(10)):
  6. extend_([u"I like ", escape_(i, True), u"
  7. "])
  8. return self

模板中的for循环被转换成了代码中的for循环,而且用上了变量loop(ForLoop对象后续再来看)。

while循环

模板内容

</>复制代码

  1. $def with (name_list)
  2. $while name_list:
  3. hello, $name_list.pop()

函数内容

</>复制代码

  1. def __template__ (name_list):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([u"
  6. "])
  7. while name_list:
  8. extend_([u"hello, ", escape_(name_list.pop(), True), u"
  9. "])
  10. return self

注意和for循环的区别,没有使用loop变量。

for循环的loop变量

模板内容

</>复制代码

  1. $def with (name_list)
  2. $for name in name_list:
  3. $loop.index
  4. $loop.index0
  5. $loop.first
  6. $loop.last
  7. $loop.odd
  8. $loop.even
  9. $loop.parity
  10. $loop.parent
  11. hello, $name
  12. $for i in range(10):
  13. $for name in name_list:
  14. $loop.parent
  15. hello, $name

函数内容

</>复制代码

  1. def __template__ (name_list):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. for name in loop.setup(name_list):
  6. extend_([escape_(loop.index, True), u"
  7. "])
  8. extend_([escape_(loop.index0, True), u"
  9. "])
  10. extend_([escape_(loop.first, True), u"
  11. "])
  12. extend_([escape_(loop.last, True), u"
  13. "])
  14. extend_([escape_(loop.odd, True), u"
  15. "])
  16. extend_([escape_(loop.even, True), u"
  17. "])
  18. extend_([escape_(loop.parity, True), u"
  19. "])
  20. extend_([escape_(loop.parent, True), u"
  21. "])
  22. extend_([u"hello, ", escape_(name, True), u"
  23. "])
  24. extend_([u"
  25. "])
  26. for i in loop.setup(range(10)):
  27. for name in loop.setup(name_list):
  28. extend_([escape_(loop.parent, True), u"
  29. "])
  30. extend_([u"hello, ", escape_(name, True), u"
  31. "])
  32. return self

这里展示了loop变量的成员,以及嵌套循环的使用。

if-else

模板内容

</>复制代码

  1. $def with (name_list)
  2. $if name_list:
  3. $len(name_list)
  4. $else:
  5. 0

函数内容

</>复制代码

  1. def __template__ (name_list):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. if name_list:
  6. extend_([escape_(len(name_list), True), u"
  7. "])
  8. else:
  9. extend_([u"0
  10. "])
  11. return self

elif语句也支持。

函数定义

模板内容

</>复制代码

  1. $def with (name_list)
  2. $def hello(name):
  3. hello, $name
  4. $def hello_to_all(nlist):
  5. $for each in nlist:
  6. $hello(each)
  7. $hello_to_all(name_list)

函数内容

</>复制代码

  1. def __template__ (name_list):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([u"
  6. "])
  7. __lineoffset__ -= 3
  8. def hello(name):
  9. self = TemplateResult(); extend_ = self.extend
  10. extend_([u"hello, ", escape_(name, True), u"
  11. "])
  12. extend_([u"
  13. "])
  14. return self
  15. __lineoffset__ -= 3
  16. def hello_to_all(nlist):
  17. self = TemplateResult(); extend_ = self.extend
  18. for each in loop.setup(nlist):
  19. extend_([escape_(hello(each), True), u"
  20. "])
  21. extend_([u"
  22. "])
  23. return self
  24. extend_([escape_(hello_to_all(name_list), True), u"
  25. "])
  26. return self

模板对函数的支持其实就是定义内部函数并且调用,每个内部函数的返回结果也都是TemplateResult实例。

code

模板内容

</>复制代码

  1. $def with (name_list)
  2. $code:
  3. new_list = [x.upper() for x in name_list]
  4. def hello(name):
  5. return "hello, %s" % (name)
  6. more_new_list = []
  7. for each in new_list:
  8. more_new_list.append(hello(each))
  9. $hello("everybody")
  10. $len(more_new_list)

函数内容

</>复制代码

  1. def __template__ (name_list):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. extend_([u"
  6. "])
  7. new_list = [x.upper() for x in name_list]
  8. def hello(name):
  9. return "hello, %s" % (name)
  10. more_new_list = []
  11. for each in new_list:
  12. more_new_list.append(hello(each))
  13. extend_([escape_(hello("everybody"), True), u"
  14. "])
  15. extend_([escape_(len(more_new_list), True), u"
  16. "])
  17. return self

code的语法有点复杂,其内部是用来定义原始的Python代码的,有如下几个特点:

code内部定义的函数也是内部函数,在模板的其他地方可以调用,但是不会把结果存放在TemplateResult实例中返回。

code中定义的变量都会作为__template__()函数的局部变量,在模板的其他地方可以调用。

注意,code块中不要使用print语句打印输出(虽然默认已经禁止了)。

var

模板内容

</>复制代码

  1. $def with (name_list)
  2. $var title: hi
  3. $var title2: "hi"
  4. $var name: $name_list[0]
  5. $var name2: name_list[0]

函数内容

</>复制代码

  1. def __template__ (name_list):
  2. __lineoffset__ = -4
  3. loop = ForLoop()
  4. self = TemplateResult(); extend_ = self.extend
  5. self["title"] = join_(u"hi")
  6. self["title2"] = join_(u""hi"")
  7. self["name"] = join_(escape_(name_list[0], True))
  8. self["name2"] = join_(u"name_list[0]")
  9. return self

var是用来为模板设置属性的,从生成的代码来看,就是为TemplateResult实例设置属性。上面的模板内容里有一些关键细节:

var name: value中的冒号后面的内容默认是作为字符串处理的,可以不用加引号,如果加了引号,则引号也会作为内容的一部分。

如果要在冒号后面访问一些变量值,需要使用$前缀。

总结

写了这些web.py模板支持的语法和生成的代码的对应关系,希望有助于大家理解模板的语法,了解每种语法的用途,避免踩坑。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/45358.html

相关文章

  • web.py源码分析: 模板(1)

    摘要:模板函数到底长什么样下面我们就可以来看看模板函数到底长什么样了。当然,首先得创建一个模板文件。总结通过打印中间结果和分析代码,我们已经大概知道了的模板是如何转化成内容的。下一篇文章会阐述模板的各种语法所对应的动态函数内容。 web.py模板的实现原理 web.py的模板实现利用了Python的可执行对象的动态特性:根据模板内容和渲染函数的参数创建一个函数,该函数执行的时候会返回一个Te...

    Rocko 评论0 收藏0
  • web.py源码分析: 模板(3)

    摘要:前两篇文章主要说明了的模板系统将模板文件处理后得到的结果函数。生成函数的代码这个是模板生成过程中最长最复杂的一段,会应用到的分析功能以及动态编译功能。参数都是一个,表示还未解析的模板内容。 前两篇文章主要说明了web.py的模板系统将模板文件处理后得到的结果:__template__()函数。本文主要讲述模板文件是如何变成__template__()函数的。 Render和frende...

    OnlyMyRailgun 评论0 收藏0
  • web.py源码分析: application(1)

    摘要:本文主要分析的是库的这个模块中的代码。将结果转换成一个迭代器。函数函数的定义如下位置位置位置该函数的参数中就是,是路由映射表则是,是本次请求路径。位置,如果是其他情况,比如直接指定一个类对象作为处理对象。 本文主要分析的是web.py库的application.py这个模块中的代码。总的来说,这个模块主要实现了WSGI兼容的接口,以便应用程序能够被WSGI应用服务器调用。WSGI是We...

    edgardeng 评论0 收藏0
  • 基于Linux环境的Web.py框架介绍

    摘要:前言在文章基于环境搭建框架方法介绍中介绍了客户端和服务器的交互过程,服务器接收客户端的请求后,由应用服务器对浏览器的请求进行处理,将生成的响应传递给服务器,再由服务器返回给客户端。 前言 在文章《基于Linux环境搭建Nginx+uWSGI+Python框架方法介绍》中介绍了客户端和Web服务器的交互过程,Web服务器接收客户端的请求后,由Web应用服务器对浏览器的请求进行处理,将生成...

    caikeal 评论0 收藏0
  • newrelic python agent 源码分析-1

    摘要:是应用性能管理监控解决方案提供商。目录是列出的命令脚本所在目录。包含文件如下的函数是命令执行的入口。而对于硬件信息的检测则由进行。文档地址源码仔细看下去,太复杂了。下一篇再分析一个请求到结束探针工作的完整过程吧。 Newrelic 是APM(Application Performance Management)(应用性能管理/监控)解决方案提供商。项目中,通常用它来追踪应用的性能。最近...

    szysky 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<