AJAX之跨域请求

  1、引子

  我如今开启了两个django项目,分别叫Demo1和Demo2,Demo1中有一个路径‘http://127.0.0.1:8000/index/’,对应的视图是index视图返回一个index页面,页面中只有一个button按钮,按钮绑定了一个单击事件,点击以后会发送一个ajax请求,请求的路径为‘http://127.0.0.1:8001/ajax/’,Demo1的ip和端口号是:‘http://127.0.0.1:8000/’,Demo2的ip和端口号是:‘http://127.0.0.1:8001/’。前面一段描述就是在Demo1项目的一个页面向Demo2项目发送一个ajax请求,在Demo2有对应的路径和视图来处理请求,并返回值。咱们运行一下。报错以下:html

  这就是一个已拦截的跨域请求的错误,错误内容是CORS头少“Access-Control-Allow-Origin”。这就是咱们用ajax发送一个跨域请求出现的错误,这就是今天我要处理的一个问题。前端

  2、同源策略

  同源策略(Same origin policy)是一种约定,它是浏览器最核心也最基本的安全功能,若是缺乏了同源策略,则浏览器的正常功能就会受到影响,能够说web是构建在同源策略基础之上的,浏览器只是针对同源策略的一种实现。web

同源策略,它是由Netscape提出的一个著名的安全策略。如今全部支持JavaScript的浏览器都会用这个策略。所谓的同源,即指:域名、端口、协议同时相同。好比说你用ajax发请求时,会检查你发送的请求是否发送路径是否ajax所在的JavaScript属于同源,便是二者的ip、端口、协议相同。若是不是同源,浏览器就会报错,提示拒绝访问。ajax

  想要实现用ajax发送跨域请求有两种方式,一是jsonp,而是cors。django

  3、基于JSONP的ajax跨域请求

  1,版本1

  咱们在用script引入js代码,有时用的是网上路径,意思就是在咱们的项目去请求其余项目的,如:json

  这种状况下是能够请求成功的,因而,咱们能够运用这一功能,给button按钮绑定事件,事件的功能是建立一个script标签,而后添加到页面上,这样当咱们点击button按钮就会发送一个跨域请求,并且仍是容许的。跨域

  Demo1项目index.html页面上添加js代码:浏览器

复制代码
function get_ele_script(url){
            var ele_script=$('<script>');
            ele_script.attr('src',url);
            ele_script.attr('id','ele_script');
            $('body').append(ele_script);
            $('#ele_script').remove()
        }
 $('.cli').click(function () {
            get_ele_script('http://127.0.0.1:8001/ajax/')
        })
复制代码

  这样就成功用点击事件发送了一个跨域请求,在Demo2的视图中,有返回值,但若是我不定义返回的值,HTML文件就会报没有定义的错误,并且视图返回的值在前端页面是以变量的形式显示。安全

  Demo2项目的视图:app

def ajax(request):
    dic={'name':'zhnag','age':46}
    dic_json=json.dumps(dic)
    return HttpResponse('f(%s)'%dic_json)

  此时,咱们就应该在HTML文件中定义f()

function f(params) {
            console.log(params);
            console.log(typeof params);
        }

  这样,当咱们点击button时,会往Demo2发送一个请求,Demo2会返回带参数的f(),因为已经定义了f()函数,因此当返回时会执行这个函数,这种方法就实现了发送跨域请求,而后接受返回值,并对返回值进行处理。

  2,版本2

  其实版本1在Demo2项目中把返回的函数名写死了,若是照版本1,全部来反问Demo2的浏览器都会获得一个叫f()的函数,咱们在声明的时候也只能写成f()。其实咱们能够这样在请求的时候把咱们定义的函数名一块儿发过去,让他就给我返回我发送过去的函数,这样,咱们定义函数名就能够随意了。

  Demo1项目的index.html页面上就应这样写:

function f(params) {
            console.log(params);
            console.log(typeof params);
        }
$('.cli').click(function () {
            get_ele_script('http://127.0.0.1:8001/ajax/?callbacks=f')

  Demo2项目的视图:

def ajax(request):
    a=request.GET.get('callbacks')
    dic={'name':'zhnag','age':46}
    dic_json=json.dumps(dic)
    return HttpResponse('%s(%s)'%(a,dic_json))

  3,版本3

  上面两个版本就没有基于ajax请求,这个版本就基于ajax发送跨域请求。

  Demo1项目下的index.html页面上:

复制代码
function f(params) {
            console.log(params);
            console.log(typeof params);
        }
$('.cli').click(function () {
            $.ajax({
                url:'http://127.0.0.1:8001/ajax/callbacks=f',
                type:'GET',
                dataType:'jsonp',
            })
        })
复制代码

  Demo2项目的视图:

def ajax(request):
    a=request.GET.get('callbacks')
    dic={'name':'zhnag','age':46}
    dic_json=json.dumps(dic)
    print(type(dic_json))
    return HttpResponse('%s(%s)'%(a,dic_json))

  4,版本4

  这也是基于ajax发送的跨域请求,只是比上一版本更简单。

  Demo1项目下的index.html:

复制代码
function f(params) {
            console.log(params);
            console.log(typeof params);
        }
$('.cli').click(function () {
            $.ajax({
                url:'http://127.0.0.1:8001/ajax/',
                type:'GET',
                dataType:'jsonp',
                jsonp:'callbacks',
                jsonpCallback:'f'
            })
        })
复制代码

  Demo2项目的视图:

def ajax(request):
    a=request.GET.get('callbacks')
    dic={'name':'zhnag','age':46}
    dic_json=json.dumps(dic)
    print(type(dic_json))
    return HttpResponse('%s(%s)'%(a,dic_json))

  5,版本5(终极版本)

  基于ajax的跨域请求,并把返回值给ajax的success。

  Demo1项目下的index.html:

复制代码
$('.cli').click(function () {
            $.ajax({
                url:'http://127.0.0.1:8001/ajax/',
                type:'GET',
                dataType:'jsonp',
                jsonp:'callbacks',
                success:function (data) {
                    console.log(data)
                }
            })
        })
复制代码

  Demo2项目下的视图:

def ajax(request):
    a=request.GET.get('callbacks')
    dic={'name':'zhnag','age':46}
    dic_json=json.dumps(dic)
    return HttpResponse('%s(%s)'%(a,dic_json))

  这个版本就比较简单,书写方便

  4、基于CORS的ajax跨域请求

  最开始,咱们讲的报错是:CORS头少:‘Access-Control-Allow-Origin’,既然少一个,咱们就加一个,就能够了,其实这种方式才是最经常使用。

  Demo1项目下的index.html:

复制代码
$('.cli').click(function () {
            $.ajax({
                url:'http://127.0.0.1:8001/ajax/',
                type:'get',
                data:{'a':1},
                success:function (res) {
                    console.log(res)
                }
            })
        })
复制代码

  Demo2项目下的视图:

def ajax(request):
    dic = {'name': 'zhnag', 'age': 46}
    http=HttpResponse(json.dumps(dic))
    http['Access-Control-Allow-Origin']='http://127.0.0.1:8000'     #这至关于加了一个白名单,对于‘http://127.0.01:8000’的跨域请求就容许通行
    return http

 

 

出处:https://www.cnblogs.com/12345huangchun/p/10339812.html