zoukankan      html  css  js  c++  java
  • 跨域问题

      什么是同源策略

        同源策略(Same origin policy) 是一种约定,它是浏览器最核心也最基本的安全功能,如果缺少了同源策略,则浏览器的正常功能可能都会受到影响.可以说Web是构建在同源策略基础之上的,浏览器只是针对同源策略的一种实现.

      具体体现:

        比如:我在本地上的域名是127.0.0.1:8000,请求另外一个域名:127.0.0.1:8001 一段数据

        浏览器上就会报错,这就是同源策略的保护,如果浏览器对javascript没有同源策略的保护,那么一些重要的机密网站将会很危险

      CORS(跨域资源共享)简介

        CORS需要浏览器和服务器同时支持.目前,所有浏览器都支持该功能,IE浏览器不能低于IE10.

        整个CORS通信过程,都是浏览器自动完成,不需要用户参与.对于开发者来说.CORS通信与同源的AJAX通信没有差别,代码完全一样.浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息,有时还会多出一次附加的请求,但用户不会有感觉.

        因此,实现CORS通信的关键是服务器.只要服务器实现了CORS接口,就可以跨源通信.

      CORS基本流程

        浏览器将CORS请求分成两类:简单请求(simple request) 和非简单请求(not-so-simple request).

        浏览器发出  CORS简单请求,只需要在头信息之中增加一个Origin字段.

        浏览器发出CORS非简单请求,会在正式通信之前,增加一次HTTP查询请求,称为"预检"请求(preflight).

        浏览器先询问服务器,当前网页所在的域名是否在服务器的许可名单之中,以及可以使用哪些HTTP动词和头信息字段.只有得到肯定答复,浏览器才会发出正式的XMLHttpRequest请求,否则就报错.

      简单请求和非简单请求

        只要同时满足一下两大条件,就属于简单请求.

        (1)请求方法是以下三种方法之一:

        HEAD

        GET

        POST

        (2)HTTP的头信息不超出以下几种字段:

        Accept

        Accept-Language

        Content-Language

        Last-Event-ID

        Content-Type: 只限于三个值application/x-www.form-urlencoded.multipart/form-data.text/plain

        反之就属于非简单请求

      简单请求和非简单请求的区别

        简单请求:一次请求

        非简单请求:两次请求,在发送数据之前会先发一次请求用于做"预检",只有"预检"通过后才再发送一次请求用于数据传输

        关于"预检"

          请求方式:OPTIONS

          "预检"其实做检查,检查如果通过则允许传输数据,检查不通过则不再发送真正想要发送的消息

          如何"预检"

          =>如果复杂请求是PUT等请求,则服务端需要设置允许某请求,否则"预检"不通过

          Access-Control-Request-Method

          =>如果复杂请求设置了请求头,则服务端需要设置允许某请求头,否则"预检"不通过

          Access-Control-Request-Headers

      对于简单请求

        解决方法:

        服务器设置响应头: Access-Control-Allow-Origin = '域名'或'*'

      对于非简单请求

        解决方法:

        将下述两个都设置一下就行了

          "预检"请求时,允许请求方式则需服务端设置响应头: Access-Control-Request-Method

          "预检"请求时,允许请求头则需服务端设置响应头: Access-Control-Request-Headers

        Django是支持CORS的,我们可以直接在视图函数中为响应头设置上述的属性,也可以写在中间件中,就不需要再每次设置了

      自定制CORS中间件

      

    from django.utils.deprecation import MiddlewareMixin 
    class CORSMiddle(MiddlewareMixin): 
        white_domains = ['127.0.0.1:8000', '127.0.0.1:8002', '127.0.0.1:8003'] 
        def _get_ip(self, request): 
            domain = request.META.get('HTTP_HOST') 
            return domain 
        def _check_domain(self, request, response): 
            domain = self._get_ip(request) 
            for domain_name in self.white_domains: 
                if domain == domain_name: 
                    response['Access-Control-Allow-Origin'] = domain
                    break 
                if request.method == 'OPTIONS': 
                    response['Access-Control-Allow-Methods'] = 'PUT, DELETE, GET, POST' 
                    response['Access-Control-Allow-Headers'] = 'Content-Type' 
                return response 
        def process_response(self, request, response): 
            result = self._check_domain(request, response) 
            return result
  • 相关阅读:
    Spring Boot 使用 Dom4j XStream 操作 Xml
    Spring Boot 使用 JAX-WS 调用 WebService 服务
    Spring Boot 使用 CXF 调用 WebService 服务
    Spring Boot 开发 WebService 服务
    Spring Boot 中使用 HttpClient 进行 POST GET PUT DELETE
    Spring Boot Ftp Client 客户端示例支持断点续传
    Spring Boot 发送邮件
    Spring Boot 定时任务 Quartz 使用教程
    Spring Boot 缓存应用 Memcached 入门教程
    ThreadLocal,Java中特殊的线程绑定机制
  • 原文地址:https://www.cnblogs.com/xiaocaiyang/p/10458494.html
Copyright © 2011-2022 走看看