今天看啥  ›  专栏  ›  崐崐

XHR跨域问题CORS解决方案(后台服务webservice)

崐崐  · 简书  ·  · 2020-01-07 15:30

前言

对于前后端分离的网站设计,跨域几乎是无法避免的。对于跨域的解决方案在网上也是一搜一大堆,众说纷纭,也许说的很有道理,但是很多都是不太符合自己的需要。我初次遇到这个跨域问题也是非常头疼,看着那个跨域警告们非常无奈。适合自己的才是最好的,最终我鼓捣明白了。采用后台服务webservice,前端随便一段JS脚本测试。实现确实非常简单。下面就将我的处理过程写下来,希望对需要的网友们能够有所帮助吧。

1.常见的跨域场景

URL                                      说明                    是否允许通信
http://www.domain.com/a.js
http://www.domain.com/b.js         同一域名,不同文件或路径           允许
http://www.domain.com/lab/c.js

http://www.domain.com:8000/a.js
http://www.domain.com/b.js         同一域名,不同端口                不允许
 
http://www.domain.com/a.js
https://www.domain.com/b.js        同一域名,不同协议                不允许
 
http://www.domain.com/a.js
http://192.168.4.12/b.js           域名和域名对应相同ip              不允许
 
http://www.domain.com/a.js
http://x.domain.com/b.js           主域相同,子域不同                不允许
http://domain.com/c.js
 
http://www.domain1.com/a.js
http://www.domain2.com/b.js        不同域名                         不允许

2. CORS 简介

CORS:全称"跨域资源共享"(Cross-origin resource sharing)。
CORS需要浏览器和服务器同时支持,才可以实现跨域请求,目前几乎所有浏览器都支持CORS,IE则不能低于IE10。CORS的整个过程都由浏览器自动完成,前端无需做任何设置,跟平时发送ajax请求并无差异。so,实现CORS的关键在于服务器,只要服务器实现CORS接口,就可以实现跨域通信。

浏览器默认的安全限制为同源策略,即JavaScript或Cookie只能访问同源(相同协议,相同域名,相同端口)下的内容。但由于跨域访问资源需要,出现了CORS机制,这种机制让web服务器能跨站访问控制,使跨站数据传输更安全。CORS需要阅览器和服务器同时支持,目前,主流的阅览器都支持cors。
更多信息请查阅: https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Access_control_CORS

3.跨域Demo

客户端请求 HTML示例代码:

<html>
<head>
    <meta charset="utf-8" />
    <title>This is Demo</title>
    <script>
        function ss() {
            console.log("开始 测试");
            var url = "http://localhost:27992/KYZ_BXP.asmx/HelloWorld?" + Math.random();
            var xhr = new XMLHttpRequest();
            xhr.onreadystatechange = function() {
                console.log("响应结果:" + xhr.readyState);
                console.log("状态码结果:" + xhr.status);

                if (xhr.readyState === 4) {
                    if (xhr.status == 200) {
                        alert(xhr.responseText);

                    }
                }
            }

            xhr.open('get', url, true);
            // xhr.setRequestHeader('X-Custom-Header', 'value');
            xhr.send(null);

        }
        ss();
    </script>
</head>

<body>
    This is a demo
</body>

</html>

服务端(webservice)代码:

/// <summary>
    /// 服务端简单说明
    /// </summary>
    [WebService(Namespace = "http://tempuri.org/")]
    [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
    [System.ComponentModel.ToolboxItem(false)]
    // 若要允许使用 ASP.NET AJAX 从脚本中调用此 Web 服务,请取消对下行的注释。
    // [System.Web.Script.Services.ScriptService]
    public class KYZ_BXP : System.Web.Services.WebService
    {


        [WebMethod]
        public string HelloWorld()
        {
          
            return "Hello World123";
        }
}

运行结果:


捕获01.JPG

捕获02.JPG

从中可以看出,由于客户端访问请求和服务端响应接口端口不同,属于跨域请求类型,结果就出现了跨域警告。实际返回的数据被浏览器劫持了,无法正常显示弹窗结果。
要想实现CORS跨域,只需要对服务端代码做如下改动:

/// <summary>
    /// 服务端简单说明
    /// </summary>
    [WebService(Namespace = "http://tempuri.org/")]
    [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
    [System.ComponentModel.ToolboxItem(false)]
    // 若要允许使用 ASP.NET AJAX 从脚本中调用此 Web 服务,请取消对下行的注释。
    // [System.Web.Script.Services.ScriptService]
    public class KYZ_BXP : System.Web.Services.WebService
    {


        [WebMethod]
        public string HelloWorld()
        {
            Context.Response.AddHeader("Access-Control-Allow-Origin", "*");//关键 没有这一句的设置 就无法完成跨域请求
            Context.Response.AddHeader("Access-Control-Allow-Methods", "*");
            Context.Response.AddHeader("Access-Control-Allow-Headers", "x-requested-with,content-type");
            return "Hello World123";
        }
}

注意:修改代码后,需要将服务重启一下。
然后再试一下,结果如下:


捕获03.JPG

捕获04.JPG

可以看到,请求的数据已经能够正常的加载弹出。XHR跨域问题已经解决!
Demo 运行环境:.NET4.0 WebService; .Net4.0 Web;狐火浏览器64位 71.0。

4.最后

总的来说,使用CORS简单请求,非常容易,对于前端来说无需做任何配置,与发送普通ajax请求无异。CORS的配置,完全在后端设置,配置起来也比较容易,目前对于大部分浏览器兼容性也比较好。CORS优势也比较明显,可以实现任何类型的请求。




原文地址:访问原文地址
快照地址: 访问文章快照