本文實例講述了Python實現(xiàn)的簡單模板引擎功能。分享給大家供大家參考,具體如下:
#coding:utf- 8__author__="sdm"__author_email='sdmzhu3@gmail.com'__date__ ="$2009-8-25 21:04:13$"'' 'pytpl 類似 php的模板類'' 'import sysimport StringIOimport os.pathimport os#模 板的緩存_tpl_cache={}class Pytpl: def __init__(self,tpl_path='./' ): self.tpl_path=tpl_path self.data={} self.output = StringIO.StringIO() pass def set(self,name,value): '' ' 設(shè) 置模板變量 '' ' self.data[name]=value; pass def get(self,name): '' ' 得 到模板變量 '' ' t={} return t.get(name, '' ) pass def tpl(self,tplname): '' ' 渲 染模板 '' ' f=self.tpl_path+tplname if not os.path.exists(f): raise Exception('tpl:[%s] is not exists' % f) mtime=os.stat(f).st_mtime if not _tpl_cache.has_key(f) or _tpl_cache[f][ 'time' ]<mtime: src_code=self.__compile__(open(f).read()) try : t=open(f+'.py' , 'w' ) t.write(src_code) t.close() except: pass py_code=compile(src_code, f+'.py' , 'exec' ) _tpl_cache[f]={'code' :py_code, 'time' :mtime} else : py_code= _tpl_cache[f]['code' ] exec(py_code, {'self' :self}, self.data) return self.output.getvalue() def execute(self,code,data,tplname): '' ' 執(zhí) 行這個模板 '' ' py_file_name=tplname+'.py' f=open(py_file_name,'w' ) f.write(code) f.close() execfile(py_file_name, {'self' :self}, data) def __compile__ (self,code): '' ' 編 譯模板 查找 <?標記 '' ' tlen=len(code); flag_start='<?' flag_end='?>' # 默認普通標記 status=0 i=0 #分塊 標記 pos_end=0 pos_start=0 #縮 進 global indent indent=0 py_code=[] def place_t_code(c,t_indent): '' ' 基 本的代碼處理 '' ' global indent if (c[ 0 ]== '=' ): return ( ' ' * 4 *indent) + 'echo ( /'%s/' % (' +c[ 1 :]+ '))' lines=c.split("/n" ) t=[] for i in lines: indent2=indent tmp=i.strip(" /n/r" ) c=tmp[len(tmp)-1 :len(tmp)] # 判定最后一個字符 if (c== '{' ): indent+=1 tmp=tmp[0 :len(tmp)- 1 ]+ ":" elif(c=='}' ): indent-=1 tmp=tmp[0 :len(tmp)- 1 ] t.append((' ' * 4 *indent2) +tmp ) return "/n" .join(t) while 1 : if i>=tlen: break c=code[i]; if status== 0 : # 編譯加速 pos_start=code.find(flag_start,pos_end); if (pos_start>- 1 ): s=code[pos_end:pos_start] t_code= 'echo ( ' +repr(s)+ ')' t_code=' ' *indent* 4 +t_code if s: py_code.append(t_code) i=pos_start last_pos=i # 進入代碼狀態(tài) status=1 continue else : # 沒有沒有找到 pos_start=tlen t_code='echo ( ' +repr(code[pos_end:pos_start])+ ' ) ' t_code=' ' *indent* 4 +t_code py_code.append(t_code) break if status== 1 : # 查找結(jié)束標記 pos_end=code.find(flag_end,i) if (pos_end>- 1 ): # 需要跳過<? 這個標記 t_code=place_t_code(code[pos_start+2 :pos_end],indent) # 跳過?>結(jié)束標記 pos_end+=2 py_code.append(t_code) else : # 沒查找到直接結(jié)束 pos_end=tlen # 需要跳過<? 這個標記 t_code=place_t_code(code[pos_start+2 :pos_end],indent) py_code.append(t_code) break status=0 i=pos_end pass i+=1 py_code_str="#coding:utf-8/nimport sys;global echo;echo=self.output.write/n" py_code_str+="/n" .join(py_code) py_code_str=py_code_str.replace("/t" , " " ) return py_code_strdef test(): tpl=Pytpl('./' ); tpl.set('title' , '標題3' ) print tpl.tpl('test.html' ) passif __name__ == "__main__" : test()
新聞熱點
疑難解答