继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

[pthon2.7+django1.2+sae]博客评论的异步提交

holdtom
关注TA
已关注
手记 1842
粉丝 240
获赞 991


在开发博客系统或者是电商系统,总之,凡是带有回复、评论、回帖之类针对一个内容进行回复的功能的时候,现在几乎全部都是采用ajax异步提交,并且同步在正确的位置显示提交结果,而不是在提交之后重新刷新整个页面来显示提交之后的内容。

ajax由于google的使用,被广为推广。它在客户端使用javascript语言编写,使用XMLHttpRequest对象,实现和服务的数据交互,详细信息参看:AJAX - XMLHttpRequest 对象。

ajax提交或者获取数据,常用数据形式包括:

普通文本,自己定义规则,自己解析,无成熟类库,需要自己编写。

xml文本,使用xml解析器解析,有成熟类库。

json文本,流行的数据形式,以体积小、灵活而著称,有成熟类库。

ajax有几个好处:

防止页面刷新,因为提交的只是一部分内容,页面大部分内容没有发生改变,如果还是刷新进而重新获取全部的话,对于服务器和客户端都是一种负担。而且页面刷新还会产生其他问题,诸如重复提交之类的。

减轻服务器负担,可以为更多人提供服务。

也有一些缺点:

给编程和调试带来一些小麻烦,不够使用习惯了,还是有一些方法和窍门的。

由于是ajax,所以页面地址没有变化,所以如果想分享地址的话,难以实现。需要做额外的工作,例如把地址打印出来,让用户复制分享。

用户不容易知道自己的提交正在进行,因为页面没有刷新,用户可能会重新点击提交按钮。这也可以解决,提交过程给出明显提示,然后灰掉提交按钮,或者做一些防止重复提交的工作。

 

今天的例子是博客评论的异步提交,环境是python2.7+django1.2+SAE。

下面是后台的处理代码,view中的代码

 

                      

@csrf_exempt                  

def comment_new(request,blog_id): 

    blog=get_object_or_404(Blog,pk=blog_id) 

     

    #str(request.raw_post_data) 

    if request.method=="POST" and request.is_ajax(): 

        title=request.POST['comment_title'] 

        name=request.POST['comment_author_name'] 

        email=request.POST['comment_author_email'] 

        url=request.POST['comment_author_url'] 

        content=request.POST['comment_content'] 

        comment=Comment(title=title,author_name=name,author_email=email,author_url=url,content=content); 

        comment.blog=blog 

        comment=comment.save() 

 

        _dict={} 

        _dict["title"]=title 

        _dict["author_name"]=name 

        _dict["author_email"]=email 

        _dict["author_url"]=url 

        _dict["content"]=content 

          _dict["created_at"]=unicode(datetime.datetime.now()) 

     

     

    return HttpResponse(simplejson.JSONEncoder().encode(str(_dict))) 

 

 

def obj2dict(obj): 

    """ 

    summary: 

        将object转换成dict类型     

    """ 

    memberlist = [m for m in dir(obj)] 

    _dict = {} 

    for m in memberlist: 

        if m[0] != "_" and not callable(m): 

            _dict[m] = getattr(obj,m) 

 

    return _dict 

 

下面是前段页面的代码

 

<script type="text/javascript" src="/site_static/js/tiny_mce_jquery/tiny_mce.js"></script> 

<script type="text/javascript" src="/site_static/js/textareas_simple.js"></script> 

 

{% if comments %} 

<p> 

<h2>Comments:</h2> 

<div id="blog_comments"> 

{% for comment in comments %} 

<p> 

<a href="{{ comment.author_url }}">{{ comment.author_name }}</a>&nbsp;&nbsp;Said at {{ comment.created_at }} 

</p> 

<p> 

<h3>Title:</h3> 

{{ comment.title }} 

</p> 

<p> 

    <h3>Content:</h3> 

    {% autoescape off %} 

    {{ comment.content }} 

    {% endautoescape %} 

 

</p> 

<hr> 

{% endfor %} 

 

</div> 

 

</p> 

{% else %} 

<h2>you are the first comment man, please!</h2> 

{% endif %} 

<p> 

<h2>New Comment:</h2> 

<script type="text/javascript"> 

    $(document).ready(function(){ 

        $("#comment_form").submit(function(e){ 

                     

            var data=$('#comment_form').serialize(); 

             

            $.ajax({ 

                url:"/blog/{{ blog.id }}/comment/new/", 

                type:"POST", 

                contentType:"application/json; charset=utf-8", 

                dataType:"json", 

                data:data, 

                success:function(data, textStatus, jqXHR){ 

                    //alert(typeof(data)); 

                    datadata=data.replace(/'/gi,"\""); 

                    datadata=data.replace(/u"/gi,"\""); 

                    //alert(data); 

                    var obj=$.parseJSON(data); 

                    //alert(obj.content); 

                    $("#blog_comments").append("<p><a href=\""+obj.author_url+"\">"+obj.author_name+"</a>&nbsp;&nbsp;Said at "+obj.created_at+"</p><p>"+obj.title+"</p><p>"+obj.content+"</p><hr>"); 

                     

                    $("#comment_form")[0].reset(); 

                }, 

                error:function(jqXHR, textStatus, errorThrown){ 

                    alert(textStatus); 

                    alert(errorThrown); 

                } 

            });      

             

            return false; 

 

        }); 

    }); 

     

</script> 

<form method="post" action="" id="comment_form"> 

 

    {% csrf_token %} 

     

    {% comment %} 

    {% for field in comment_form %} 

    <p> 

        {{ field.errors }} 

        {{ field.label_tag }}:{{ field }} 

    </p> 

    {% endfor %} 

         

    {% endcomment %} 

 

    <p> 

    <label for="comment_title">{{ comment_form.title.label }}</label>: 

    <input type="text" id="comment_title" name="comment_title"  placeholder="Title"/> 

    {{ comment_form.title.help_text }} 

    </p> 

     

    <p> 

    <label for="comment_author_name">{{ comment_form.author_name.label }}</label>: 

    <input type="text" id="comment_author_name" name="comment_author_name" placeholder="author name"/> 

    {{ comment_form.author_name.help_text }} 

    </p> 

     

    <p> 

    <label for="comment_author_email">{{ comment_form.author_email.label }}</label>: 

    <input type="text" id="comment_author_email" name="comment_author_email"  placeholder="author email"/> 

    {{ comment_form.author_email.help_text }} 

    </p> 

     

    <p> 

    <label for="comment_author_url">{{ comment_form.author_url.label }}</label>: 

    <input type="text" id="comment_author_url" name="comment_author_url"  placeholder="author url"/> 

    {{ comment_form.author_url.help_text }} 

    </p> 

     

    <p> 

    <label for="comment_content">{{ comment_form.content.label }}</label>: 

    <textarea rows="10" cols="50" placeholder="content" id="comment_content" name="comment_content"></textarea> 

     

    {{ comment_form.content.help_text }} 

    </p> 

     

 

 

 

     

     

    <input type="submit" value="Save" /> 

</form> 

 

</p> 

在客户端用到了三个类库,分别是jquery-1.7.1.min.js,jquery.json-2.3.min.js和jquery.placeholder.min.js。

jquery是一个javascript类库,封装了javascript的很多操作,使用方便。

jquery.json是jquery的一个扩展,用来处理json文本和object之间的转换。

jquery.placeholder是jquery的一个扩展,用来实现在输入框没有内容的时候,添加一些提示信息。在输入框没有内容的时候,显示一些内容,提示你应该输入什么,输入的格式是什么。【placeholder是html5支持的一个属性,专门用来做提示的。】

 

datadata=data.replace(/'/gi,"\"");  

datadata=data.replace(/u"/gi,"\"");  

前段页面的上面这两句需要大家注意,第一句用正则表达式替换【单引号】为【双引号】,第二句替换【u+双引号】为【双引号】。这都是为了后面使用

$.parseJSON(data); 

做准备的。因为python后台返回的json字符串,也就是data的值是下面的格式

 

{'title':u'blog1','author':u'andyshi'} 

字符串中有单引号和u,这都需要替换,jquery解析的json必须是标准格式的,就是双引号,而且不能包含其他内容,所以我进行了替换,然后才可以使用jquery.json进行解析。

u是因为python后台返回的是unicode字符串的缘故。应该有办法解决,待我再研究一下,稍后补充上来。

 

补充:

之前ajax返回的数据还需要处理单引号和字母u的问题,现在修正一下。

首先将前端的ajax代码变更为

 

$.ajax({ 

                url:"/blog/{{ blog.id }}/comment/new/", 

                type:"POST", 

                contentType:"application/json; charset=utf-8", 

                 

                data:data, 

                success:function(data, textStatus, jqXHR){ 

 

                    var obj=$.parseJSON(data); 

                     

                    $("#blog_comments").append("<p><a href=\""+obj.author_url+"\">"+obj.author_name+"</a>&nbsp;&nbsp;Said at "+obj.created_at+"</p><p>"+obj.title+"</p><p>"+obj.content+"</p><hr>"); 

                    $("#comment_form")[0].reset(); 

                }, 

                error:function(jqXHR, textStatus, errorThrown){ 

                    alert(textStatus); 

                    alert(errorThrown); 

                } 

            });  

很明显,就是去掉了ajax参数dataType,返回的结果变成在服务端组织好的json字符串,然后用jquery.json解析成对象。顺便里面的两次替换就可以去掉了。

python后台的view代码变更为

 

@csrf_exempt                  

#@csrf_protect 

def comment_new(request,blog_id): 

    blog=get_object_or_404(Blog,pk=blog_id) 

     

    #str(request.raw_post_data) 

    if request.method=="POST" and request.is_ajax(): 

        title=request.POST['comment_title'] 

        name=request.POST['comment_author_name'] 

        email=request.POST['comment_author_email'] 

        url=request.POST['comment_author_url'] 

        content=request.POST['comment_content'] 

        comment=Comment(title=title,author_name=name,author_email=email,author_url=url,content=content); 

        comment.blog=blog 

        comment=comment.save() 

         

    #return string 

    return HttpResponse("{\"title\":\"%s\",\"author_name\":\"%s\",\"author_email\":\"%s\",\"author_url\":\"%s\",\"content\":\"%s\",\"created_at\":\"%s\"}"  

        % (title,name,email,url,content,unicode(datetime.datetime.now()))) 

直接返回我们自己构造的字符串,构造的时候就使用双引号拼接。这样就给前台省去了正则替换的操作。

 

©著作权归作者所有:来自51CTO博客作者virusswb的原创作品,如需转载,请注明出处,否则将追究法律责任

ajaxpostpythonPython

1

分享


打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP