`
sdmzhu3
  • 浏览: 30857 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

分享一个简单的python模板引擎

阅读更多
python模板引擎也很多,但是希望可以使用python原生的来作模板 而不用在创建一个新的语法, 但是python本身的缩进又不适合做模板 所以要把缩进 去掉就可以了




#coding:utf-8
__author__="sdm"
__author_email='sdmzhu3@gmail.com'

__date__ ="$2009-8-25 21:04:13$"

'''
pytpl 类似php的模板类


'''
import sys
import StringIO
import os.path
import 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):
        '''
        设置模板变量
        '''
        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):
        '''
        执行这个模板
        '''
        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
                    #进入代码状态
                    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:
                #查找结束标记
                pos_end=code.find(flag_end,i)
                if(pos_end>-1):
                    #需要跳过<? 这个标记
                    t_code=place_t_code(code[pos_start+2:pos_end],indent)

                    #跳过?>结束标记
                    pos_end+=2
                    py_code.append(t_code)
                else:
                    #没查找到直接结束
                    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_str


def test():
    tpl=Pytpl('./');
    tpl.set('title', '标题3')
    print tpl.tpl('test.html')


    pass


if __name__ == "__main__":
    test()

test.py
import pytpl
tpl=pytpl.Pytpl('./')
tpl.set('title', 'test title')
print tpl.tpl('test.html')

执行结果
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <title>test title</title>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
  </head>
  <body>

small3
11
    
  </body>
</html>

模板test.html
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <title><?=title?></title>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
  </head>
  <body>

<?
      
def add(a,b){
    if(a+b<10){
        echo('small');
    }
return a+b;
}
echo(add(1,2));
echo("\n");
echo(add(10,1));
      ?>
    
  </body>
</html>


---
编译的中间模板
#coding:utf-8
import sys;global echo;echo=self.output.write
echo ( '<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">\n<html>\n  <head>\n    <title>')
echo ( '%s' % (title))
echo ( '</title>\n    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">\n  </head>\n  <body>\n\n')


def add(a,b):
    if(a+b<10):
        echo('small');
        
    return a+b;
    
echo(add(1,2));
echo("\n");
echo(add(10,1));

echo ( '\n    \n  </body>\n</html>\n\n' ) 



项目已经加入 googlecode
http://code.google.com/p/pythontpl/
分享到:
评论
1 楼 sdmzhu3 2010-03-14  
增加了 include 功能
在模板中包含代码直接写 即可

<? include('head.html')?>

tpl.set 的数据都是继承的

相关推荐

    Python实现的简单模板引擎功能示例

    本文实例讲述了Python实现的简单模板引擎功能。分享给大家供大家参考,具体如下: #coding:utf- 8 __author__=sdm __author_email='sdmzhu3@gmail.com' __date__ =$2009-8-25 21:04:13$ '' ' pytpl 类似 php的模板...

    个人写的图片分享社区 基于Python开发.zip

    这个压缩包文件“个人写的图片分享社区 基于Python开发.zip”似乎是一个个人项目,由熟悉Python编程语言的开发者创建。项目的核心是一个图片分享社区平台,它利用Python的灵活性和强大功能来实现用户上传、浏览和...

    Python使用模板共8页.pdf.zip

    这个压缩包可能是为了方便分享和存储一系列有关Python模板使用的教程或指南。从标签来看,我们可以推测这份资料可能涵盖了多种IT相关的主题,包括数据集的处理、源代码的编写、计算机学习资源、Python编程、STM32微...

    每天分享几个python项目 —— simple-book-library-system-using-python

    "simple-book-library-system-using-python"项目提供了一个很好的机会,让我们通过实际操作来学习和理解Python后端开发的基本概念。这个项目旨在创建一个简单的图书管理系统,用于记录书籍信息、借阅情况以及管理...

    每天分享几个python项目 —— simple-mobile-contact-management-system

    本项目名为"simple-mobile-contact-management-system",是一个基于Python开发的简易移动联系人管理系统。这个系统旨在帮助用户方便地管理他们的个人联系信息,包括添加、删除、查找和编辑联系人。它展示了Python在...

    Python-BBS论坛源码

    这个项目旨在提供一个功能完善的论坛系统,让用户能够进行交流、分享知识和讨论各种话题。通过分析这个源码,我们可以深入理解Python在Web开发中的应用,以及如何构建一个动态的交互式社区。 首先,Python是这个BBS...

    面试宝典Python 391页.pdf

    6. **Web开发框架**:对于Python的Web开发,可能会涉及到Django和Flask这两个主流框架的基本使用,包括路由配置、模板引擎、数据库模型、表单处理以及中间件等。 7. **数据分析与科学计算**:讲解了NumPy、Pandas和...

    python3flask开发的知识分享系统.zip

    在本项目中,我们关注的是使用Python3和Flask框架构建一个知识分享系统。Flask是一个轻量级的Web服务器网关接口(WSGI)Web应用框架,它为开发者提供了构建Web应用的基本工具,而无需复杂的配置。让我们深入探讨如何...

    Python-Vataxia社交网络引擎

    Vataxia 社交网络引擎是一款基于Python语言开发的社交平台构建框架,旨在为开发者提供一个高效、可扩展且易于定制的解决方案。Python以其简洁的语法和强大的库支持,成为了开发此类应用程序的理想选择。Vataxia的...

    python tornado

    Python Tornado是一个强大的异步网络库,用于构建高性能、高并发的网络应用。Tornado源自FriendFeed团队,后来被Facebook收购并开源,现在是开源社区维护的一个独立项目。它的核心特性包括HTTP服务器、Web框架、...

    Python-中国独立开发者项目列表分享大家都在做什么

    例如,开发者可能利用Django的ORM(对象关系映射)来处理数据库操作,或者使用Flask的模板引擎 Jinja2 来设计动态网页。 2. 数据分析与科学计算:Python在数据分析领域有着显著的优势,Pandas库提供了强大的数据...

    Python-TurboGears是一个混合Web框架既可以作为FullStack框架也可以作为Microframework

    作为一个全栈框架,TurboGears提供了一整套工具和组件,涵盖了Web应用开发的各个方面,包括模板引擎、数据库抽象层、ORM(对象关系映射)、URL路由、表单处理、身份验证和授权等。这种全方位的覆盖使得开发者无需...

    python小博客项目

    这个项目旨在提供一个简单易用、功能齐全的平台,让技术爱好者可以方便地发布文章、分享知识,并且进行互动交流。通过学习和实践这个项目,我们可以深入了解Python Web开发的基础,以及相关框架和库的应用。 该项目...

    Python-PyZhPython经典的技术文章的翻译和收集

    《Python-PyZhPython经典的技术文章的翻译和收集》是一个致力于分享与学习Python技术的资源库,它将readthedocs上的优质Python技术文章进行了收集和汉化,为中文读者提供了便利的学习材料。作为Python开发的学习教程...

    Python-IMGKit将HTML转换成图像Python库

    IMGKit是一个Python库,基于Node.js的`wkhtmltoimage`工具,能够将HTML代码或URL转换为高质量的JPEG或PNG图像。这个库的优势在于它能够保留HTML页面的布局、样式和JavaScript效果,使得生成的图片尽可能接近原始网页...

    Python于Web_2.0网站的应用

    Web 2.0是指互联网发展的一个阶段,它强调用户参与、互动和分享,催生了诸如博客、社交媒体、在线协作等新型应用。Python在Web 2.0网站中的应用主要体现在以下几个方面: 1. **Web框架**:Python有许多优秀的Web...

    Python-HelloGitHubcom网站源码

    这个项目提供了对网站源代码的深入理解,有助于学习者掌握如何用Python构建一个在线平台,特别是关注于代码管理和分享的社区网站。 首先,我们要了解Python在Web开发中的应用。Python以其简洁易读的语法和丰富的库...

    Python库 | py_thanos-0.0.1-py3-none-any.whl

    在Python的世界里,`py_thanos-0.0.1-py3-none-any.whl` 是一个特定版本的Python库的打包文件,名为`py_thanos`,版本为`0.0.1`,适用于Python 3环境。这种`.whl`格式的文件是一种预编译的Python二进制分发包,它...

    Python-Cactus设计师设计的静态站点生成器

    Python-Cactus是一款专为设计师打造的静态站点生成器,它基于Python编程语言,并采用了强大的Django模板引擎。Cactus的主要目的是简化静态网站的创建流程,使得设计师无需深入理解复杂的后端技术,也能轻松地构建...

Global site tag (gtag.js) - Google Analytics