Django做了大量工作简化我们的Web开发工作, 这其中当然也包括接收来自客户端的数据这一普遍需求. 大部分时候,从客户端传入的数据主要是FORM的POST数据,和来自URL的GET数据, 在Django中对应了HttpRequest对象的POST和GET属性, 例如读取FORM表单中的用户名username输入框的内容:

def someView(request):    username = request.POST.get('username');    ......

在WEB2.0时代, 如果我们从客户端传入的是JSON格式的数据, 上面的方法就无法正常读出数据了. 可能有人要问,为什么一定要用JSON格式的数据, 不直接编码成FORM形式的数据呢? 因为JSON格式能够更好地体现数据的层次结构, 而FORM形式的数据往往只能是单层.

下面通过一个例子演示如何从HTML中发送json,并在Django中接收json数据.

在HTML中,可以通过JSON对象将数据以Json格式发送到服务器,假设在play.html中有以下内容要发送到服务器:

  • 用户名username

  • 密码password

  • 一个数组,其中每个元素包含: 游戏编号game_id和得分level

那么, 可以使用如下JavaScript(JQuery)代码:

$(function() { $('#upload').click(function() { var json_obj = { username: $('#username').val(), password: $('#password').val(), datas: [         {         game_id: $('#datas1').val(),         level: $('#level1').val()         },         {         game_id: $('#datas2').val(),         level: $('#level2').val()         }    ] };    //JSON对象    var json_str = JSON.stringify(json_obj);    //将JSON对象转变成JSON格式的字符串     $.post("/datasave", json_str, callback, "json"); }) function callback(json){ $('#response').html('code:'+json['status']['code'] + "\tmessage:" + json['status']['message']); }})

在上面的代码中, 数据将发送到/datasave, 在Django中,视图方法若还使用request.POST来读数据, 得到的全是None, 这时得使用request.raw_post_data, 并借助simplejson来将其转换为字典dict数据类型, 然后的使用就简单了,如下:

def datasave(request):    dict = {}    info = 'Data log save success'    try:        if request.method == 'POST':            req = simplejson.loads(request.raw_post_data)            username = req['username']            password = req['password']            datas = req['datas']            game_id1 = datas[0]['game_id']   except:        import sys        info = "%s || %s" % (sys.exc_info()[0], sys.exc_info()[1])    dict['message']=info    dict['create_at']=str(ctime())    json=simplejson.dumps(dict)    return HttpResponse(json)

request.raw_post_data表示的是从客户端发送过来的原始数据,为了纯字符串, 通过simplejsonloads方法将其转换为字典数据类型req.

上面的代码也演示了如何以JSON格式作为响应值, 而非HTML, 即通过simplejsondumps方法, 将字典数据dict序列化为字符串形式,将通过HttpResponse返回.

from:http://blog.donews.com/crazyone/archive/2011/06/23/816689.aspx