Head First Python(Web开发)

来源:互联网 发布:软件系统功能分析 编辑:程序博客网 时间:2024/05/10 19:26

把程序放在Web上

Web应用有以下特点:

  1. 能够访问你的网站的每一个人都可以使用这个程序。
  2. 位于Web服务器上的某个位置。
  3. 需要更新功能是很容易更新。
Web请求作为某个用户交互的结果由Web浏览器发送到Web服务器。在Web服务器上,会生成Web响应并发回到Web浏览器。整个过程可以总结为5个步骤。

  • 用户在他选择的Web浏览器中输入一个Web地址,选择一个超链接,或者点击一个按钮
  • Web浏览器将用户的动作转换为一个Web请求,并通过互联网把他发送到一个服务器
  • Web服务器收到Web请求,必须决定接下来做什么
  • Web服务器处理Web请求,创建一个Web响应,通过互联网发送给等待着的Web浏览器
  • Web浏览器接收到Web响应,并显示在用户的屏幕上

第4步(可能)有很多子步骤:Web服务器要找到所有执行的程序、执行找到的程序,然后捕捉程序的输出作为Web响应......发回Web浏览器。这个动态生成内容的过程,称为通用网关接口(Common Gateway Interface,CGI),符合这个标准的程序称为CGI脚本。

Web应用需要做什么

可能需要3个Web页面:一个”欢迎“页面、一个”选择选手“页面、一个”显示时间“页面。

采用MVC设计Web应用

好的Web应用应当遵循模型-视图-控制器(Model-View-Control)模式。

  1. 模型:存储(以及及时处理)Web应用数据的代码
  2. 视图:格式化和显示Web应用用户界面的代码
  3. 控制器:将Web应用”粘合“在一起并提供业务逻辑的代码

为数据建模

Web服务器需要存储数据的一个副本。这里的数据就是Kelly教练的计时值。

启动Web应用时,需要把文本文件中的数据转换为AthleteList对象实例,存储在一个字典中,然后保存为pickle文件。这个功能放在put_to_store()函数中。

运行Web应用时,pickle文件中的数据可以作为一个字典供应用使用。这个功能放在get_from_store()函数中。

<span style="font-size:18px;">def put_to_store(files_list):    all_athletes={}    for each_file in files_list:        ath=get_coach_data(each_file)        all_athletes[ath.name]=ath    try:        with open ('athletes.pickle', 'wb') as athf:            pickle.dump(all_athletes,athf)    except IOError as ioerr:        print('File error (put_and_stroe): '+str(ioerr))    return(all_athletes)def get_from_store():    all_athletes={}    try:        with open('athletes.pickle', 'rb') as athf:            all_athletes=pickle.load(athf)    except IOError as ioerr:        print('File error (get_from_store): '+str(ioerr))    return(all_athletes)</span>
将代码键入一个IDLE编辑窗口,并把代码保存到一个文件夹中。按下F5将代码导入IDLE shell,然后使用dir()命令确认导入成功。

<span style="font-size:18px;">>>>> dir()['AthleteList', '__builtins__', '__doc__', '__file__', '__loader__', '__name__', '__package__', 'get_coach_data', 'get_from_store', 'pickle', 'put_to_store']</span>
创建一个要处理的文件列表,然后调用put_to_store()函数将文件列表中的数据转换成一个字典,存储在pickle中:

<span style="font-size:18px;">>>> the_files=['sarah2.txt','james2.txt','mikey2.txt','julie2.txt']>>> data=put_to_store(the_files)>>> data{'James Lee': ['2-34', '3:21', '2.34', '2.45', '3.01', '2:01', '2:01', '3:10', '2-22', '2-01', '2.01', '2:16'], 'Mikey McManus': ['2:22', '3.01', '3:01', '3.02', '3:02', '3.02', '3:22', '2.49', '2:38', '2:40', '2.22', '2-31'], 'Sarah Sweeney': ['2:58', '2.58', '2:39', '2-25', '2-55', '2:54', '2.18', '2:55', '2:55', '2:22', '2-21', '2.22'], 'Julie Jones': ['2.59', '2.11', '2:11', '2:23', '3-10', '2-23', '3:10', '3.21', '3-21', '3.01', '3.02', '2:59']}</span>
使用字典中现有的数据显示各个选手的名字和出生日期:

<span style="font-size:18px;">>>> for each_athlete in data:print(data[each_athlete].name+''+data[each_athlete].dob)James Lee2002-3-14Mikey McManus2002-2-24Sarah Sweeney2002-6-17Julie Jones2002-8-17</span>
使用get_from_store()函数将腌制数据加载到另一个字典中:

>>> data_copy=get_from_store()>>> for each_athlete in data_copy:print(data_copy[each_athlete].name+''+data_copy[each_athlete].dob)James Lee2002-3-14Mikey McManus2002-2-24Sarah Sweeney2002-6-17Julie Jones2002-8-17
查看界面

视图代码创建Web应用的用户界面(user interface,UI)。用户界面用Web的标记技术HTML来创建。

YATE是一个包括一些生成HTML的辅助函数的模版。

<span style="font-size:18px;">from string import Template从标准库的"string"模块导入"Template"类。它支持简单的字符串替换模版。def start_response(resp="text/html"):    return('Content-type: ' + resp + '\n\n')这个函数需要一个(可选的)字符串作为参数,用它来创建一个CGI"Content-type:"行,参数缺省值是"text/html"。def include_header(the_title):    with open('templates/header.html') as headf:        head_text = headf.read()    header = Template(head_text)    return(header.substitute(title=the_title))这个函数需要一个字符串作为参数,用在HTML页面最前面的标题中。页面本身存储在一个单独的文件"templates/header.html"中,可以根据需要替换标题。def include_footer(the_links):    with open('templates/footer.html') as footf:        foot_text = footf.read()    link_string = ''    for key in the_links:        link_string += '<a href="' + the_links[key] + '">' + key + '</a>    '    footer = Template(foot_text)    return(footer.substitute(links=link_string))使用一个字符串作为参数,来创建一个HTML页面的尾部。页面本身存储在一个单独的文件"templates/footer.html"中,参数用于动态的创建一组HTML链接标记。从这些标记的使用来看,参数应当是一个字典。def start_form(the_url, form_type="POST"):    return('<form action="' + the_url + '" method="' + form_type + '">')这个函数返回表单最前面的HTML,允许调用者指定URl(表单数据将发送到这个URL),还可以指定所要使用的方法。    return('<p></p><input type=submit value="' + submit_msg + '"></form>')这个函数返回表单末尾的HTML标记,同时还允许调用者定制表单"submit"(提交)按钮的文本。def radio_button(rb_name, rb_value):    return('<input type="radio" name="' + rb_name +                             '" value="' + rb_value + '"> ' + rb_value + '<br />')给定一个单选钮名和值,创建一个HTML单选钮(通常包括在一个HTML表单中)def u_list(items):    u_string = '<ul>'    for item in items:        u_string += '<li>' + item + '</li>'    u_string += '</ul>'    return(u_string)给定一个项列表,这个函数会把该列表转换为一个HTML无序列表。一个简单的"for"循环就可以完成全部工作,每次迭代会向UL元素增加一个LI元素。def header(header_text, header_level=2):    return('<h' + str(header_level) + '>' + header_text +           '</h' + str(header_level) + '>')创建并返回一个HTML标题标记(H1、H2、H3等),默认为2级标题。"header_text"参数是必要的。def para(para_text):    return('<p>' + para_text + '</p>') </span></span>
<span style="font-size:18px;">用HTML段落标记包围一个文本段(一个字符串)。</span>
模块加载到IDLE,按下F5来执行。

默认的CGI响应首部的几种类型。

>>> start_response()'Content-type: text/html\n\n'>>> start_response("text/plain")'Content-type: text/plain\n\n'>>> start_response("application/json")'Content-type: application/json\n\n'
<span style="background-color: rgb(0, 153, 0);">include_header()函数生成Web页面的开始部分,允许定制页面的标题:</span>>>> include_header("Welcome to my home on the web!")'<html>\n<head>\n<title>Welcome to my home on the web!</title>\n<link type="text/css" rel="stylesheet" href="/coach.css" />\n</head>\n<body>\n<h1>Welcome to my home on the web!</h1>\n'
<span style="background-color: rgb(0, 153, 0);">include_footer()函数会生成一个Web页面末尾的HTML,并提供链接。第二个没有包含链接。</span>>>> include_footer({'Home':'/index.html','Select':'/cgi-bin/select/py'})'<p>\n<a href="/cgi-bin/select/py">Select</a>    <a href="/index.html">Home</a>    \n</p>\n</body>\n</html>\n'>>> include_footer({})'<p>\n\n</p>\n</body>\n</html>\n'
<span style="background-color: rgb(0, 153, 0);">start_form()和end_form()会建立一个HTML表单,并用参数来调整所生产的HTML的内容</span>>>> start_form("/cgi-bin/process-athlete.py")'<form action="/cgi-bin/process-athlete.py" method="POST">'>>> end_form()'<p></p><input type=submit value="Submit"></form>'>>> end_form("Click to Confirm Your Order")'<p></p><input type=submit value="Click to Confirm Your Order"></form>'
<span style="background-color: rgb(0, 153, 0);">radio_button()函数创建单选钮。</span>>>> for fab in ['John','Paul','George','Ringo']:radio_button(fab,fab)'<input type="radio" name="John" value="John"> John<br />''<input type="radio" name="Paul" value="Paul"> Paul<br />''<input type="radio" name="George" value="George"> George<br />''<input type="radio" name="Ringo" value="Ringo"> Ringo<br />'
<span style="background-color: rgb(0, 153, 0);">u_list()函数创建无序列表。</span>>>> u_list(['Life of Brian','Holy Grail'])'<ul><li>Life of Brian</li><li>Holy Grail</li></ul>'
<span style="background-color: rgb(0, 153, 0);">header()允许快速建立选定级别的HTML标题(默认级别为2)</span>>>> header("Welcome to my home on the web")'<h2>Welcome to my home on the web</h2>'>>> header("This is a sub-sub-sub-sub heading",5)'<h5>This is a sub-sub-sub-sub heading</h5>'
<span style="background-color: rgb(0, 153, 0);">para()把一个文本块包围在HTML段落标记中间</span>>>> para("Was it worth the wait?We hope it was...")'<p>Was it worth the wait?We hope it was...</p>'
控制你的代码

合理安排Web应用的目录结构,保证它有条理有组织。

CGI让Web服务器运行程序

Python提供的Web服务器

from http.server import HTTPServer, CGIHTTPRequestHandlerport = 8080httpd = HTTPServer(('', port), CGIHTTPRequestHandler)print("Starting simple_httpd on port: " + str(httpd.server_port))httpd.serve_forever()
显示选手列表

创建generate_list.py程序。

<span style="font-size:18px;">import athletemodelimport yateimport glob  #利用”glob“模块可以向操作系统查询一个文件名列表data_files=glob.glob("data/*.txt")athletes=athletemodel.put_to_store(data_files)</span></span>
<span style="font-size:18px;"><span style="font-size:18px;">print(yate.start_response())print(yate.include_header("Coach Kelly's List of Athletes"))print(yate.start_form("generate_timing_data.py"))print(yate.para("Select an athlete from the list to work with: "))</span></span>
<span style="font-size:18px;"><span style="font-size:18px;">for each_athlete in athletes:    print(yate.radio_button("which_athlete",athletes[each_athlete].name))print(yate.end_form("Select"))print(yate.include_footer({"Home" : "/index.html"}))</span>
启动一个Web服务器并运行。进入webapp文件夹打开一个终端窗口,启动Web服务器:



在Web浏览器使用Web地址:http://localhost:8080.

Web服务器开始工作,它会把它处理的所有Web请求记录下来(显示在屏幕上)。


正确的生成了Web页面。



而且Web服务器记录了Web请求要允许"generate_list.py"CGI脚本。



创建另一个CGI脚本

需要生成一个新HTML页面,包含所有选手的前3个时间

点击一个单选钮然后按“选择”按钮时,会有新的Web请求发送到服务器。Web请求会指定要执行的CGI脚本,还会提供表单的数据。Web服务器会把表单数据作为输入发送到这个CGI脚本。可以使用CGI模块来访问表单数据。

<span style="font-size:18px;">import cgiform_data=cgi.FieldStorage()athlete_name=form_data['which_athlete'].value</span>
编写generate_timing_data.pyCGI脚本代码。

<span style="font-size:18px;">import cgitbcgitb.enableimport cgiimport athletemodelimport yateathletes=athletemodel.get_from_store()form_data=cgi.FieldStorage()athlete_name=form_data['which_athlete'].valueprint(yate.start_response())print(yate.include_header("Coach Kelly's Timing Date"))print(yate.header("Athlete:"+athlete_name+",DOB:"+athletes[athlete_name].dob+"."))print(yate.para("The top times for this athlete are:"))print(yate.u_list(athletes[athlete_name].top3))print(yate.include_footer({"Home":"/index.html","Select another athlete":"generate_list.py"}))</span>

测试驱动





遇到一个TypeError异常。

在CGI脚本的最前面增加这两行代码,启动Python的CGI跟踪技术。

<span style="font-size:18px;">import cgitbcgitb.enable()</span>

在Web浏览器后退,再次按下“选择”按钮。



top3()未定义!!!

一个小改变会让一切大不同

AthleteList类的top3()方法被重新指定为一个类属性。

<span style="font-size:18px;">@property  #这个修饰符允许访问"top3()"返回的数据时把它看作是一个类属性。    def top3(self):        return(sorted(set([self.sanitize(t) for t in self]))[0:3])</span>
要把top3()方法看作是一个类属性,应当这样调用:

<span style="font-size:18px;">print(yate.u_list(athletes[athlete_name].top3))</span>
而不是

<span style="font-size:18px;">print(yate.u_list(athletes[athlete_name].top3()))</span>
<span style="font-size:18px;">测试驱动结果:<img src="" alt="" /></span>








0 0
原创粉丝点击