关于Jsonp,你知道多少?

article/2025/9/30 19:53:52
# 一、Jsonp为跨域而生

为什么会出现跨域问题?

🙋‍♀️出于浏览器的同源策略限制。

什么是?

🙋‍♀️(origin)就是协议(protocol),主机(host)和端口号(port)。

同源策略是浏览器的行为,是浏览器最核心也最基本的安全功能。浏览器的同源策略限制从一个源加载的文档或脚本与来自另一个源的资源进行交互。浏览器执行javascript脚本时,会检查这个脚本属于哪个页面,如果不是同源页面,就不会被执行。它拦截的是前端请求回来的数据,即请求发送了,服务器响应了,但是因为同源策略无法被浏览器接收。

同源策略需要同时满足以下三点要求

1)协议相同
2)域名相同
3)端口相同

当一个请求url的协议、域名、端口三者之间任意一个与当前页面url不同时,需要跨域才能成功访问。

二、Jsonp与Json有个P关系

Jsonp 全称是Json with Padding,但你可千万别以为它和Json有关系。

Json是一种数据交换格式,而Jsonp是一种依靠开发人员的聪明才智创造出的一种非官方跨域数据交互协议。打个形象的比方:Json是地下党们用来书写和交换情报的“暗号”,而Jsonp则是把用暗号书写的情报传递给自己同志时使用的接头方式。看到没?一个是描述信息的格式,一个是信息传输的协议。

由于浏览器的同源策略搞事情,虽然后端确确实实已经给前端返回Json数据,但浏览器一旦发现不是同源页面的资源,于是它就会将传过来的响应报文丢掉🙄。

但是机智的人类发现在页面上直接发起一个跨域的ajax请求是不可以的,但是,在页面上引入不同域上的js脚本却是可以自由访问的。例如可以在自己的页面上使用<img src=""> 标签来随意显示其他域上的图片:

于是,人类想到了利用这个“漏洞”就可以很好的解决跨域请求:🤔后端不返回Json格式的数据,而是返回一段调用某个函数的js代码,供前端来调用和处理,实现跨域。

那怎么在后端设法把数据装进js格式的文件里呢?

众所周知,后端一般给前端返回json数据,而使用Jsonp的特殊之处就在于前端会传递一个callback参数(key)给后端,接着后端返回数据时会将这个callback参数的值(value)作为函数名来包裹住json数据,最终返给前端的就是一段js代码了,这样就巧妙地解决了跨域的问题✨。

三、手撕Jsonp 实现跨域请求实例

基于vue引入Jsonp实现开源接口的跨域

npm 安装 vue-Jsonp

npm install vue-Jsonp --save

src/main.js 中添加

import {VueJsonp} from 'vue-Jsonp'
Vue.use(VueJsonp)

vue组件中使用Jsonp:

1.直接使用 Jsonp

1️⃣在src/main.js引入vue-resource

vue-resource是一个通过XMLHttpRequrestJsonp技术实现异步加载后端数据的Vue插件

它提供了一般的HTTP请求接口,并且提供了全局方法和vue组件实例方法。

import  VueResource  from 'vue-resource'Vue.use(VueResource)

2️⃣在vue文件中就可以直接用Jsonp

this.$http.jsonp("https://sp0.baidu.com/5a1Fazu8AA54nxGko9WTAnF6hhy/su",{params: {//请求参数wd:'张杰'},jsonp:'cb', //回调参数key//jsonpCallback:"test"	//回调函数value}).then(function(res){console.log("this.$http.jsonp-res:",res)   },function(){console.log("抱歉无法调取")})

2.引入jQuery,搭配 ajax 使用Jsonp

1️⃣在vue文件中直接引入jQuery

import $ from 'jquery'

2️⃣在ajax中引入jsonp

$.ajax({url:"http://suggestion.baidu.com/su?",type:"get",dataType:"jsonp",data:{wd:"王菲",},jsonp:'cb', //回调函数参数jsonpCallback:"test" //回调函数值}).then(res=>{console.log("ajax-res:",res);})

🎈一些解释

怎么告诉后端我要用哪个方法所获得的数据呢❓

前端与后端商量好,前端通过定义jsonp的值,从而与后端的某个函数关联起来。

听我细细说来~

使用Jsonp进行跨域时,请求的url地址后面会自动带上一个callback=xxx传给后端,后端需要对返回给前端的json数据做处理,callback是回调函数参数,xxx即是回调函数名称,回调函数名传到后端后,会被拿来包裹住要返回给前端的json数据,最终返回给前端的数据是xxx(json)的形式。而前端将会用script的方式处理返回数据,来对json数据做处理,以此完成一个有效的Jsonp请求。

此处的callback可通过jsonp来自定义,xxx可通过jsonpCallback来自定义。

jsonp: 回调函数的参数,是与后端约定好的参数,必须与后端保持一致。不另外定义jsonp的话,一般默认为jsonp:'callback'

jsonpCallback: 回调函数名,用来包裹住json数据,不另外定义的话,这个参数的值往往是随机生成的。

默认随机生成的jsonpCallback

下面我画了一张图表示了Jsonp请求的全过程👇

四、使用Jsonp的一些其他话

  • 使用Jsonp跨域,是需要后端配合的,设置callback,需要后端给前端传的是Jsonp格式(也就是一段js脚本)的数据,才能完成跨域请求。

    关于协商

    jsonpCallback 就是你告诉后端最终应该生成哪个方法来包裹住返回给前端的json数据,jsonp 是用来处理后端字段名不是默认值 callback 的情况,可以自定传递给后端的字段名。

    举栗说明:

    https://xxx.com/ip?<jsonp>=<jsonpCallback>
    

    假设💦前端发出的请求是: https://xxx.com/ip?callback=cb,一般参数名即 callback 这个字段名是后端固定、前端默认的,所以你和后端要协商的就是 callback 的值,也就是这里的 cb ,但是这完全取决于后端。

    因为有可能在一些后端没有实现 callback 这个参数,他只能返回固定的函数名比如 runCode,即返回如下

    runCode(1234)
    

    那就没办法协商,你只能按照他的来,你就要在前端处理 runCode 这个函数。

    如果后端实现了允许自定义返回处理的函数,但是它固定的参数名就叫 callback ,这时候你就可以传递一个 jsonp: 'callback'jsonpCallback: 'runCode',注意 jsonp 默认值就是 callback,所以也可以省略。

    如果后端实现了允许你自定义 jsonpCallback ,但是它接受一个等同于 callback 的参数,但是他不叫 callback,而是叫 cb ,那这时候你就可以传递 jsonp: 'cb',同时传递 jsonpCallback: runCode,最终构成了

    后面两种情况,你都可以不传递 jsonpCallback 参数,而是由 jQuery 自动生成,你最终只需要在 success 里面就可以处理。

  • Jsonp只支持get请求,就算前端指定成post方式,会自动转为get方式。而后端如果设置成post方式了,那Jsonp请求跨域就失效了。

    Jsonp 是一种【请求一段 js 脚本,把执行这段脚本的结果当做数据】的玩法。

    1.拼接一个script标签,在请求的url中传入一个要执行的函数的方法名,从而触发对指定地址的get请求
    2.后端对这个get请求进行处理,并返回字符串 “myCallback(‘response value’)”
    3.前端script加载完之后,其实就是在script中执行myCallback(‘response value’)
    4.是不是就完成了跨域的请求
    5.是不是就是只能用get

    所以Jsonp不会对后端代码或者内容做更改,因为它只能发送get请求

  • Jsonp异步

    前端发送请求给后端,在等待后端响应返回数据时:

    同步——前端不能动,要一直等着后端的反馈。
    异步——前端可以自由活动,可以先去执行别的方法。

    Jsonp动态创建一个script标签拼接一个回调函数,去后端找指定的函数返回数据,函数的执行是异步的。

  • 使用Jsonp跨域的优缺点

    优点:

    Jsonp不像XMLHttpRequest对象实现的Ajax请求那样受到同源策略的限制,Jsonp可以跨越同源策略;

    Jsonp的兼容性更好,在更加古老的浏览器中都可以运行,不需要XMLHttpRequestActiveX的支持

    ③ 在请求完毕后可以通过调用callback的方式回传结果。将回调方法的权限给了调用方。这个就相当于将controller层和view层终于分开了。我提供的Jsonp服务只提供纯服务的数据,至于提供服务以后的页面渲染和后续view操作都由调用者来自己定义就好了。如果有两个页面需要渲染同一份数据,你们只需要有不同的渲染逻辑就可以了,逻辑都可以使用同一个Jsonp服务。

    缺点:

    Jsonp只支持GET请求,而不支持POST等其它类型的HTTP请求
    Jsonp只支持跨域HTTP请求这种情况,不能解决不同域的两个页面之间如何进行JavaScript调用的问题。
    Jsonp在调用失败的时候不会返回各种HTTP状态码。
    Jsonp安全性不够。假如提供Jsonp的服务存在页面注入漏洞,即它返回的javascript的内容被人控制的。那么结果是什么?所有调用这个Jsonp的网站都会存在漏洞。于是无法把危险控制在一个域名下,所以在使用Jsonp的时候必须要保证使用的Jsonp服务必须是安全可信的。

如有不对的地方,欢迎指教~
欢迎关注我的公主号:昕之一方
在这里插入图片描述

参考资料:

Jsonp跨域请求原理及优缺点-杜小白Zero

Jquery Ajax客户端跨域请求 以及服务端Python代码实现-彭俏君


http://chatgpt.dhexx.cn/article/o74KPoTa.shtml

相关文章

JSONP原理及实现

文章目录 基本原理执行过程优缺点 案例分析前期准备后端代码前端简单实现jQuery中ajax实现封装一个JSONP方法简易版同时多个请求最终版JSONP方法 基本原理 基本原理&#xff1a; 主要就是利用了 script 标签的src没有跨域限制来完成的。 执行过程 前端定义一个解析函数(如: …

什么是 JSONP?

前言 首先我们得先了解JSONP是怎么产生的。 最开始跨域请求数据没有现在方便&#xff0c;Ajax直接请求普通文件存在跨域无权限访问的问题&#xff0c;然后聪明的程序员想出了一套非官方的解决办法&#xff0c;程序员发现凡是带有“src”这个属性的标签都拥有跨域的能力&#x…

jsonp介绍

为什么要知道jsonp&#xff0c;jsonp的作用是什么&#xff1a; Jsonp(JSON with Padding) 是 json 的一种"使用模式"&#xff0c;可以让网页从别的域名&#xff08;网站&#xff09;获取资料&#xff0c;即跨域读取数据。原则上浏览器是不允许请求不同域名的数据的&…

前端跨域jsonp的细节,挡住面试官的连环提问

1.前言 在前端面试中&#xff0c;想必每一个人都会被问到跨域相关的问题&#xff0c;背过八股文的小伙伴肯定对跨域的解决对答如流&#xff0c;常见的跨域解决方案在网上有很多整理&#xff0c;但是如果问到实现的细节&#xff0c;你是否能够手写实现或者深入解读呢&#xff1f…

jsonp 的原理及应用

1. 什么是jsonp jsonp全称json with padding&#xff0c;填充式的json,jsonp是为跨域而生的 2. 那么有哪些标签可以跨域呢 <img src""> //图片 <link href""> //css <script src""> //程序我们可以使用script来帮助我们跨域…

JSONP详解

Jsonp(JSON with Padding) 是 json 的一种”使用模式”&#xff0c;可以让网页从别的域名&#xff08;网站&#xff09;那获取资料&#xff0c;即跨域读取数据。 为什么我们从不同的域&#xff08;网站&#xff09;访问数据需要一个特殊的技术(JSONP )呢&#xff1f;这是因为同…

使用JSONP解决跨域

1.首先需要知道什么是跨域 浏览器从一个域名的网页去请求另一个域名的资源时&#xff0c;域名、端口、协议任一不同&#xff0c;都是跨域 出于浏览器的同源策略限制 同源策略&#xff08;Sameoriginpolicy&#xff09;是一种约定&#xff0c;它是浏览器最核心也最基本的安全功…

虚拟机安装ubuntu全教程

主要流程 - 准备安装包&#xff08;包括ubuntu镜像、虚拟机压缩包、分区助手)安装虚拟机安装ubuntu、安装vmtool&#xff08;解决ubuntu全屏的问题&#xff09; 一、准备安装包 下载地址&#xff1a;http://pan.baidu.com/s/1hr39WGG 密码&#xff1a;tttq 二、安装虚拟机 …

ubuntu服务器ubuntu Server安装教程

记录一次系统安装到拷贝大数据文件的过程。 说在前: 1.系统U盘启动安装软件Rufus&#xff0c;自行百度下载2.下载Ubuntu Server镜像&#xff0c;官方地址即可一、安装 1、选择Ubuntu Server 2、语言选择&#xff0c;默认英语 3、有网络的话选择第一项升级系统&#xff0c;…

ubuntu安装图文教程

作为目前世界上最安全的操作系统&#xff0c;Linux逐渐被大多数人使用&#xff0c;而ubuntu作为Linux分支中最华丽美观的操作系统&#xff0c;有必要有一个好多安装教程 ubuntu系统是一个linux操作系统;ubuntu安装教程的每个版本类似&#xff0c;下面给您带来的是12.04版本的ub…

Ubuntu/Windows 双系统安装教程

前言 由于工作所用的开发环境是linux的&#xff0c;所以决定把自己电脑装一个windows/ubuntu双系统。Ubuntu不同版本的物理机安装流程都是一样的&#xff0c;而且极其简单&#xff0c;不要怕自己没装过把电脑整坏了&#xff0c;大不了连windows一起给它重装了。 双系统安装步…

安装Ubuntu系统详细教程

一. 前言 本篇文章详解介绍一下如何安装Ubuntu系统&#xff0c;笔者在安装的过程中踩过很多坑&#xff0c;重装了很多次&#xff0c;现在把安装过程中遇到的问题也列出来&#xff0c;供大家参考。 二. 准备工作 这个环节很重要&#xff0c;工欲善其事&#xff0c;必先利其器。 …

VMware虚拟机安装Ubuntu(超详细图文教程)

VMware虚拟机安装Ubuntu 1 Ubuntu下载2 打开VMware3 然后就可启动虚拟机4 等待吧5 重启后就完了&#xff0c; 到这就基本结束了6 下面可以调一下软件下载源 1 Ubuntu下载 Ubuntu下载地址&#xff1a;点这里 注&#xff1a;但官网下载比较慢 也可关注公众号Time木回复&#xf…

Ubuntu系统安装教程

1、首先打开VMware&#xff0c;然后点击创建新虚拟机 2、点击完新建虚拟机后&#xff0c;选择稍后安装操作系统&#xff0c;点击下一步 3、选择Linux&#xff0c;版本选择Ubuntu64位&#xff0c;点击下一步 4、虚拟机名称自己取&#xff0c;位置选择一个盘&#xff0c;创建一…

使用VMware安装Ubuntu虚拟机 - 完整教程

【前言】 本教程将演示通过 VMware 安装 Ubuntu &#xff0c;请提前下载好以下文件哦&#xff1a; ① VMware 软件 ② Ubuntu 的 光盘镜像文件&#xff08;.iso&#xff09; 【下载地址】 VMware 官网链接 https://www.vmware.com/ 本教程使用版本&#xff1a;VMware Worksta…

Ubuntu详细安装教程(小白友好型)

鼠标右键 ------>> 以管理员身份运行   2.右键“文件”——>>“新建虚拟机”   选择“自定义”&#xff0c;“下一步”         “下一步”         “稍后安装操作系统”——>>“下一步”         “Linux”——>>“Ubuntu64位”…

ubuntu 21.04安装教程

ubuntu 21.04安装教程 制作启动U盘*&#xff08;虚拟机安装此步省略&#xff09;*U盘刻录工具balenaEtcher 开始安装选择语言选择键盘布局网络设置代理设置源设置源设置为国内源&#xff0c;这里设置为阿里源&#xff1a; 分区设置用户名工具安装安装完成![在这里插入图片描述]…

Ubuntu安装教程【超多图】

大家好&#xff0c;我是坚果&#xff0c;我的公众号“坚果前端”&#xff0c; 文章目录 01前言02虚拟机的安装03Ubuntu镜像的下载04虚拟机硬件配置1.虚拟机安装完毕之后&#xff0c;界面如下图所示&#xff1a;2.在弹出的对话框中选择自定义&#xff0c;然后点击下一步&#x…

Ubuntu20.04安装详细图文教程(双系统)

Ubuntu安装 前言 最近想把自己开发环境换成linux的&#xff0c;查了一下还是ubuntu桌面比较美观并且作为生产系统生态良好&#xff0c;决定使用ubuntu。开始了着手查找安装Ubuntu双系统的方法。安装有三种&#xff1a; 虚拟机安装wubi安装U盘安装 第一种发挥不出硬件本身的…

新手安装 Ubuntu 操作系统步骤教程

新手安装 Ubuntu 操作系统 最近学习linux编程&#xff0c;需要安装一个 Ubuntu 操作系统&#xff0c;由于虚拟机的体验不是很好&#xff0c;所以便在电脑上试下装双系统。嘿嘿。话不多说&#xff0c;下面直接进入正题&#xff01; 1、下载 Ubuntu 操作系统 我们可以去官网下载一…