首页 > 编程语言 > 详细

python基础教程2第20章 项目1:即时标记

时间:2019-10-14 19:02:52      阅读:81      评论:0      收藏:0      [点我收藏+]

simple_markup.py

 1 import sys, re
 2 from util import *
 3 
 4 print(<html><head><title>...</title><body>)
 5 
 6 title = True
 7 for block in blocks(sys.stdin):
 8     block = re.sub(r\*(.+?)\*,r<em>\1</em>,block)
 9     if title:
10         print(<h1>)
11         print(block)
12         print(</h1>)
13         title = False
14     else:
15         print(<p>)
16         print(block)
17         print(</p>)
18 
19 print(</body></html>)

 

handler_first.py

 1 class Handler:
 2     """
 3     处理从Parser调用的方法的对象。
 4 
 5     这个解析器会在每个块的开始部分调用start()和end()方法,使用合适的块作为参数。
 6     sub()方法会作用于正则表达式的替换中。当使用了‘emphasis‘这样的名字调用时,它会
 7     返回合适的替换函数。
 8     """
 9     def callback(self, prefix, name, *args):
10         method = getattr(self, prefix+name, None)
11         if callable(method): return method(*args)

handler.py

 1 class Handler:
 2     """
 3     处理从Parser调用的方法的对象。
 4 
 5     这个解析器会在每个块的开始部分调用start()和end()方法,使适用合适的
 6     块名作为参数。sub()方法会用于正则表达式替换中。当使用了‘emphasis‘
 7     这样的名字调用时,它会返回合适的替换函数。
 8     """
 9     def callback(self, prefix, name, *args):
10         method = getattr(self, prefix+name, None)
11         if callable(method):return method(*args)
12     def start(self, name):
13         self.callback(start_, name)
14     def end(self, name):
15         self.callback(end_, name)
16     def sub(self, name):
17         def substitution(match):
18             result = self.callback(sub_,name, match)
19             if result is None:result= match.group(0)
20             return result
21         return substitution
22 
23 class HTMLRenderer(Handler):
24     """
25     用于生成HTML的具体处理程序
26 
27     HTMLRenderer内的方法都可以通过超类处理程序的start()、
28     end()和sub()方法来访问。它们实现了用于HTML文档的基本标签。
29     """
30     def start_document(self):
31         print(<html><head><title>...</title></head><body>)
32     def end_document(self):
33         print(</body></html>)
34     def start_paragraph(self):
35         print(<p>)
36     def end_paragraph(self):
37         print(</p>)
38     def start_heading(self):
39         print(<h2>)
40     def end_heading(self):
41         print(</h2>)
42     def start_list(self):
43         print(<ul>)
44     def end_list(self):
45         print(</ul>)
46     def start_listitem(self):
47         print(<li>)
48     def end_listitem(self):
49         print(</li>)
50     def start_title(self):
51         print(<h1>)
52     def end_title(self):
53         print(</h1>)
54     def sub_emphasis(self, match):
55         return <em>%s</em> % match.group(1)
56     def sub_url(self, match):
57         return <a href="%s">%s</a> % (match.group(1), match.group(1))
58     def sub_email(self, match):
59         return <a href="mailto:%s">%s</a> % (match.group(1), match.group(1))
60     def feed(self, data):
61         print(data)

markup.py

 1 import sys, re
 2 from handlers import *
 3 from util import *
 4 from rules import *
 5 
 6 class Parser:
 7     """
 8     语法分析器读取文本文件、应用规则并且控制处理器程序
 9     """
10 
11     # 初始化成员,handler,rules(),filters()
12     def __init__(self, handler):
13         self.handler = handler
14         self.rules = []
15         self.filters = []
16 
17     # 添加rule,方便扩展
18     def addRule(self, rule):
19         self.rules.append(rule)
20 
21     # 添加过滤器, 方便扩展
22     def addFilter(self, pattern, name):
23         def filter(block, handler):
24             return re.sub(pattern, handler.sub(name), block)
25 
26         self.filters.append(filter)
27 
28 
29     # 方法parse,读取文本(调用util.py的blocks(file))并分成block,
30     # 使用循环用规则(rule)和过滤器(filter(block, handler)处理block,
31     def parse(self, file):
32         self.handler.start(document)
33         for block in blocks(file):
34             for filter in self.filters:
35                 block = filter(block, self.handler)
36             for rule in self.rules:
37                 if rule.condition(block):
38                     if rule.action(block,self.handler):break
39 
40         self.handler.end(document)
41 
42 # Parser派生出的具体的类(通过添加具体的规则和过滤器):用于添加HTML标记
43 
44 class BasicTextParser(Parser):
45     """
46     在构造函数中增加规则和过滤器的具体语法分析器
47     """
48     def __init__(self, handler):
49         Parser.__init__(self, handler)
50         self.addRule(ListRule())
51         self.addRule(ListItemRule())
52         self.addRule(TitleRule())
53         self.addRule(HeadingRule())
54         self.addRule(ParagraphRule())
55 
56         self.addFilter(r\*(.+?)\*, emphasis)
57         self.addFilter(r(http://[\.a-zA-Z/]+), url)
58         self.addFilter(r([\.a-zA-Z]+@[\.a-zA-Z]+[a-zA-Z]+), mail)
59 
60 # 主程序:构造handler实例,构造parser(使用对应的handler)实例,调用parser的方法parser进行对文本的处理
61 handler = HTMLRenderer()
62 parser = BasicTextParser(handler)
63 
64 parser.parse(sys.stdin)

rules.py

 1 class Rule:
 2     """
 3     所有规则的基类
 4     """
 5 
 6     def action(self, block, handler):
 7         handler.start(self.type)
 8         handler.feed(block)
 9         handler.end(self.type)
10         return True
11 
12 class HeadingRule(Rule):
13     """
14     A heading is a single line that is at most 70 characters and
15     that doesn‘t end with a colon.
16     """
17     type = heading
18 
19     def condition(self, block):
20         return not \n in block and len(block) <= 70 and not block[-1] == :
21 
22 class TitleRule(HeadingRule):
23     """
24     The title is the first block in the document, provided that
25     it is a heading.
26     """
27     type = title
28     first = True
29 
30     def condition(self, block):
31         if not self.first: return False
32         self.first = False
33         return HeadingRule.condition(self, block)
34 
35 class ListItemRule(Rule):
36     """
37     A list item is a paragraph that begins with a hyphen. As part of the
38     formatting, the hyphen is removed.
39     """
40     type = listitem
41 
42     def condition(self, block):
43         return block[0] == -
44 
45     def action(self, block, handler):
46         handler.start(self.type)
47         handler.feed(block[1:].strip())
48         handler.end(self.type)
49         return True
50 
51 class ListRule(ListItemRule):
52     """
53     A list begins between a block that is not a list item and a
54     subsequent list item. It ends after the last consecutive list item.
55     """
56     type = list
57     inside = False
58 
59     def condition(self, block):
60         return True
61 
62     def action(self, block, handler):
63         if not self.inside and ListItemRule.condition(self, block):
64             handler.start(self.type)
65             self.inside = True
66         elif self.inside and not ListItemRule.condition(self, block):
67             handler.end(self.type)
68             self.inside = False
69         return False
70 
71 class ParagraphRule(Rule):
72     """
73     A paragraph is simply a block that isn‘t covered by any of the other rules.
74     """
75     type = paragraph
76 
77     def condition(self, block):
78         return True

util.py

 1 def lines(file):
 2     # 将文件打包成行生成器,只是在文件的最后一行添加一个空行
 3     for line in file:yield line
 4     yield \n
 5 
 6 def blocks(file):
 7     block=[]
 8     for line in lines(file):
 9         #搜集所有的行,返回一个字符串
10         if line.strip():
11             block.append(line)
12         #遇到空格,然后返回所有的行,也不会搜集空行
13         elif block: #如果不判断就会将初始化的空行返回,遇到空行结束搜集,如果文件尾没有空行,则不会结束。
14             yield ‘‘.join(block).strip()
15             block=[]#新生成一个空列表,继续以上的步骤

test_input.py

技术分享图片
 1 Welcome to World Wide Spam, Inc.
 2 
 3 These are the corporate web pages of *World Wide Spam*, Inc. We hope
 4 you find your stay enjoyable, and that you will sample many of our
 5 products.
 6 
 7 A short history of the company
 8 
 9 World Wide Spam was started in the summer of 2000. The business
10 concept was to ride the dot-com wave and to make money both through
11 bulk email and by selling canned meat online.
12 
13 After receiving several complaints from customers who werent
14 satisfied by their bulk email, World Wide Spam altered their profile,
15 and focused 100% on canned goods. Today, they rank as the worlds
16 13,892nd online supplier of SPAM.
17 
18 Destinations
19 
20 From this page you may visit several of our interesting web pages:
21 
22    - What is SPAM? (http://wwspam.fu/whatisspam)
23 
24    - How do they make it? (http://wwspam.fu/howtomakeit)
25 
26    - Why should I eat it? (http://wwspam.fu/whyeatit)
27 
28 How to get in touch with us
29 
30 You can get in touch with us in *many* ways: By phone(555-1234), by
31 email (wwspam@wwspam.fu) or by visiting our customer feedback page
32 (http://wwspam.fu/feedback).
View Code

test_output.py

技术分享图片
 1 <html><head><title>...</title></head><body>
 2 <h1>
 3 Welcome to World Wide Spam, Inc.
 4 </h1>
 5 <p>
 6 These are the corporate web pages of <em>World Wide Spam</em>, Inc. We hope
 7 you find your stay enjoyable, and that you will sample many of our
 8 products.
 9 </p>
10 <h2>
11 A short history of the company
12 </h2>
13 <p>
14 World Wide Spam was started in the summer of 2000. The business
15 concept was to ride the dot-com wave and to make money both through
16 bulk email and by selling canned meat online.
17 </p>
18 <p>
19 After receiving several complaints from customers who werent
20 satisfied by their bulk email, World Wide Spam altered their profile,
21 and focused 100% on canned goods. Today, they rank as the worlds
22 13,892nd online supplier of SPAM.
23 </p>
24 <h2>
25 Destinations
26 </h2>
27 <p>
28 From this page you may visit several of our interesting web pages:
29 </p>
30 <ul>
31 <li>
32 What is SPAM? (<a href="http://wwspam.fu/whatisspam">http://wwspam.fu/whatisspam</a>)
33 </li>
34 <li>
35 How do they make it? (<a href="http://wwspam.fu/howtomakeit">http://wwspam.fu/howtomakeit</a>)
36 </li>
37 <li>
38 Why should I eat it? (<a href="http://wwspam.fu/whyeatit">http://wwspam.fu/whyeatit</a>)
39 </li>
40 </ul>
41 <h2>
42 How to get in touch with us
43 </h2>
44 <p>
45 You can get in touch with us in <em>many</em> ways: By phone(555-1234), by
46 email (wwspam@wwspam.fu) or by visiting our customer feedback page
47 (<a href="http://wwspam.fu/feedback">http://wwspam.fu/feedback</a>).
48 </p>
49 </body></html>
View Code

readme.txt

技术分享图片
 1 20.1 问题是什么
 2     将文本由程序自动转成HTML
 3 
 4     要做的工作基本上就是首先将各种文本元素进行分类,比如标题和被强调的文本,然后清晰地标记出它们。为此,
 5     要将HTML标记添加到文本中,使文档能在浏览器中作为网页显示并使用。
 6 
 7     编写原型
 8 
 9     定义目标
10 
11     输入不应该包含人工代码或者标签
12     应该可以处理不同的块
13 20.2 有用的工具
14     要对文件能读写,或者至少能从标准输入(sys.stdin)中读写,在print输出
15     需要对所输入的行进行迭代
16     需要使用一些字符串方法
17     需要一个或者两个生成器
18     可能需要用到re模块
19 20.3 准备工作
20     测试套件评估进度 test_input.txt
21 20.4 初次实现
22     首先要做的是本文本切成段落,段落被一个或者多个空行隔开。比段落更准确的是块(block)
23     20.4.1 找出文本块
24         找出块的简单方法就是搜集遇到的行,直到遇到一个空行,然后返回已经收集到的行。那些返回的行就是一个块。
25         if 。。elif的逻辑选择,
26     20.5.2 添加一些标记
27 
28 总结:
29     主程序(markup.py)
30         1、将处理过程封装为方法parse,这个整个程序的处理过程
31         2、将规则条件抽象成规则类,避免了大量if语句使用。
32         3、self.addRule增加规则对象。利用规则对象的方法和属性实际操作块的处理。
33         4、生成一个html处理器对象(HTMLRenderer),传给BasicTextParser生成对象parser,改该对象调用rule对象接受handler
34         对象处理block
35         5、最重要的是如何将功能抽象成具体的类,以及各个类所有的功能,高内聚低耦合。
View Code

fun_study.py

技术分享图片
 1 # def count():
 2 #     ‘‘‘
 3 #     创建以一个空列表,每次讲定义的函数名存入列表,函数执行三次,列表值[f,f,f]
 4 #     第三次时i为3,fi()=f(),这是f同样保存了,第三次时的父函数的作用域i=3
 5 #     所以f1=3*3=9
 6 #     ‘‘‘
 7 #     fs = []
 8 #     for i in range(1, 4):
 9 #         def f():
10 #              return i*i
11 #         fs.append(f)
12 #     print(i)
13 #     return fs
14 
15 
16 def count():
17     def f(j):
18         def g():
19             return j*j
20         return g
21     fs = []
22     for i in range(1, 4):
23         fs.append(f(i))
24     return fs
25 ‘‘‘体现闭包的用法。fs中函数对象将f(1),f(2),f(3)作用域存放在函数中‘‘‘
26 f1,f2,f3=count()
27 print(f1())
28 print(f2())
29 print(f3())
View Code

 

python基础教程2第20章 项目1:即时标记

原文:https://www.cnblogs.com/landerhu/p/11673067.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!