通过jsonp解决跨域的源码实现及其特点


prtyaa
prtyaa 2023-12-25 16:26:14 65854
分类专栏: 资讯

一、Jsonp的特点

1、Jsonp是解决跨域的方式之一。
2、Jsonp的核心则是动态添加
3、Jsonp只支持get请求。
4、Jsonp在调用失败的时候不会返回各种HTTP状态码。
5、在请求完毕后通过调用callback的方式回传结果,将回调方法的权限给了调用方。所以在调用jsonp接口时,需要与被调用方协商好用于callback的参数名字,参数的值为函数名。例如cb=_jsonp1234。cb为双方约定好的参数名,_jsonp1234指定被调用方所要执行的函数名。所以调用方在调用前要保证已方拥有该函数用于接收值。

二、通过Promise封装Jsonp

/*
* 实现对 Jsonp 的封装
* url:请求地址
* params:传递的参数对象
* jsonp:与服务端协商的用于存放函数名字的参数
* */
export default function jsonp({url = "", params = {}, jsonp = "cb"}) {
    return new Promise((resolve, reject) => {
        // 定义 body 用于接收数据
        let  body = null;
        // 自定义函数的名字
        const cbName = "_jsonp" + Math.random().toString(36).substr(2);
        // 将jsonp放入params
        params[jsonp] =  cbName;
        // 自定义函数,用于接收值
        window[cbName] = function (data) {
            body = data;
        }
        function _handler({type}){
            // 删除script标签
            document.body.removeChild(script);
            // 删除自定义的函数
            delete window[cbName];
            // 加载完毕并得到数据执行resolve
            console.log(1111,body)
            if(type === "load" && !body)  resolve(body);
            // 异常执行 reject
            else if(type === "error") reject("加载失败");
        }
        // 将对象转为urlencoded格式
        const urlencoded = Object.keys(params).map(v => v + "=" + params[v]).join("&");
        // 将地址url与参数进行拼接。
        url += (url.includes("?") ? "&" : "?") + urlencoded;
        // 创建 script 标签
        const script = document.createElement("script");
        script.src = url;
        // 加载完毕
        script.onload = _handler;
        // 加载异常
        script.onerror = _handler;
        // 指定类型
        script.type = 'text/javascript';
        // 脚本可用,异步执行
        script.async = true;
        // 创建好的script放入body.
        document.body.appendChild(script);
    })
}

四、前端调用

通过百度jsonp的接口进行调用测试:

<script type="module">
    import jsonp from "./jsonp.js";
    // https://www.baidu.com/sugrec?prod=pc&wd=web前端&cb=cb
    (async () => {
        try {
            const res = await jsonp({
                url: "https://www.baidu.com/sugrec",
                params: {
                    prod: "pc",
                    wd: "web前端",// 指定关键字
                },
                jsonp: "cb"
            });
            console.log(res);
        } catch (err) {
            console.log(e)
        }
    })();
</script>

网站声明:如果转载,请联系本站管理员。否则一切后果自行承担。

本文链接:https://www.xckfsq.com/news/show.html?id=30625
赞同 0
评论 0 条
prtyaaL0
粉丝 1 发表 2554 + 关注 私信
上周热门
银河麒麟添加网络打印机时,出现“client-error-not-possible”错误提示  1448
银河麒麟打印带有图像的文档时出错  1365
银河麒麟添加打印机时,出现“server-error-internal-error”  1151
统信桌面专业版【如何查询系统安装时间】  1073
统信操作系统各版本介绍  1070
统信桌面专业版【全盘安装UOS系统】介绍  1028
麒麟系统也能完整体验微信啦!  984
统信【启动盘制作工具】使用介绍  627
统信桌面专业版【一个U盘做多个系统启动盘】的方法  575
信刻全自动档案蓝光光盘检测一体机  484
本周热议
我的信创开放社区兼职赚钱历程 40
今天你签到了吗? 27
信创开放社区邀请他人注册的具体步骤如下 15
如何玩转信创开放社区—从小白进阶到专家 15
方德桌面操作系统 14
我有15积分有什么用? 13
用抖音玩法闯信创开放社区——用平台宣传企业产品服务 13
如何让你先人一步获得悬赏问题信息?(创作者必看) 12
2024中国信创产业发展大会暨中国信息科技创新与应用博览会 9
中央国家机关政府采购中心:应当将CPU、操作系统符合安全可靠测评要求纳入采购需求 8

添加我为好友,拉您入交流群!

请使用微信扫一扫!