优雅的处理挂载window上的函数可能不存在的情况
背景
在做一个Web JS SDK(A)时,内部会用到另一个Web JS SDK(B)的方法。(文中后续用A/B代替两者)
B通常会提供Script和NPM包两种使用方式
使用npm pkg的缺点
- 增加包体积
- 如果这个SDK被Web应用已经引入过页面,那么理论上可直接使用,不必要再整一个
如果SDK B包含script引入的方式,目标页面也存在可能会引入B的情况,那么优先考虑使用Script引入依赖的SDK的情况:例如
- 目标页面已经引入过JQuery(符合SDK A的使用需求),那么SDK A就可以直接使用已经存在的
$
进行操作即可,不必再创建jQuery的script - 通常页面都会接入埋点监控等基建服务SDK B,SDK A也需要通过B进行数据的上报
衍生需求
- 挂载在window上的函数不存在时,自动通过script或者polyfill(垫片方法)补全这个方法
- 调用方依旧按照SDK B的文档进行使用
window.sdkB(options)
解决方案
编写一个通用的工具函数,处理上述的衍生需求
方法定义如下
function patchWindowFun(
key: string,
value: string | Function,
options?: {
afterScriptLoad?: Function
beforeAppendScript?: Function
alreadyExistCB?: Function
async?: boolean
defer?: boolean
},
)
总共支持传入3个参数:
key
:带判断的方法在window上的属性名value
:不存在时的取值(function 表明直接使用此方法代替,string类型表明方法来源外部加载的js资源)options
:是一些可选的配置项,主要用于处理使用过外部js资源加载方法的场景afterScriptLoad
:资源加载完成后的回掉beforeAppendScript
:资源加载前的回掉alreadyExistCB
:方法如果已经存在执行的回掉async
:控制script的async属性defer
:控制script的defer属性
由于大多数web sdk都会存在需要调用特定函数或者方法进行初始化的情况,固提供了afterScriptLoad
,beforeAppendScript
,alreadyExistCB
三个钩子函数处理不同时机初始化的情况
方法实现
如果目标属性存在则直接执行相应的回掉,不做进一步处理
if (window[key]) {
alreadyExistCB && alreadyExistCB()
console.log(key, 'already exist')
return
}
目标属性不存在,传入的方法存在时直接进行赋值
// 函数直接赋值
if (typeof value === 'function') {
window[key] = value
return
}
剩余逻辑则是处理方法从外部js资源加载的情况
由于加载script大部分情况是异步的,业务代码中可能已经调用了相关方法,为此临时创建一个方法收集传入的参数
let params = []
window[key] = function () {
params.push(arguments)
}
下面的逻辑就是处理script
加载的逻辑
在js资源加载完成后通过apply
配合forEach
将提前调用方法产生的参数重新正确的执行一次
const script = document.createElement('script')
script.src = value
script.async = !!defer
script.defer = !!async
script.onload = function () {
afterScriptLoad && afterScriptLoad()
// 处理原来没处理的
params.forEach(param => {
window[key].apply(this, param)
})
}
beforeAppendScript && beforeAppendScript()
document.body.append(script)
完整源码如下
function patchWindowFun(
key: string,
value: string | Function,
options?: {
afterScriptLoad?: Function
beforeAppendScript?: Function
alreadyExistCB?: Function
async?: boolean
defer?: boolean
},
) {
// 存在不处理
const { alreadyExistCB, afterScriptLoad, beforeAppendScript, defer, async } = options || {}
if (window[key]) {
alreadyExistCB && alreadyExistCB()
console.log(key, 'already exist')
return
}
// 函数直接赋值
if (typeof value === 'function') {
window[key] = value
return
}
// script url
if (typeof value === 'string') {
let params = []
window[key] = function () {
params.push(arguments)
}
const script = document.createElement('script')
script.src = value
script.async = !!defer
script.defer = !!async
script.onload = function () {
afterScriptLoad && afterScriptLoad()
// 处理原来没处理的
params.forEach(param => {
window[key].apply(this, param)
})
}
beforeAppendScript && beforeAppendScript()
document.body.append(script)
}
}
小结
目前的方法实现仅适用于,调用的方法相对独立不影响正常的交互
如果业务代码依赖方法的返回值,那么异步通过script
加载的方法方式将不太适用
优雅的处理挂载window上的函数可能不存在的情况的更多相关文章
- 油猴Tampermonkey 全局函数 它的注入函数都在 onload里面,直接写函数 都是内部函数,外部要是调用,就要挂靠到window上
油猴Tampermonkey 全局函数 它的注入函数都在 onload里面,直接写函数 都是内部函数,外部要是调用,就要挂靠到window上 window.like111 = function (){ ...
- window上利用pip安装pandas
官网推荐的是直接使用Anoconda,它集成了pandas,可以直接使用.安装挺简单的,有windows下的安装包.如果不想安装庞大的Anoconda,那就一步一步用pip来安装pandas.下面我主 ...
- vuejs件同一个挂载点上切换组
vuejs件同一个挂载点上切换组 动态组件 http://cn.vuejs.org/guide/components.html#动态组件 多个组件可以使用同一个挂载点,然后动态地在它们之间切换.使用保 ...
- 在window上安装pandas
之前在ubuntu上安装pandas,用的easy_install.这次在window上同样方法装遇到"unable to find vcvarsall.bat",看一些网上帖子好 ...
- linux和window下mkdir函数问题(转-锦曦月)
通过WIN32宏进行判断 window下mkdir函数 #include<direct.h> int _mkdir( const char *dirname ); linux下 ...
- iOS UIButton加在window上点击无效果问题
UIButton加在window上,点击没有效果,找了很久,原来是没有加上这名:[self.window makeKeyAndVisible]; self.window = [[UIWindow al ...
- linux和window下mkdir函数
通过WIN32宏进行判断 window下mkdir函数 #include<direct.h> int _mkdir( const char *dirname ); linux下 ...
- 【原】window上安装elasticserach
[window上安装elasticserach] 系统环境:2008R2 x64测试安装用的服务器IP:192.168.12.52elasticsearch版本:2.3.4JDK版本:jdk 1.8. ...
- 关于Windows高DPI的一些简单总结(Window上一般默认是96 dpi 作为100% 的缩放比率)
我们知道,关于高DPI的支持, Windows XP时代就开始有了, 那时关于高DPI的支持比较简单, 但是从Vista/Win7 到现在Win8 /Win8.1, Windows关于高DPI的支持已 ...
- 如果是在有master上开启了该参数,记得在slave端也要开启这个参数(salve需要stop后再重新start),否则在master上创建函数会导致replaction中断。
如果是在有master上开启了该参数,记得在slave端也要开启这个参数(salve需要stop后再重新start),否则在master上创建函数会导致replaction中断.
随机推荐
- Power BI 9 DAY
图形决策树
- 解决highlightjs中纯文本被解析成HTML无法展示的问题,记一次工作中bug修复的思考
壹 ❀ 引 在本周迭代bug修复工作中,遇到了两个比较头疼的bug(同一个客户所提),bug问题描述也很奇怪,客户表示产品的富文本编辑器里的代码块功能,在纯文本语言模式下贴特定代码进去有的看不见,有的 ...
- NC23413 小A买彩票
题目链接 题目 题目描述 小A最近开始沉迷买彩票,并且希望能够通过买彩票发家致富.已知购买一张彩票需要3元,而彩票中奖的金额分别为1,2,3,4元,并且比较独特的是这个彩票中奖的各种金额都是等可能的. ...
- 【OpenGL ES】透视变换原理
1 前言 MVP矩阵变换 中主要介绍了模型变换(平移.旋转.对称.缩放)和观测变换基本原理,本文将介绍透视变换的基本原理. 如下图,近平面和远平面间棱台称为视锥体,表示可见区域范围,视锥体以外 ...
- 如何查看当前Ubuntu系统的版本-【转 https://www.cnblogs.com/chenxiaomeng/p/10038492.html】
如何查看当前Ubuntu系统的版本 一.系统位数 在控制台输入:sudo uname --m 如果显示i686,则表示安装了32位操作系统 如果显示 x86_64,则表示安装了64位操作系统 sudo ...
- linux 快速安装LAMP教程
最近学习linux,安装lamp遇到一些问题,记录下来分享一下: ------------------------------------------------------------------- ...
- 开源大语言模型作为 LangChain 智能体
概要 开源大型语言模型 (LLMs) 现已达到一种性能水平,使它们适合作为推动智能体工作流的推理引擎: Mixtral 甚至在我们的基准测试中 超过了 GPT-3.5,并且通过微调,其性能可以轻易的得 ...
- SQL Server使用常见问题
普通分页查询 三种方式: Top Not IN 方式:查询靠前的数据较快 ROW_NUMBER() OVER()方式:查询靠后的数据速度比上一种较快,在老版本的SQL Server中最常使用 offs ...
- djang中orm使用iterator()
当查询结果有很多对象时,QuerySet的缓存行为会导致使用大量内存.如果你需要对查询结果进行好几次循环,这种缓存是有意义的,但是对于 queryset 只循环一次的情况,缓存就没什么意义了.在这种情 ...
- 如何在 WindowManager.addView 中使用 Jetpack Compose
如何在 WindowManager.addView 中使用 Jetpack Compose 一.引出问题 Android 开发中,很常见的一个场景,通过 WindowManager.addView() ...