想象有这样一个场景:A 团队开发了一套组件库,B 和 C 团队都在各自的业务项目中使用了该组件库。现在 A 团队需要对某个组件进行更新(比如修改颜色),按照以往的做法,A 团队需要先发布一个新的版本,然后其他两个团队各自更新业务项目中所依赖的组件库的版本后发布上线。
有没有更快速的方法呢?比如能否做到只更新组件库,其他依赖它的项目可以自动获取到其最新的版本,即实现远程动态组件。答案是有的,webpack 5 新增的 Module Federation 就可以实现这个需求,但是今天我们要讨论的是另外一种方法。
远程动态组件库项目结构如下所示:
.
├── babel.config.js
├── package.json
├── rollup.config.js
└── src
├── Button.js
├── Text.js
我们简单实现了两个组件 Button 和 Text :
import react from 'react'
export default ({children}) => {
return <button style={{color: 'blue'}}>{children}</button>
}
import React from 'react'
export default ({children}) => {
return <span style={{color: 'blue'}}>{children}</span>
}
我们使用 rollup 对其进行打包,之所以用 rollup 是因为其打包出来的代码非常简洁,方便我们查看,rollup 配置为:
import babel from 'rollup-plugin-babel'
import fs from 'fs'
const components = fs.readdirSync('./src')
export default components.map((filename) => {
return {
input: `./src/${filename}`,
output: {
file: `dist/${filename}`,
format: 'cjs',
},
plugins: [babel()],
}
})
打包后的结果如下所示:
.
├── dist
│ └── Button.js
│ └── Text.js
其中 Button.js 如下所示:
'use strict'
var React = require('react')
function _interopDefaultLegacy(e) {
return e && typeof e === 'object' && 'default' in e ? e : {default: e}
}
var React__default = /*#__PURE__*/ _interopDefaultLegacy(React)
var Button = function (_ref) {
var children = _ref.children
return /*#__PURE__*/ React__default['default'].createElement(
'button',
{
style: {
color: 'blue',
},
},
children
)
}
module.exports = Button
然后我们使用 http-server 在 dist 目录下开启一个静态文件服务,则可以通过 http://localhost:8080/Button.js 获取到打包后的代码。
远程组件库介绍完毕,接下来介绍业务项目中如何使用。
我们使用 create-react-app 创建一个 React 应用,并新增一个 DynamicComponent 组件:
const DynamicComponent = ({name, children, ...props}) => {
const Component = useMemo(() => {
return React.lazy(async () => fetchComponent(name))
}, [name])
return (
<Suspense
fallback={
<div style={{alignItems: 'center', justifyContent: 'center', flex: 1}}>
<span style={{fontSize: 50}}>Loading...</span>
</div>
}>
<Component {...props}>{children}</Component>
</Suspense>
)
}
export default React.memo(DynamicComponent)
这里使用到了 React 中的 Suspense 组件和 React.lazy 方法,关于他们的用法这里不做过多解释,整个 DynamicComponent 组件的含义是远程加载目标组件,这个过程该组件会渲染传入 Suspense 参数 fallback 之中的内容,最后会使用加载成功的组件进行替换。接下来看看 fetchComponent 是如何实现的:
const fetchComponent = async (name) => {
const text = await fetch(
`http://127.0.0.1:8080/${name}.js?ts=${Date.now()}`
).then((a) => {
if (!a.ok) {
throw new Error('Network response was not ok')
}
return a.text()
})
const module = getParsedModule(text, name)
return {default: module.exports}
}
该方法会发起网络请求得到组件的代码,并交给 getParsedModule 去解析,最后得到模块返回。我们来看一下 getParsedModule 是怎么实现的:
const packages = {
react: React,
}
const getParsedModule = (code) => {
let module = {
exports: {},
}
const require = (name) => {
return packages[name]
}
Function('require, exports, module', code)(require, module.exports, module)
return module
}
这里使用 Function 来运行传入的代码,因为打包远程组件的时候并没有将 react 库打包进去,所以这里需要实现 require 这个方法。
我们结合之前打包得到的 Button.js 来看这段代码,它其实同下面这个代码是等价的:
const packages = {
react: React,
}
const getParsedModule = (code, moduleName) => {
let module = {
exports: {},
}
const require = (name) => {
return packages[name]
}
;((require, exports, module) => {
'use strict'
var React = require('react')
function _interopDefaultLegacy(e) {
return e && typeof e === 'object' && 'default' in e ? e : {default: e}
}
var React__default = /*#__PURE__*/ _interopDefaultLegacy(React)
var Button = function (_ref) {
var children = _ref.children
return /*#__PURE__*/ React__default['default'].createElement(
'button',
{
style: {
color: 'blue',
},
},
children
)
}
module.exports = Button
})(require, module.exports, module)
return module
}
最后我们可以按照如下方式来使用 DynamicComponent 组件:
import DynamicComponent from './DynamicComponent'
function App() {
return (
<div>
<DynamicComponent name='Button'>Click Me</DynamicComponent>
<DynamicComponent name='Text'>Remote Component</DynamicComponent>
</div>
)
}
export default App
现在我们尝试修改远程动态组件库中的组件,重新打包后就可以马上看到修改后的效果了。
本文介绍了一种实现远程动态组件的方式,不过比较简陋,事实上我们还有很多优化的空间。比如按照现在的实现方式,如果页面上面使用了两个 Button ,会发起两次请求,这显然不合理。针对这个问题,我们可以通过提前加载以及模块缓存的方式来解决。
来自:http://www.paradeto.com/2021/12/10/react-dynamic-component/
RPC(Remote Procedure Call)—远程过程调用,它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议。 运行时,一次客户机对服务器的RPC调用,其内部操作大致有如下十步:
git branch -r #查看远程所有分支;git branch #查看本地所有分支;git branch -a #查看本地及远程的所有分支,git fetch #将某个远程主机的更新,全部取回本地:git branch -a #查看远程分支
初始化;查看当前仓库状态;将项目的文件添加到仓库中;将add的文件commit到仓库;将本地的仓库关联到远程仓库上;代码合并 拉取;代码上传到远程仓库
React 是一个专注的组件库。因此,它对如何请求远程数据没有什么建议。如果要通过 HTTP 请求数据并将其发送到 Web API ,可以考虑下面四种方法。
希望可以保留现在这个远程仓库,然后清空里面的文件和文件夹,实现过程总结为:找个空文件夹把项目clone下来,删除内容,然后push
内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!