micro-app的简单学习

news2025/1/11 23:39:40

本文承接上一篇手把手教你使用vue2搭建micro-app,对micro-app进行简单的认识与学习。

简述

因为上一篇只是对micro-app的搭建,并没有对具体的内容进行深入了解,所以本文是在上一篇文章代码的基础上对micro-app官网,的初步了解。

1、在基座中调用子应用

在了解完微前端之后,我们知道,微前端是由一个基座和多个子应用构成的,那么如何在基座中调用子应用呢?官网中如下介绍:

// 官网中看到是使用此标签在父子间中嵌入子组件的
<!--
   name(必传):应用名称
   url(必传):应用地址,会被自动补全为http://localhost:3000/index.html
   baseroute(可选):基座应用分配给子应用的基础路由,就是上面的 `/my-page`
-->
<micro-app name='app1' url='http://localhost:3000/' baseroute='/my-page'></micro-app>

参照官网例子嵌入子应用,效果如下:
嵌入示例
代码如下:

<!--
   name:我暂时还没有看出来name是不是必须要是什么,就写的子应用的名称,也没有报什么错
   url(必传):配置的子应用路径
-->
<micro-app name='app_first' url='http://localhost:3001/'></micro-app>
<micro-app name='app_second' url='http://localhost:3002/'></micro-app>

上述例子中是单个子应用的,下面也分析了一下搭建的实例中的子应用嵌入,这边是通过点击router-link时,动态嵌入对应的子应用。具体内容我都加了注释说明
注释说明
base/src/App.vue

<template>
   <div id="app">
<!--      导航部分-->
      <div id="nav">
         <router-link to="/">Home</router-link> |
         <router-link to="/about">About</router-link>|
         <router-link :to="`/${prefix}/first-child/home`">FirstChildHome</router-link>|
         <router-link :to="`/${prefix}/first-child/about`">FirstChildAbout</router-link>|
         <router-link :to="`/${prefix}/second-child/home`">SecondChildHome</router-link>|
         <router-link :to="`/${prefix}/second-child/about`">SecondChildAbout</router-link>
<!--         <micro-app name='app_first' url='http://localhost:3001/'></micro-app>-->
<!--         <micro-app name='app_second' url='http://localhost:3002/'></micro-app>-->
      </div>
<!--      视图、组件调用部分-->
      <div style="background-color: pink">
         <micro-app
            v-if="isChild"
            v-bind="micro"
            destory
         ></micro-app>
         <router-view v-else></router-view>
      </div>
   </div>
</template>


<script>
import { MICRO_APPS, CHILD_PREFIX } from './micro/config.js'


export default {
   name: 'App',
   data () {
      return {
         // 是否为子模块
         isChild: false,
         micro: {
            url: '' ,        /**子模块地址 */
            key: '' ,        /**vue 标签的 key 值,用于不同子模块间的切换时,组件重新渲染 */
            name: '',            /**子模块名称,唯一 */
            data: {},        /**子模块数据 */
            baseroute: ''     /**子模块数据 */
         },
         prefix: CHILD_PREFIX   /**子模块链接前缀 */
      }
   },
   watch: {
      $route (val) { /**监听路由变化修改视图显示 */
         this.changeChild(val)
      }
   },
   created () {
      this.changeChild(this.$route)
   },
   methods: {
      /**
       * 获取子模块 url 和 name
       * */
      getAppUrl (name) {
         console.log('获取子模块 url 和 name', MICRO_APPS.find(app => app.name === name) || {})
         return MICRO_APPS.find(app => app.name === name) || {}
      },
      /**
       * 修改子视图显示
       * @param route 在watch中监听到的路由变化
       * */
      changeChild (route) {
         let path = route.path.toLowerCase() // 路由路径,将路径都转成小写字母,如:/app/first-child/home
         let paths = path.split('/')  // 将上述路径已'/'截取,获得如:['', 'app', 'first-child', 'home']
         console.log('path', path)
         console.log('paths', paths)
         // 判断是否为子模块,子模块有固定的前缀,在 micro/config 设置
         this.isChild = paths.length > 2 && paths[1] === CHILD_PREFIX
         console.log('CHILD_PREFIX', CHILD_PREFIX)
         if (this.isChild) { // 子模块
            // 获取子模块的name和url
            let app = this.getAppUrl(paths[2])
            console.log('app', app)
            this.micro = {
               ...app
               , data: { name: route.name }
               , key: `${app.name}`
               , baseroute: `/${CHILD_PREFIX}/${paths[2]}`
            }
            console.log('micro', this.micro)
         }
      }
   }
}
</script>


<style>
#app {
   font-family: Avenir, Helvetica, Arial, sans-serif;
   -webkit-font-smoothing: antialiased;
   -moz-osx-font-smoothing: grayscale;
   text-align: center;
   color: #2c3e50;
}


#nav {
   padding: 30px;
}


#nav a {
   font-weight: bold;
   color: #2c3e50;
}


#nav a.router-link-exact-active {
   color: #42b983;
}
</style>

2、配置项

通过控制项,可以决定开启或关闭某些功能,配置项都是对标签的配置。
如果希望在某个应用中使用,可以如下配置:

<micro-app
  name='xx' // 应用名称,必传,多个应用同时渲染时,name不可重复
  url='xx' // 应用地址
  baseroute='/my-page/' // 子应用的基础路由
  inline='false' // 是否使用内联script,一般开发环境中使用
  destroy='false' // 卸载时是否强制删除缓存资源,若开启再次渲染时会重新请求数据
  disableScopecss='false' // 禁用样式隔离,应用直接样式不会互相污染
  disableSandbox='false' // 禁用js沙箱,可能会导致一些不可预料的问题,慎用
  ssr='false'  // 是否开启ssr模式
  keep-alive='false' // 是否开启keep-alive模式
  shadowDOM='false' // 是否开启shadowDOM,
/>

如果想要做一个全局配置,可以如下配置:

import microApp from '@micro-zoe/micro-app'

microApp.start({
  inline: true, // 默认值false
  destroy: true, // 默认值false
  disableScopecss: true, // 默认值false
  disableSandbox: true, // 默认值false
  shadowDOM: true, // 默认值false
  ssr: true, // 默认值false
  })

上方代码做了简单描述,具体还得看官网地址

3、生命周期

通过CustomEvent定义生命周期,在组件渲染过程中会触发相应的生命周期事件。

3-1、micro-app标签有如下五个生命周期

(1)created:标签初始化后,加载资源前触发
(2)beforemount:加载资源完成后,开始渲染之前触发
(3)muonted:子应用渲染结束后触发
(4)unmount:子应用卸载时触发
(5)error:子应用渲染出错时触发,只有会导致渲染终止的错误才会触发此生命周期

3-2、在单个组件中监听生命周期

首先看一下对声明周期使用的例子和效果,方便对代码的理解,正常情况下组件能使用到的生命周期就4个,在后期开发中可根据合适的场景调用对应的生命周期。
在这里插入图片描述

<micro-app
   v-if="isChild"
   v-bind="micro"
   destory
   @created='created'
   @beforemount='beforemount'
   @mounted='mounted'
   @unmount='unmount'
   @error='error'
></micro-app>
/**
* 子模块创建(标签初始化后,加载资源前触发)
*/
created () {
   console.log(`子模块创建(标签初始化后,加载资源前触发)-created`)
},
/**
* 子模块挂载之前(加载资源完成后,开始渲染之前触发)
*/
beforemount () {
   console.log(`子模块挂载之前(加载资源完成后,开始渲染之前触发)-beforemount`)
},
/**
* 子模块挂载(子应用渲染结束后触发)
*/
mounted () {
   console.log(`子模块挂载(子应用渲染结束后触发)-mounted`)
},
/**
* 子模块卸载(子应用卸载时触发)
*/
unmount () {
   console.log(`子模块卸载(子应用卸载时触发)-unmount`)
},
/**
* 子模块异常(子应用渲染出错时触发,只有会导致渲染终止的错误才会触发此生命周期)
*/
error () {
   console.log(`子模块异常-error`)
}

3-3、全局监听

如果在开发中,有用到一些通用的监听方法,可在全局进行生命周期配置,配置如下:

import microApp from '@micro-zoe/micro-app'

microApp.start({
  lifeCycles: {
    created (e) {
      console.log('created')
    },
    beforemount (e) {
      console.log('beforemount')
    },
    mounted (e) {
      console.log('mounted')
    },
    unmount (e) {
      console.log('unmount')
    },
    error (e) {
      console.log('error')
    }
  }})

4、环境变量

之前在搭建micro-app的时候,有配置一些环境变量,当时就是照着例子抄的,现在要大概了解一下大致的环境变量配置。

4-1、判断应用是否在微前端环境中

是否微前端

// 设置 webpack 的公共路径
if (window.__MICRO_APP_ENVIRONMENT__) { // 判断是否在微前端环境中
    // eslint-disable-next-line
    __webpack_public_path__ = window.__MICRO_APP_PUBLIC_PATH__ || '/'
}

4-2、应用名称

在这里插入图片描述

/**
* 微前端环境下,注册mount和unmount方法
*/
if (window.__MICRO_APP_ENVIRONMENT__)
   window[`micro-app-${window.__MICRO_APP_NAME__}`] = { mount, unmount }
else
   mount()

4-3、子应用的静态资源前缀

在这里插入图片描述

// 设置 webpack 的公共路径
if (window.__MICRO_APP_ENVIRONMENT__) { // 判断是否在微前端环境中
    // eslint-disable-next-line
    __webpack_public_path__ = window.__MICRO_APP_PUBLIC_PATH__ || '/'
}
/**引入 publicPath 设置 */
import './micro'

4-4、子应用的基础路由

在这里插入图片描述

import Vue from 'vue'
import VueRouter from 'vue-router'


Vue.use(VueRouter)


const routes = [
   {
      path: window.__MICRO_APP_BASE_ROUTE__ || '/' /**根据项目运行的不同环境,设置路径的前缀 */
      , name: 'Home'
      , redirect: { name: 'FirstHome' }
      , component: () => import('../views/Empty.vue')
      , children: [
         {
            path: 'home'
            , name: 'FirstHome'
            , component: () => import('../views/Home.vue')
         }
         , {
            path: 'about'
            , name: 'FirstAbout'
            , component: () => import('../views/About.vue')
         }
      ]
   }
]


const router = new VueRouter({
   mode: 'history',
   routes
})


export default router
<template>
   <div id="app">
      <div id="nav">
         <router-link :to="`${prefix}/home`">子应用1Home</router-link> |
         <router-link :to="`${prefix}/about`">子应用1About</router-link> |
         <button @click="goto('SecondHome')">SecondHome</button> |
         <button @click="goto('SecondAbout')">SecondAbout</button>
      </div>
      <router-view />
   </div>
</template>


<script>
export default {
   name: 'App'
   , data () {
      return {
         prefix: window.__MICRO_APP_BASE_ROUTE__ || ''
      }
   }
   , methods: {
      dataListener (data) {
         console.log('data11111111111111', data)
         console.log('this.$route.name111111111111', this.$route.name)
         if (data.name !== this.$route.name) /** 不判断时会报一个“冗余导航【NavigationDuplicated】”的异常 */
            this.$router.push({ name: data.name })
      }
      , goto (name) {
            // 向基项目发送数据
         window.microApp && window.microApp.dispatch({ route: { name } })
      }
   },
   created () {
      /** 绑定数据【data属性】监听事件 */
      window.microApp && window.microApp.addDataListener(this.dataListener)
   }
   , destroyed () {
      /** 移除数据【data属性】监听事件 */
      window.microApp && window.microApp.removeDataListener(this.dataListener)
   }
}
</script>


<style>
#app {
   font-family: Avenir, Helvetica, Arial, sans-serif;
   -webkit-font-smoothing: antialiased;
   -moz-osx-font-smoothing: grayscale;
   text-align: center;
   color: #2c3e50;
}


#nav {
   padding: 30px;
}


#nav a {
   font-weight: bold;
   color: #2c3e50;
}


#nav a.router-link-exact-active {
   color: #42b983;
}
</style>

4-5、判断是否是基座应用

在这里插入图片描述

import microApp from '@micro-zoe/micro-app'
import * as config from './config'


/**启用 micro */
microApp.start({
    preFetchApps: config.MICRO_APPS
    , globalAssets: config.GLOBAL_ASSETS
})


if (window.__MICRO_APP_BASE_APPLICATION__) {
    console.log('我是基座应用11111111111111111111111111111111111111111111111')
}

5、JS沙箱

js沙箱,就是让你的程序跑在一个隔离的环境下,不对外界的其他程序造成影响,通过创建类似沙盒的独立作业环境,在其内部运行的程序并不能对硬盘产生永久性的影响。

micro-app使用Proxy拦截了用户全局操作的行为,防止对windows的访问和修改,避免全局变量污染。micro-app中的每个子应用都运行在沙箱环境,以获取相对纯净的运行空间。

沙箱默认是开启的,正常情况不建议关闭,因为关闭可能会出现一些不可预知的问题

目前有以下3种方式在子应用中获取外部真实window

1、new Function(“return window”)() 或Function(“return window”)()
2、(0, eval)(‘window’)
3、window.rawWindow
在这里插入图片描述
子应用中使用window.a,父应用也有window.a,两者会互相影响,使用如下方法解决:

(function(window, self) {
  with(window) {
    子应用的js代码
  }
}).call(代理对象, 代理对象, 代理对象)

6、样式隔离

micro-app的样式隔离默认是开启的,开启后会以标签作为样式作用域,利用标签的name属性为每个样式添加前缀,将子应用的样式影响禁锢在当前标签区域。 但基座应用的样式依然会对子应用产生影响,如果发生样式污染,推荐通过约定前缀或CSS Modules方式解决。

.test {
  color: red;}


/* 转换为 */micro-app[name=xxx] .test {
  color: red;
}

7、元素隔离

micro-app模拟实现了类似ShadowDom的功能,元素不会逃离元素边界,子应用只能对自身的元素进行增、删、改、查的操作。
如:基座应用和子应用都有一个元素<div id='root'></div>,此时子应用通过document.querySelector(‘#root’)获取到的是自己内部的#root元素,而不是基座应用的。
当然,基座应用是可以获取子应用元素的, 在微前端下基座拥有统筹全局的作用,没有对基座应用操作子应用元素的行为进行限制。
在这里插入图片描述

8、数据通信

micro-app提供了一套灵活的数据通信机制,方便基座应用和子应用之间的数据传输。
正常情况下,基座应用和子应用之间的通信是绑定的,基座应用只能向指定的子应用发送数据,子应用只能向基座发送数据,这种方式可以有效的避免数据污染,防止多个子应用之间相互影响。
同时还提供了全局通信,方便跨应用之间的数据通信。

8-1、子应用获取来自基座应用的数据

8-1-1、直接获取数据

// 官网是这样介绍的
const data = window.microApp.getData() // 返回基座下发的data数据

根据官网介绍,我直接在子应用打印了上述变量结果,但是只获取到子应用名称,效果如下:
在这里插入图片描述
接下来我好奇这getData()获取到到底是啥东西,于是去基座子应用调用添加了data,这样在子应用打印出来的内容就会多一点了,所以感觉如果想要在子应用中获取到基座数据,就需要将数据添加到data中
在这里插入图片描述

8-1-2、绑定监听函数

// 官网是如下介绍的

function dataListener (data) {
  console.log('来自基座应用的数据', data)}


/**
* 绑定监听函数,监听函数只有在数据变化时才会触发
* dataListener: 绑定函数
* autoTrigger: 在初次绑定监听函数时如果有缓存数据,是否需要主动触发一次,默认为false
* !!!重要说明: 因为子应用是异步渲染的,而基座发送数据是同步的,
* 如果在子应用渲染结束前基座应用发送数据,则在绑定监听函数前数据已经发送,在初始化后不会触发绑定函数,
* 但这个数据会放入缓存中,此时可以设置autoTrigger为true主动触发一次监听函数来获取数据。
*/
window.microApp.addDataListener(dataListener: Function, autoTrigger?: boolean)


// 解绑监听函数
window.microApp.removeDataListener(dataListener: Function)


// 清空当前子应用的所有绑定函数(全局数据函数除外)
window.microApp.clearDataListener()

同样,我在项目中试了一下,是可以拿到基座data数据的
在这里插入图片描述

methods: {
   dataListener (data) {
      console.log('来自基座应用的数据', data)
   }
},
created () {
   // const data = window.microApp.getData() // 返回基座下发的data数据
   // console.log('子应用获取父应用数据', data)
   /** 绑定数据【data属性】监听事件 */
   window.microApp && window.microApp.addDataListener(this.dataListener, true)
},
destroyed () {
   /** 移除数据【data属性】监听事件 */
   window.microApp && window.microApp.removeDataListener(this.dataListener)
}

8-2、子应用向基座应用发送数据

// 官网中如下介绍
// dispatch只接受对象作为参数
window.microApp.dispatch({type: '子应用发送的数据'})

由于这边只介绍了子应用向基座应用发送数据,但是没有介绍发送过去的数据在基座怎么接收,所以这个内容写完后,我也不知道到底有没有发送成功,就先去找了一个基座接收子应用数据的方法,来证明子应用有没有向基座发送成功。
在这里插入图片描述
在这里插入图片描述
子应用发送数据:

<button @click="goto('SecondHome')">向基座发送数据Home</button> |
<button @click="goto('SecondAbout')">向基座发送数据About</button>
methods: {
   goto (name) {
           // 向基项目发送数据
      window.microApp && window.microApp.dispatch({ name: name,type: '子应用发送的数据', msg: '成功' })
   }
},

基座接收数据:

<micro-app
   v-if="isChild"
   v-bind="micro"
   :data="micro"
   destory
   @created='created'
   @beforemount='beforemount'
   @mounted='mounted'
   @unmount='unmount'
   @error='error'
   @datachange='handleDataChange'
></micro-app>
methods: {
   handleDataChange (event) {
      console.log('接收子应用向基座发送得数据', event)
   }
}

8-3、基座应用向子应用发送数据

基座向子应用发送数据的方式有两种

8-3-1、通过data属性发送数据

我觉得和8-1中我子应用获取基座应用时,基座应该的数据写法是一致的,就是中要添加:data=‘数据’,具体可看8-1中我的示例代码,官方介绍如下:

<template>
  <micro-app
    name='my-app'
    url='xx'
    :data='dataForChild' // data只接受对象类型,数据变化时会重新发送
  /></template>


<script>export default {
  data () {
    return {
      dataForChild: {type: '发送给子应用的数据'}
    }
  }}
</script>

8-3-2、 手动发送数据

手动发送数据需要通过name指定接受数据的子应用,此值和元素中的name一致。官网介绍如下:

import microApp from '@micro-zoe/micro-app'
// 发送数据给子应用 my-app,setData第二个参数只接受对象类型
microApp.setData('my-app', {type: '新的数据'})

在这里插入图片描述
在这里插入图片描述
基座:

import microApp from "@micro-zoe/micro-app";
/**
* 子模块创建(标签初始化后,加载资源前触发)
*/
created () {
   microApp.setData('first-child', {type: '新的数据'})
},

子应用:

methods: {
   dataListener (data) {
      console.log('来自基座应用的数据', data)
   }
},
created () {
   // const data = window.microApp.getData() // 返回基座下发的data数据
   // console.log('子应用获取父应用数据', data)
   /** 绑定数据【data属性】监听事件 */
   window.microApp && window.microApp.addDataListener(this.dataListener, true)
},
destroyed () {
   /** 移除数据【data属性】监听事件 */
   window.microApp && window.microApp.removeDataListener(this.dataListener)
}

8-4、基座应用获取来自子应用的数据

关于在基座中获取子应用的数据,官方介绍了三种方法,其中第二种方式在8-2中已经尝试过了,这边就只尝试一三两种方法了。

8-4-1、直接获取数据

// 官方介绍
import microApp from '@micro-zoe/micro-app'

const childData = microApp.getData(appName) // 返回子应用的data数据

在这里插入图片描述
基座:

mounted () {
   const childData = microApp.getData('first-child')
   console.log('接收子应用向基座发送得数据', childData)
   // console.log(`子模块挂载(子应用渲染结束后触发)-mounted`)
},

子应用:

created () {
   // 向基项目发送数据
   window.microApp && window.microApp.dispatch({ name: '就将计就计',type: '子应用发送的数据', msg: '成功' })
},

8-4-2、监听自定义事件

监听自定义事件(datachange),8-2中已介绍,在此不多介绍,看8-2即可

8-4-3、绑定监听函数

绑定监听函数需要通过name指定子应用,此值和元素中的name一致。看官方介绍和8-1很像但不完全一样,官方介绍如下:

import microApp from '@micro-zoe/micro-app'


function dataListener (data) {
  console.log('来自子应用my-app的数据', data)}


/**
* 绑定监听函数
* appName: 应用名称
* dataListener: 绑定函数
* autoTrigger: 在初次绑定监听函数时如果有缓存数据,是否需要主动触发一次,默认为false
*/
microApp.addDataListener(appName: string, dataListener: Function, autoTrigger?: boolean)


// 解绑监听my-app子应用的函数
microApp.removeDataListener(appName: string, dataListener: Function)


// 清空所有监听appName子应用的函数
microApp.clearDataListener(appName: string)

在这里插入图片描述
在这里插入图片描述
基座:

dataListener (data) {
   console.log('接收子应用向基座发送得数据--方法三', data)
},
/**
* 子模块创建(标签初始化后,加载资源前触发)
*/
created () {
   /** 绑定数据【data属性】监听事件 */
   microApp.addDataListener('first-child', this.dataListener, true)
},
destroyed () {
   /** 移除数据【data属性】监听事件 */
   window.microApp && window.microApp.removeDataListener('first-child', this.dataListener)
},

子应用:

created () {
   // 向基项目发送数据
   window.microApp && window.microApp.dispatch({ name: '就将计就计',type: '子应用发送的数据', msg: '成功' })
},

8-5、全局数据通信

全局数据通信会向基座应用和所有子应用发送数据,在跨应用通信的场景中适用。全局通信在基座应用和子应用的用法跟上面8-3第三种方法以及8-1很类似,只是属性名变成了全局而已,具体可参考8-3和8-1中的方法。
官网介绍如下:
基座发送全局数据:

import microApp from '@micro-zoe/micro-app'


// setGlobalData只接受对象作为参数
microApp.setGlobalData({type: '全局数据'})

基座获取全局数据:

import microApp from '@micro-zoe/micro-app'


// 直接获取数据const globalData = microApp.getGlobalData() // 返回全局数据


function dataListener (data) {
  console.log('全局数据', data)}


/**
* 绑定监听函数
* dataListener: 绑定函数
* autoTrigger: 在初次绑定监听函数时如果有缓存数据,是否需要主动触发一次,默认为false
*/
microApp.addGlobalDataListener(dataListener: Function, autoTrigger?: boolean)


// 解绑监听函数
microApp.removeGlobalDataListener(dataListener: Function)


// 清空基座应用绑定的所有全局数据监听函数
microApp.clearGlobalDataListener()

子应用发送全局数据:

// setGlobalData只接受对象作为参数
window.microApp.setGlobalData({type: '全局数据'})

子应用获取全局数据:

// 直接获取数据const globalData = window.microApp.getGlobalData() // 返回全局数据


function dataListener (data) {
  console.log('全局数据', data)}


/**
* 绑定监听函数
* dataListener: 绑定函数
* autoTrigger: 在初次绑定监听函数时如果有缓存数据,是否需要主动触发一次,默认为false
*/
window.microApp.addGlobalDataListener(dataListener: Function, autoTrigger?: boolean)


// 解绑监听函数
window.microApp.removeGlobalDataListener(dataListener: Function)


// 清空当前子应用绑定的所有全局数据监听函数
window.microApp.clearGlobalDataListener()

8-6、关闭沙箱后的通信方式

沙箱关闭后,子应用默认的通信功能失效,此时可以通过手动注册通信对象实现一致的功能。
注册方式:在基座应用中为子应用初始化通信对象

import { EventCenterForMicroApp } from '@micro-zoe/micro-app'


// 注意:每个子应用根据appName单独分配一个通信对象
window.eventCenterForAppxx = new EventCenterForMicroApp(appName)

子应用就可以通过注册的eventCenterForAppxx对象进行通信,其api和window.microApp一致,基座通信方式没有任何变化。
子应用通信方式:

// 直接获取数据const data = window.eventCenterForAppxx.getData() // 返回data数据


function dataListener (data) {
  console.log('来自基座应用的数据', data)}


/**
* 绑定监听函数
* dataListener: 绑定函数
* autoTrigger: 在初次绑定监听函数时如果有缓存数据,是否需要主动触发一次,默认为false
*/
window.eventCenterForAppxx.addDataListener(dataListener: Function, autoTrigger?: boolean)


// 解绑监听函数
window.eventCenterForAppxx.removeDataListener(dataListener: Function)


// 清空当前子应用的所有绑定函数(全局数据函数除外)
window.eventCenterForAppxx.clearDataListener()


// 子应用向基座应用发送数据,只接受对象作为参数
window.eventCenterForAppxx.dispatch({type: '子应用发送的数据'})

9、静态资源

9-1、资源路径自动补全

micro-app子应用默认提供资源路径自动补全功能。如: 图片/myapp/test.png,在最终渲染时会补全为http://localhost:8080/myapp/test.png
主要争对于link、script、img、background-imge、@font-face等

9-2、publicPath

如果自动补全失败,可以采用运行时publicPath方案解决。我们之前配置环境变量时已经有配置了,app_first为例
在这里插入图片描述
app_first/src/micro/index.js

// 设置 webpack 的公共路径
if (window.__MICRO_APP_ENVIRONMENT__) { // 判断是否在微前端环境中
    // eslint-disable-next-line
    __webpack_public_path__ = window.__MICRO_APP_PUBLIC_PATH__ || '/'
}

app_first/src/main.js

/**引入 publicPath 设置 */
import './micro'

9-3、资源共享

当多个子应用拥有相同的js或css资源,可以指定这些资源在多个子应用之间共享,在子应用加载时直接从缓存中提取数据,从而提高渲染效率和性能。

9-3-1、 globalAssets

globalAssets用于设置全局共享资源,它和预加载的思路相同,在浏览器空闲时加载资源并放入缓存。
当子应用加载相同地址的js或css资源时,会直接从缓存中提取数据,从而提升渲染速度。

// index.js--官方介绍
import microApp from '@micro-zoe/micro-app'

microApp.start({
  globalAssets: {
    js: ['js地址1', 'js地址2', ...], // js地址
    css: ['css地址1', 'css地址2', ...], // css地址
  }})

我们搭建的项目已经做好了配置,在base基座下配置
在这里插入图片描述

import microApp from '@micro-zoe/micro-app'
import * as config from './config'


/**启用 micro */
microApp.start({
    preFetchApps: config.MICRO_APPS
    , globalAssets: config.GLOBAL_ASSETS
})
/**
* 全局资源
*/
export const GLOBAL_ASSETS = {
    js: [],
    css: []
}

9-3-2、global属性

在link、script设置global属性会将文件提取为公共文件,共享给其它应用。
设置global属性后文件第一次加载会放入公共缓存,其它子应用加载相同的资源时直接从缓存中读取内容,从而提升渲染速度。

<link rel="stylesheet" href="xx.css" exclude>
<script src="xx.js" exclude></script>
<style exclude></style>

10、预加载

预加载是指在应用尚未渲染时提前加载资源并缓存,从而提升首屏渲染速度。
预加载并不是同步执行的,它会在浏览器空闲时间,依照开发者传入的顺序,依次加载每个应用的静态资源,以确保不会影响基座应用的性能。

import microApp from '@micro-zoe/micro-app'


// 方式一
microApp.preFetch([
  { name: 'my-app', url: 'xxx' }])


// 方式二
microApp.preFetch(() => [
  { name: 'my-app', url: 'xxx' }])


// 方式三
microApp.start({
  preFetchApps: [
    { name: 'my-app', url: 'xxx' }
  ],
  // 函数类型
  // preFetchApps: () => [
  //   { name: 'my-app', url: 'xxx' }
  // ],})

在基座代码中进行配置配置:
在这里插入图片描述

import microApp from '@micro-zoe/micro-app'
import * as config from './config'


/**启用 micro */
microApp.start({
    preFetchApps: config.MICRO_APPS
    , globalAssets: config.GLOBAL_ASSETS
})
/**
* 子应用地址
*/
export const MICRO_APPS = [
    { name: 'first-child', url: `http://localhost:3001/` },
    { name: 'second-child', url: `http://localhost:3002/` }
]

11、插件系统

插件系统可以赋予开发者灵活处理静态资源的能力,对有问题的资源文件进行修改
插件系统的主要作用是对js的修改,每一个js文件都会经过插件系统,可以对js进行拦截和处理,通常用于修复js中的错误或向子应用注入一些全局变量。
适用场景:无法控制的js
使用方式:

import microApp from '@micro-zoe/micro-app'

microApp.start({
  plugins: {
    // 全局插件,作用于所有子应用的js文件
    global?: Array<{
      // 可选,强隔离的全局变量(默认情况下子应用无法找到的全局变量会兜底到基座应用中,scopeProperties可以禁止这种情况)
      scopeProperties?: string[],
      // 可选,可以逃逸到外部的全局变量(escapeProperties中的变量会同时赋值到子应用和外部真实的window上)
      escapeProperties?: string[],
      // 可选,传递给loader的配置项
      options?: any,
      // 必填,js处理函数,必须返回code值
      loader?: (code: string, url: string, options: any) => code
    }>
  
    // 子应用插件
    modules?: {
      // appName为应用的名称,这些插件只会作用于指定的应用
      [appName: string]: Array<{
        // 可选,强隔离的全局变量(默认情况下子应用无法找到的全局变量会兜底到基座应用中,scopeProperties可以禁止这种情况)
        scopeProperties?: string[],
        // 可选,可以逃逸到外部的全局变量(escapeProperties中的变量会同时赋值到子应用和外部真实的window上)
        escapeProperties?: string[],
        // 可选,传递给loader的配置项
        options?: any,
        // 必填,js处理函数,必须返回code值
        loader?: (code: string, url: string, options: any) => code
      }>
    }
  }})

案例:

import microApp from '@micro-zoe/micro-app'


microApp.start({
  plugins: {
    global: [
      {
        scopeProperties: ['key', 'key', ...], // 可选
        escapeProperties: ['key', 'key', ...], // 可选
        options: 配置项, // 可选
        loader(code, url, options) { // 必填
          console.log('全局插件')
          return code
        }
      }
    ],
    modules: {
      'appName1': [{
        loader(code, url, options) {
          if (url === 'xxx.js') {
            code = code.replace('var abc =', 'window.abc =')
          }
          return code
        }
      }],
      'appName2': [{
        scopeProperties: ['key', 'key', ...], // 可选
        escapeProperties: ['key', 'key', ...], // 可选
        options: 配置项, // 可选
        loader(code, url, options) { // 必填
          console.log('只适用于appName2的插件')
          return code
        }
      }]
    }
  }})

12、多层嵌套

micro-app支持多层嵌套,即子应用可以嵌入其它子应用,但为了防止标签名冲突,子应用中需要做一些修改。
在子应用中设置tagName:

microApp.start({
  tagName: 'micro-app-xxx', // 标签名称必须以 `micro-app-` 开头
})

在子应用中使用新定义的标签进行渲染,如:

<micro-app-xxx name='xx' url='xx'></micro-app-xxx>

注: 无论嵌套多少层,name都要保证全局唯一。
在这里插入图片描述
安装micro-app插件,安装在app_first目录下,因为我这边想把first嵌套到second下面

npm install @micro-zoe/micro-app --save

在这里插入图片描述
app_first/src/micro/index.js

import microApp from '@micro-zoe/micro-app'
// 设置 webpack 的公共路径
if (window.__MICRO_APP_ENVIRONMENT__) { // 判断是否在微前端环境中
    // eslint-disable-next-line
    __webpack_public_path__ = window.__MICRO_APP_PUBLIC_PATH__ || '/'
}
/**启用 micro */
microApp.start({
    tagName: 'micro-app-first', // 标签名称必须以 `micro-app-` 开头
})

然后在app_second/src/App.vue中使用新定义的标签:

<div style="background: pink">
  <micro-app-first name='first-child' url='http://localhost:3001/'></micro-app-first>
</div>

在这里插入图片描述

13、高级功能

13-1、自定义fetch

通过自定义fetch替换框架自带的fetch,可以修改fetch配置(添加cookie或header信息等等),或拦截HTML、JS、CSS等静态资源。
自定义的fetch必须是一个返回string类型的Promise。

import microApp from '@micro-zoe/micro-app'

microApp.start({
  /**
   * 自定义fetch
   * @param {string} url 静态资源地址
   * @param {object} options fetch请求配置项
   * @param {string|null} appName 应用名称
   * @returns Promise<string>
  */
  fetch (url, options, appName) {
    if (url === 'http://localhost:3001/error.js') {
      // 删除 http://localhost:3001/error.js 的内容
      return Promise.resolve('')
    }
    
    const config = {
      // fetch 默认不带cookie,如果需要添加cookie需要配置credentials
      credentials: 'include', // 请求时带上cookie
    }

    return window.fetch(url, Object.assign(options, config)).then((res) => {
      return res.text()
    })
  }})

注: 如果跨域请求带cookie,那么Access-Control-Allow-Origin不能设置为*

13-2、性能内存优化

micro-app支持两种渲染微前端的模式,默认模式和umd模式。
默认模式:子应用在初次渲染和后续渲染时会顺序执行所有js,以保证多次渲染的一致性。(可以满足大多数项目)
umd模式:子应用暴露出mount、unmount方法,此时只在初次渲染时执行所有js,后续渲染时只会执行这两个方法。(多次渲染时性能会更好)
我的项目是否需要切换为umd模式?
如果子应用渲染和卸载不频繁,那么使用默认模式即可,如果子应用渲染和卸载非常频繁建议使用umd模式。
切换为umd模式:子应用在window上注册mount和unmount方法
在这里插入图片描述
src/app_first/src/main.js

/**引入 publicPath 设置 */
import './micro'


import Vue from 'vue'
import App from './App.vue'
import router from './router'


Vue.config.productionTip = false


// new Vue({
//     router,
//     render: function (h) { return h(App) }
// }).$mount('#app')


let app


/**
* 挂载函数
*/
function mount () {
   app = new Vue({
      el: '#app',
      router,
      render: function (h) { return h(App) }
   })
}


/**
* 卸载函数
*/
function unmount () {
   app.$destroy()
   app.$el.innerHTML = ''
   app = null
}


/**
* 微前端环境下,注册mount和unmount方法
*/
if (window.__MICRO_APP_ENVIRONMENT__)
   window[`micro-app-${window.__MICRO_APP_NAME__}`] = { mount, unmount }
else
   mount()

14、路由

路由类型约束:
1、 基座是hash路由,子应用也必须是hash路由
2、 基座是history路由,子应用可以是hash或history路由
基础路由:
通常基座应用和子应用各有一套路由系统,为了防止冲突,基座需要分配一个路由给子应用,称之为基础路由,子应用可以在这个路由下渲染,但不能超出这个路由的范围,这就是基础路由的作用。
使用方式:
基座应用中通过设置 的baseroute属性下发,子应用通过window.__MICRO_APP_BASE_ROUTE__获取此值并设置基础路由。
注:
1、 如果基座是history路由,子应用是hash路由,不需要设置基础路由baseroute
2、 如果子应用只有一个页面,没有使用react-router,vue-router之类,也不需要设置基础路由baseroute
页面代码配置:基座和子应用都使用的是history路由
在这里插入图片描述
路由配置:
在这里插入图片描述
base/src/router/index.js

import Vue from 'vue'
import VueRouter from 'vue-router'
import { CHILD_PREFIX } from '@/micro/config.js'

Vue.use(VueRouter)

const routes = [
   {
      path: '/'
      , name: 'Home'
      , component: () => import('../views/Home.vue')
   }
   , {
      path: '/about'
      , name: 'About'
      , component: () => import('../views/About.vue')
   }
   , {
      path: `/${CHILD_PREFIX}/first-child`
      , name: 'FirstChild'
      , children: [
         {
            path: 'home'
            , name: 'FirstHome'
         }
         , {
            path: 'about'
            , name: 'FirstAbout'
         }
      ]
   }
   , {
      path: `/${CHILD_PREFIX}/second-child`
      , name: 'SecondChild'
      , children: [
         {
            path: 'home'
            , name: 'SecondHome'
         }
         , {
            path: 'about'
            , name: 'SecondAbout'
         }
      ]
   }
]


const router = new VueRouter({
   mode: 'history'
   , routes
})


export default router

app_first/src/router/index.js

import Vue from 'vue'
import VueRouter from 'vue-router'

Vue.use(VueRouter)

const routes = [
   {
      path: window.__MICRO_APP_BASE_ROUTE__ || '/' /**根据项目运行的不同环境,设置路径的前缀 */
      , name: 'Home'
      , redirect: { name: 'FirstHome' }
      , component: () => import('../views/Empty.vue')
      , children: [
         {
            path: 'home'
            , name: 'FirstHome'
            , component: () => import('../views/Home.vue')
         }
         , {
            path: 'about'
            , name: 'FirstAbout'
            , component: () => import('../views/About.vue')
         }
      ]
   }
]

const router = new VueRouter({
   mode: 'history',
   routes
})

export default router

15、应用之间跳转

问题: 开发者想通过基座应用的侧边栏跳转,从而控制子应用的页面,这其实是做不到的,只有子应用的路由实例可以控制自身的页面。
要解决以上问题,有三种方法:

15-1、window.history

// 官网如下介绍

window.history.pushState(null, '', 'page2')

// 主动触发一次popstate事件
window.dispatchEvent(new PopStateEvent('popstate', { state: null }))

这个我在代码中试了一下,不管是基座跳转子应用,还在子应用跳转基座,还是子应用直接互相跳转都可以,主要是url写对就好了

15-1-1、基座跳转到子应用

在这里插入图片描述

<button @click="jump()">基座跳转到子应用1home</button>
jump() {
   console.log('window.history', window.history)
   window.history.pushState(null, '', 'http://localhost:3000/app/first-child/home')
   // 主动触发一次popstate事件
   window.dispatchEvent(new PopStateEvent('popstate', { state: null }))
},

15-1-2、子应用2跳转到基座

在这里插入图片描述

15-1-3、子应用1跳转到子应用2

在这里插入图片描述

15-2、通过数据通信控制跳转

适用场景:基座控制子应用跳转(类似于之前讲的数据通信,基座app中传递路径,子应用监听数据变化并执行跳转)
base/src/App.vue

import microApp from '@micro-zoe/micro-app'
microApp.setData('子应用name', { path: '/new-path/' })

app_first/src/App.vue

// 监听基座下发的数据变化
window.microApp.addDataListener((data) => {
  // 当基座下发跳转指令时进行跳转
  if (data.path) {
    router.push(data.path)
  }})

15-3、传递路由实例

适用场景:子应用控制基座跳转
base/src/App.vue

<template>
  <micro-app
    name='子应用名称'
    url='url'
    :data='microAppData'
  ></micro-app></template>


<script>
export default {
  data () {
    return {
      microAppData: {
        pushState: (path) => {
          this.$router.push(path)
        }
      }
    }
  },
}
</script>

app_first/src/App.vue

window.microApp.getData().pushState(path)

16、打包

关于打包配置,文档中说的很简单,介绍了如下配置,我在这边也是浪费了很长时间才打出来正确的包

// vue.config.js
module.exports = {
  outputDir: 'my-app',
  publicPath: process.env.NODE_ENV === 'production' ? '/my-app/' : '', // bad ❌
  publicPath: '/my-app/', // good 👍
}

我这边以app_first为例,进行了配置
在这里插入图片描述
在这里插入图片描述
打开js文件看了一下,发现有生成map文件,如下:
在这里插入图片描述
查找发现需要添加配置:

productionSourceMap: false,

在这里插入图片描述
相关配置可参考:
配置完后重新在打包,还是报错
在这里插入图片描述
解决方法
在这里插入图片描述
主要配置代码:(记得去注释router中的mode: ‘history’,)

注:
(1)后面我在部署的时候打了好多包,最后发现其实router中的mode: ‘history’,不需要注释也可以打包成功,但是我就不懂开始看打包的时候,为啥就一直报错呢
(2)部署的时候遇到基座刷新报错500的问题,打包时基座的publicPath要改成绝对路径,即:‘/‘而不是’./’。要去掉点(.)。当然下面代码中我已经改掉了。 publicPath: ‘/’
app_first/vue.config.js

module.exports = {
  productionSourceMap: false,
  outputDir: 'dist',
  publicPath: './',
  devServer: {
      host: 'localhost',
      port: 3001,
      headers: { // 设置本地运行的跨域权限
          'Access-Control-Allow-Origin': '*',
      }
  }
}

app_second/vue.config.js

module.exports = {
  productionSourceMap: false,
  outputDir: 'dist',
  publicPath: './',
  devServer: {
    open: true,
    host: 'localhost',
    port: 3002,
    headers: { // 设置本地运行的跨域权限
          'Access-Control-Allow-Origin': '*'
      }
  }
}

base/vue.config.js

module.exports = {
    productionSourceMap: false,
    outputDir: 'dist',
    publicPath: '/',
    devServer: {
      host: 'localhost',
      port: 3000
    }
}

到此打包过程就结束了,按之前对微前端的了解,每个子应用都是独立打包独立部署。那接下来就要开始部署了。

17、部署

官网这边介绍的是用nginx部署,作为前端我也没玩过这东西,也不知道怎么搞,但是微前端部署需要用到。百度大概了一下,先去官网下载一个nginx
在这里插入图片描述
然后对于配置,还是一脑壳迷雾浆糊,因为来没搞过,一点操作意识都没有,不知道从哪开始下手,找后台大哥咨询了一下才大概懂得怎么配置
下载好的nginx安装包解压后效果图如下,一般需要配置的地方如下图所示:
在这里插入图片描述
下面看看部署过程,我这边打包好的文件没有放在html中,放哪都行,只要配置路径写对即可,部署时也遇到很多问题,下面直接展示最终成功结果。
1、部署文件目录修改,我直接把打包的基座和子应用都并列放在了文件夹下

在这里插入图片描述
在这里插入图片描述
2、配置nginx.conf,主要配置如下:
这边需要注意图中标注的部分,可以看出配置了三个,分别是基座和子应用
在这里插入图片描述

配置代码:

server {
        listen       3000;
        server_name  localhost;


        #charset koi8-r;


        #access_log  logs/host.access.log  main;


        location / {
            root base/dist;
            add_header Access-Control-Allow-Origin *;
            if ( $request_uri ~* ^.+.(js|css|jpg|png|gif|tif|dpg|jpeg|eot|svg|ttf|woff|json|mp4|rmvb|rm|wmv|avi|3gp)$ ){
              add_header Cache-Control max-age=7776000;
              add_header Access-Control-Allow-Origin *;
            }
            try_files $uri $uri/ /index.html;
        }
    }


    server {
        listen       3001;
        server_name  localhost;
        location / {
            root first-child/dist;
            add_header Access-Control-Allow-Origin *;
            if ( $request_uri ~* ^.+.(js|css|jpg|png|gif|tif|dpg|jpeg|eot|svg|ttf|woff|json|mp4|rmvb|rm|wmv|avi|3gp)$ ){
              add_header Cache-Control max-age=7776000;
              add_header Access-Control-Allow-Origin *;
            }
            try_files $uri $uri/ /index.html;
        }
     }


    server {
        listen       3002;
        server_name  localhost;
        location / {
            root second-child/dist;
            add_header Access-Control-Allow-Origin *;
            if ( $request_uri ~* ^.+.(js|css|jpg|png|gif|tif|dpg|jpeg|eot|svg|ttf|woff|json|mp4|rmvb|rm|wmv|avi|3gp)$ ){
              add_header Cache-Control max-age=7776000;
              add_header Access-Control-Allow-Origin *;
            }
            try_files $uri $uri/ /index.html;
        }
     }

整个文件代码:(方便对比,但是nginx版本不一样多多少少肯定会有差距的,所以以下配置文件仅供参考,真正用到的只有上面的配置代码)

#user  nobody;
worker_processes  1;


#error_log  logs/error.log;
#error_log  logs/error.log  notice;
#error_log  logs/error.log  info;


#pid        logs/nginx.pid;




events {
    worker_connections  1024;
}




http {
    include       mime.types;
    default_type  application/octet-stream;


    #log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
    #                  '$status $body_bytes_sent "$http_referer" '
    #                  '"$http_user_agent" "$http_x_forwarded_for"';


    #access_log  logs/access.log  main;


    sendfile        on;
    #tcp_nopush     on;


    #keepalive_timeout  0;
    keepalive_timeout  65;


    #gzip  on;
server {
        listen       3000;
        server_name  localhost;


        #charset koi8-r;


        #access_log  logs/host.access.log  main;


        location / {
            root base/dist;
            add_header Access-Control-Allow-Origin *;
            if ( $request_uri ~* ^.+.(js|css|jpg|png|gif|tif|dpg|jpeg|eot|svg|ttf|woff|json|mp4|rmvb|rm|wmv|avi|3gp)$ ){
              add_header Cache-Control max-age=7776000;
              add_header Access-Control-Allow-Origin *;
            }
            try_files $uri $uri/ /index.html;
        }
    }


    server {
        listen       3001;
        server_name  localhost;
        location / {
            root first-child/dist;
            add_header Access-Control-Allow-Origin *;
            if ( $request_uri ~* ^.+.(js|css|jpg|png|gif|tif|dpg|jpeg|eot|svg|ttf|woff|json|mp4|rmvb|rm|wmv|avi|3gp)$ ){
              add_header Cache-Control max-age=7776000;
              add_header Access-Control-Allow-Origin *;
            }
            try_files $uri $uri/ /index.html;
        }
     }


    server {
        listen       3002;
        server_name  localhost;
        location / {
            root second-child/dist;
            add_header Access-Control-Allow-Origin *;
            if ( $request_uri ~* ^.+.(js|css|jpg|png|gif|tif|dpg|jpeg|eot|svg|ttf|woff|json|mp4|rmvb|rm|wmv|avi|3gp)$ ){
              add_header Cache-Control max-age=7776000;
              add_header Access-Control-Allow-Origin *;
            }
            try_files $uri $uri/ /index.html;
        }
     }


    # another virtual host using mix of IP-, name-, and port-based configuration
    #
    #server {
    #    listen       8000;
    #    listen       somename:8080;
    #    server_name  somename  alias  another.alias;


    #    location / {
    #        root   html;
    #        index  index.html index.htm;
    #    }
    #}




    # HTTPS server
    #
    #server {
    #    listen       443 ssl;
    #    server_name  localhost;


    #    ssl_certificate      cert.pem;
    #    ssl_certificate_key  cert.key;


    #    ssl_session_cache    shared:SSL:1m;
    #    ssl_session_timeout  5m;


    #    ssl_ciphers  HIGH:!aNULL:!MD5;
    #    ssl_prefer_server_ciphers  on;


    #    location / {
    #        root   html;
    #        index  index.html index.htm;
    #    }
    #}


}

3、配置完成后启动nginx
在这里插入图片描述
在这里插入图片描述

启动之后打开浏览器,在地址栏输入以下地址,可以看到对应的基座和子应用页面

(1)基座

127.0.0.1:3000

在这里插入图片描述
(2)子应用1

127.0.0.1:3001

在这里插入图片描述
(3)子应用2

127.0.0.1:3002

在这里插入图片描述

总结

到此对官网的学习就差不多结束了,对于micro-app,本人纯新手,以上都是对官网的学习,如果文章中有什么错误,欢迎下方评论纠正。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/482326.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Vue(标签属性:ref、配置项:props、混入mixin、插件、样式属性:scroped)

一、ref&#xff08;打标识&#xff09; 前面提及到了标签属性&#xff1a;keys 这里将了解ref&#xff1a;打标识 正常布置脚手架并创建入口文件main.js,引入组件 1. 可以给元素注册引用信息&#xff08;获取真实DOM&#xff09; 给一个按钮获取上方的dom的方法&#xff0c;方…

log4j2实现日志输出

引言 日志是我们在软件开发过程中非常重要的一个组成部分&#xff0c;它能够记录系统运行时的各种信息和异常&#xff0c;方便我们在需要的时候进行排查和调试。而Log4j2是目前最为流行的Java日志框架之一&#xff0c;它提供了丰富的日志输出方式和配置选项&#xff0c;可以满…

设计模式——装饰器模式(Decorator Pattern)

很久没有写博客了&#xff0c;最近也有很多事情要处理&#xff0c;也在努力的备考软件考试&#xff0c;正好模拟题中有一道关于装饰器模式的题&#xff0c;觉得还不错&#xff0c;所以特地写一篇文章希望能分享给小伙伴们。 装饰器模式的作用&#xff1a;允许向一个现有的对象…

vue3导入elcel表格并展示(使用xlsx插件+vite+element-plus)/js上传表格(js+xlsx)

表格内容(本博客演示的表格,这里其实可以更换任意表格,动态展示的) 安装插件xlsx npm install xlsx组件的所有代码(附解释) <script setup> import { ref } from "vue"; import * as XLSX from "xlsx"; // 把文件按照二进制进行读取 function read…

解密PyTorch动态计算图:打破深度学习束缚的秘密武器

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

【C++】STL之priority_queue类源码剖析

目录 概述 算法 源码 PriorityQueue.h test.cpp 测试结果 概述 priority_queue&#xff1a;优先级队列&#xff0c;包含在头文件<queue>中 优先级队列类似于堆结构&#xff0c;优先级最高的元素被置为堆顶&#xff0c;最优先被弹出top()和删除pop() 优先级队列的…

Python凸包

文章目录 ConvexHullQG三维情况ConvexHull属性 ConvexHull ConvexHull是spatial中的一个类&#xff0c;主要功能是找到一组点的边缘&#xff0c;并做一个凸包。其必要的初始化参数为一个点集&#xff0c;点集格式为 n m n\times m nm维度的数组&#xff0c;n为点集中点的个数…

定位的特殊应用

注意&#xff1a;发生固定定位&#xff0c;绝对定位后&#xff0c;元素都变成了定位元素&#xff0c;默认高宽被内容撑开&#xff0c;则可以设置宽高&#xff1b;以下只针对绝对定位和固定定位的元素&#xff0c;不包括相对定位元素。 1.定位元素块的宽充满包含块 前提&#x…

封装建立-SMD封装

1. 看规格书&#xff0c;建立需要的焊盘&#xff0c;命名。注意padstack editor保存路径中不能有中文。 2.新建.dra工程&#xff0c;layout/pin 在里面筛选需要的焊盘。 3. 放置焊盘&#xff0c;需要计算精确坐标&#xff0c;allegro里command用x 0 0命令可以定位到原点。 4…

Python综合案例-学生数据可视化

近年来,数据分析和可视化已经成为了许多领域中的重要工具。在教育领域中,通过对学生的表现和行为进行数据分析和可视化,可以更好地了解学生的学习状态,发现问题、改进教学,并提高学生成绩。本文将介绍一个 Python 综合案例,使用 Pandas 和 Seaborn 库,对学生的数据进行清…

MySQL几种备份方式对比,你用对了吗?

各备份方法对比 备份数据的策略需要根据几种维度考虑 备份能承受最大丢失数据量 备份期间系统可以处于哪种情况&#xff08;不可用&#xff0c;部分可用&#xff0c;完全可用&#xff09; 数据恢复时长 需要恢复全量数据还是增量数据 备份数据的方法 逻辑备份&#xff1a;…

推荐算法实战项目:Deep Crossing 模型原理以及案例实战(附完整 Python 代码)

本文要介绍的Deep Crossing模型是由微软研究院在论文《Deep Crossing: Web-Scale Modeling without Manually Crafted Combinatorial Features》中提出的&#xff0c;它主要是用来解决大规模特征自动组合问题&#xff0c;从而减轻或者避免手工进行特征组合的开销。 Deep Cross…

推荐算法实战项目:DCN 原理以及案例实战(附完整 Python 代码)

本文要介绍的是由斯坦福大学联合Google的研究人员发表的论文《Deep & Cross Network for Ad Click Predictions》中提出的Deep&Cross模型&#xff0c;简称DCN。 DCN模型是Wide&Deep的改进版本&#xff0c;其中Deep部分的设计思路与Wide&Deep没有发生本质的变化…

asp.net基于web的校园美食派送配送系统

1&#xff0e;系统登录&#xff1a;系统登录是用户访问系统的路口&#xff0c;设计了系统登录界面&#xff0c;包括用户名、密码和验证码&#xff0c;然后对登录进来的用户判断身份信息&#xff0c;判断是管理员用户还是普通用户。 2&#xff0e;系统用户管理&#xff1a;不管是…

OpenHarmony JS项目开发流程

一、配置OpenHarmony开发环境 1.1软件需求 1&#xff09;下载并安装好DevEco Studio 2.1 Release及以上版本&#xff0c;下载链接&#xff1a;https://developer.harmonyos.com/cn/develop/deveco-studio#download 2&#xff09;获取OpenHarmony SDK包并解压&#xff0c;下载…

学历不仅是敲门砖,也是我下不来的高台,更是孔乙己脱不下的长衫

学历不仅是敲门砖&#xff0c;也是我下不来的高台&#xff0c;更是孔乙己脱不下的长衫 鲁迅《孔乙己》是一篇具有深刻思想和感人情感的短篇小说&#xff0c;通过酒肆里的故事反映社会的残酷和人性的悲哀&#xff1b; 故事中的孔乙己是一个身世不明、生活贫困的酒鬼&#xff0c…

OpenCV学习小记

OpenCV学习小记 &#x1f388;&#x1f388;记在最前&#x1f388;&#x1f388;图像处理的基本操作✨读取图像✨显示图像✨保存图像✨获取图像属性 &#x1f388;&#x1f388;像素的操作✨像素&#x1f514;获取像素的BGR值&#x1f514;修改像素的BGR值 ✨使用NumPy模块操作…

2023年值得关注的20大网络安全趋势

随着围绕所有企业的数字革命&#xff0c;无论大小&#xff0c;企业、组织甚至政府都依赖计算机化系统来管理他们的日常活动&#xff0c;从而使网络安全成为保护数据免受各种在线攻击或任何未经授权访问的主要目标。 随着数据泄露、勒索软件和黑客攻击的新闻成为常态&#xff0…

基于计算机视觉的手势识别技术

一个不知名大学生&#xff0c;江湖人称菜狗 original author: Jacky Li Email : 3435673055qq.com Time of completion&#xff1a;2023.5.2 Last edited: 2023.5.2 手语是一种主要由听力困难或耳聋的人使用的交流方式。这种基于手势的语言可以让人们轻松地表达想法和想法&…

RTT开发之windows 环境配置

1. 安装python 有些文章说支持2.7&#xff0c; 实测3.9环境也是OK的 2. 安装scons组件 其他文章多是下载安装&#xff0c;实际操作麻烦还成功率低&#xff0c; 直接pip安装 pip install scons 然后命令测试 D:\rt-thread-5.0.0\bsp\wch\arm\ch579m>scons scons: Readin…