Access-Control-Allow-Origin跨域解决及详细介绍

守听  金牌会员 | 2024-8-12 02:45:29 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 551|帖子 551|积分 1653

起首,跨域不是问题。是一种安全机制。 这是你在开发时、上线前就必须提前思量到的安全问题并且接纳合适的本领去避免这个问题带来的程序错误。不过通常情况下,前端开发的小同伴们都非常坚信后端小同伴的接口肯定已经处置惩罚好了跨域这个需求。然而究竟上许多的前端拿到的都是没有解决跨域的接口。又出于某种原因不便与后端交涉并且对方视乎态度不是很友爱。在这种情况下作为前端的小同伴们内心简直一万头草泥马飞过。
  不过现在你不必为之犯困了,哪个后端要是不帮忙处置惩罚跨域导致的一系列问题的话,请将本文直接甩给后台,脸必须打响。要解决跨域必须由后端来一起协同解决,且主要解决工作在后端。
  为了能够更加快速的解决跨域带来的问题,下面对跨域进行详细介绍。
  一、跨域是什么
  跨域是浏览器加载了与当前域名、协议、端口差别另一站点下的资源,这与各大支持JavaScript的浏览器的同源策略是违背的。所谓同源策略,它是由Netscape提出的一个著名的安全策略。现在所有支持JavaScript 的浏览器都会使用这个策略。所谓同源是指,域名,协议,端口雷同。
  好比说,下面的几个域名是同源的:
  http://example.com/
  http://example.com:80/
  http://example.com/path/file
  它们都具有雷同的协议、雷同的域名、雷同的端口(不指定端口默认80)。
  而下面几个域名是差别源的:
  http://example.com/
  http://example.com:8080/
  http://www.example.com/
  https://example.com:80/
  https://example.com/
  http://example.org/
  http://ietf.org/
  它们有差别的协议或差别的域名或差别的端口,要留意顶级域名和二级域名也是认为差别的域名。
  二、解决跨域导致的问题
  跨域并不会克制请求的发出,也不会克制请求的担当,跨域是浏览器为了保护当前页面,你的请求得到了相应,浏览器不会把相应的数据交给页面上的回调,取而代之的是去提示你这是一个跨域数据。提示就是一个报错提示,就像如许:
              
           
  我们知道了浏览器是如那边置惩罚的了,才气对症下药来解决这个问题,下面介绍几种常用的跨域解决方法:
  1、CORS,跨域资源共享
  这是最靠谱也黑白常科学的解决方案,通过上面的截图我们可以看到,它提示了一个:从某某位置请求的资源被阻挡了,因为没有在相应头里发现:"Access-Control-Allow-Origin"的相应头。看到这个错误,我们不得不百度一下,这个Access-Control-Allow-Origin是个何方神圣。
  通过Access-Control-Allow-Origin相应头,就告诉了浏览器。如果请求我的资源的页面是我这个相应头里记录了的"源",则不要拦截此相应,允许数据通行。好比说下面示列了一个场景:
  // 从 http://example.com 界面发出了一个请求到:http://example2.com,因为差别源,导致了跨域。
  // 而 http://example2.com 返回了下面的相应头:
  Content-Type: application/json;charset=utf-8
  Content-Length: 3210
  Server: apache
  Access-Control-Allow-Origin: http://example.com
  1. // 从 http://example.com 界面发出了一个请求到:http://example2.com,因为不同源,导致了跨域。
  2. // 而 http://example2.com 返回了下面的响应头:
  3. Content-Type: application/json;charset=utf-8
  4. Content-Length: 3210
  5. Server: apache
  6. Access-Control-Allow-Origin: http://example.com
  7. ————————————————
  8. 版权声明:本文为CSDN博主「Microanswer」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
  9. 原文链接:https://blog.csdn.net/MicroAnswer/article/details/102913571
复制代码
由于浏览器检测到 http://example2.com 的相应头中显示的写着:Access-Control-Allow-Origin: http://example.com,也就是,如果请求数据的源是 http://example.com 则可以允许访问返回的数据。如许浏览器就不会抛出错误提示,而是正确的将数据交给你的ajax回调。
  在这个过程中跨域也存在,但跨域并没有导致问题了。因为后端的相应充分思量到了某个页面源要使用这个资源,早就帮对方做好了跨域资源共享。这才可以顺利的进行对接。
  以是,要最简单解决跨域导致的问题,只需要后端相应时,在相应头里指定允许调用资源的源就可以了。除了设定指定的源以外,你还可以直接写一个*号,如许就表现:此数据允许被任何其他的源进行获取。
  现在,你相识了Access-Control-Allow-Origin,其实除了它,还有与之相关的更多字段,它们也起到了更多的个性定值效果。下面进行了详细介绍。
  header头字段含义取值
  Access-Control-Allow-Credentials相应头表现是否可以将对请求的相应袒露给页面。返回true则可以,其他值均不可以。true/false
  Access-Control-Allow-Headers表现此次请求中可以使用那些header字段符合请求头规范的字符串
  Access-Control-Allow-Methods表现此次请求中可以使用那些请求方法GET/POST(多个使用逗号隔开)
  2、使用JSONP方案
  当服务端没有返回Access-Control-Allow-Origin如许的字段时,是否就意味着不能使用此资源了吗?不!只能说不建议使用此资源了。但我们还有另一种办法,那就是通过JSONP。看到这个名字,似乎和json有关,说有也有,但也可以说没有,JSONP只是大多数乃至全部人们对这种解决办法的称谓。
  为了更灵活的使用这中解决办法,就必须要先相识它的实现原理。我们知道,在页面内使用ajax加载别的域名下的数据时,是会被跨域克制的。那有没有办法让我们的请求不通过页内的ajax,而是让浏览器直接走这个请求?
  有!如果你富足细心,你会发现,<script>节点当有一个src值时,浏览器就会去加载这个js,然后并执行这个js文件,同样的,<img>也可以设置一个src,浏览器会加载这个图片并显示。那么,其中<script>节点在获取到js后还会执行,而我们的业务逻辑代码也是执行在雷同的js情况下的。我们能不能想办法,让我们的请求不通过ajax,而是通过给body中追加一个<script>节点,这个节点的src值就是我们希望的要请求的目的接口,如许,服务器端返回的数据不就绕过这个跨域限制,将数据拿回来了。
  是的,不过千万要留意,<script>要求你的返回内容必须是一段可以执行的js,因此你的返回数据就必须是一个可以执行的js语句,而不能是随便一个字符串。并且还要包管在执行js后我们要知道数据回来了。那么综合这些思量,我们想到了一个解决方案:
  我们先定义一个方法:
  // 留意这是前端代码
  var datasuccess = function (data) {
   // TODO
  }
  1. // 注意这是前端代码
  2. var datasuccess = function (data) {
  3.     // TODO
  4. }
复制代码
现在有了这个方法,我们将服务器返回的数据改成这种格式:
  // 留意这是后端代码
  response.getWrite().print("datasuccess({name: \"Jack\", age: 23});");
  1. // 注意这是后端代码
  2. response.getWrite().print("datasuccess({name: "Jack", age: 23});");
复制代码
后端通过返回一段js,而这段js实际上就是在执行之前定义好了的datasuccess方法,并且在执行的时候,还把一些数据传入了进来。嘶~~,这是什么啊,这不就正好我们可以在datasuccess方法内里拿到返回的data数据吗,而且还是在正确的时机进行执行。如许,数据就光明正大的被我们拿到了啊!
  它之以是叫JSONP,大概就是因为险些所有后端在写返回数据的时候都是将数据参数传入的一个json对象。其实你可以乃至可以定义多个参数,每个参数的意义用途你也可以自己设定。
  现在来看看一个完整的jsonp方法来进行跨域解决的代码:
  // 先定义要执行的方法。
  var datasuccess =function(data) {
   console.log("数据已获取:", data);
  }
  // 然后构建一个script节点,
  var scriptDom = document.createElement("script");
  scriptDom.src = "http://example2.com/?k=jack";
  // 将节点添加到body,浏览器就会立刻开始请求。当请求顺利,就会执行 datasuccess 方法
  // 在该方法里执行获取到请求数据的逻辑。
  1. // 先定义要执行的方法。
  2. var datasuccess =function(data) {
  3.     console.log("数据已获取:", data);
  4. }
  5. // 然后构建一个script节点,
  6. var scriptDom = document.createElement("script");
  7. scriptDom.src = "http://example2.com/?k=jack";
  8. // 将节点添加到body,浏览器就会立即开始请求。当请求顺利,就会执行 datasuccess 方法
  9. // 在该方法里执行获取到请求数据的逻辑。
  10. ————————————————
  11. 版权声明:本文为CSDN博主「Microanswer」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
  12. 原文链接:https://blog.csdn.net/MicroAnswer/article/details/102913571
复制代码
而通常我们的接搜数据的方法名称并不是一直稳定的,而是每次一个新的,在script节点中还会把方法名称传上去,让服务端知道我们获取数据的方法名,从而顺利的完成调用。
  只管这个方法很好,但是它只能走GET的请求方法,因为每次script节点的请求只有GET请求嘛。以是我们使用JSONP的接口,就只有GET方式。
  三、VUE提供的代理设置
  如果你是VUE项目,那么你在开发时,通常会设置一个代理,来完成跨域问题的修复,似乎没有后端的事情,但当你正式上线你才知道,代理没效果了。是的。现在介绍一下这个代理干了一件什么事情。
  当你在开发VUE项目时,就必然会开一个server去及时预览你的代码效果,这是毋庸置疑的。但你要留意,开了一个server,这个server能做到事情,可不就是单单给你提供预览这么简单。它还可以进行请求转发,实际上你设置的那些代理,是先会请求到你的server,你开的server查抄到你对应的设置,再请求你配的目的地点。这之间发生了什么,实际上就是把你的实际请求转到了你开的server内里去请求了,这就不存在什么浏览器同源安全的支配了,固然也就没有了跨域问题。
  而当你上线项目时,如果你的代理设置得不敷优雅,或者不敷标准,你要警惕了,非常有大概你的请求就都会失败。最佳的跨域解决方案,无非就是后端帮忙一起解决,片面可不能达到完善。
  四、总结
  解决方式还有更多各种各样的,但我认为最优雅的莫过于这两种,因此其他的解决方式可以暂时不提及,那些方法不但增加了阅读复杂度还增加了维护成本不建议使用。无论是接纳何种方式,我们都是要让后端修改代码,修改header或修改返回数据格式,都离不开后端的参与,以是碰到跨域问题,赶快找后端,一起解决这个问题。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

守听

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表