vue3使用指南

news2025/1/1 22:16:04

vue3使用指南

主要介绍vue3的使用,同时会涉及到vue2,并会讲解其中的一些差异。

安装

CDN引入

如果想快速体验,可以直接通过cdn进行引入。

<div id="app">{{msg}}</div>
<script src="https://unpkg.com/vue@3/dist/vue.global.js"></script>
<script >
  Vue.createApp({
    data() {
      return {
        msg: 'Vue3'
      }
    }
  }).mount('#app')
</script>

通过 CDN 引入 Vue 时,由于不涉及到构建步骤,可以使得设置更加简单,并且可以用于增强静态的 HTML 或与后端框架集成。但是这也意味着无法使用SFC(单文件)语法。

使用es模块构建

现代浏览器大多都已原生支持 ES 模块,可以像这样通过 CDN 以及原生 ES 模块使用 Vue

<div id="app">{{msg}}</div>
<script type="module">
  import { createApp } from 'https://unpkg.com/vue@3/dist/vue.esm-browser.js'
  createApp({
    data() {
      return {
        msg: 'Vue3'
      }
    }
  }).mount('#app')
</script>

使用import maps

<div id="app">{{msg}}</div>
<script type="importmap">
  {
    "imports": {
      "vue": "https://unpkg.com/vue@3/dist/vue.esm-browser.js"
    }
  }
</script>
<script type="module">
  import {createApp} from 'vue'
  createApp({
    data() {
      return {
        msg: 'Hello Vue!'
      }
    }
  }).mount('#app')
</script>

工程化

使用vue@latest方式

Vue 官方的项目脚手架工具,内部使用的是vite进行处理。

npm init vue@latest

这个指令会安装并执行create-vue,之后按照提示安装依赖并运行即可。
在这里插入图片描述
请注意,生成的项目中的示例组件使用的是组合式 API<script setup>,而非选项式 API

使用@vue/cli方式

虽然vue现在已经不推荐使用该方式,内部使用的webpack进行处理,但是依旧提供了vue3模板的下载。

安装@vue/cli

npm i @vue/cli -g

# 创建项目
vue create xxx

在这里插入图片描述
可以选择官方的vue3模板,也可以自行配置。
在这里插入图片描述

使用vite创建

# npm 6.x
npm init vite@latest <project-name> --template vue
# npm 7+,需要加上额外的双短横线
npm init vite@latest <project-name> -- --template vue

在这里插入图片描述

创建实例

vue2中,通过引入vue文件,进行实例的创建。

import Vue from 'vue'

new Vue({
  ...
})

而在vue3中做了改变,被移动到了由新的 createApp 方法所创建的应用实例上。

import { createApp } from 'vue'

const app = createApp({})

组合式API

vue2.x中,一个组件中通常包含了datacomputedmethodswatch等组件选项来组织逻辑,当组件变得越来越大时,逻辑关注点的列表也会增长,尤其对于那些一开始没有编写这些组件的人来说,这会导致组件难以阅读和理解。

这种碎片化使得理解和维护复杂组件变得困难。选项的分离掩盖了潜在的逻辑问题。此外,在处理单个逻辑关注点时,我们必须不断地“跳转”相关代码的选项块。

为了使得能将同一个逻辑的相关代码单独放在一起,vue3引入了组合式API的概念,使用setup来编写组件式API

来简单看看一个例子理解一下组合式API

<template>
  <!-- 计数器 -->
  <p>{{obj.count}}</p>
  <button @click="add">+1</button>

  <!-- v-if展示区域 -->
  <button @click="show">显示</button>
  <button @click="hide">隐藏</button>
  <div v-if="showDivFlag">一个被控制显隐的div</div>
</template>
<script>
import { reactive, ref } from 'vue'
// 计数器逻辑
function count() {
  const data = {
    count: 0
  }
  const obj = reactive(data);
  function add() {
    obj.count++;
  }
  return {
    obj,
    add
  }
}
// v-if展示区域逻辑
function vifDiv() {
  const showDivFlag = ref(true)
  function show() {
    showDivFlag.value = true
  }
  function hide() {
    showDivFlag.value = false
  }
  return {
    showDivFlag,
    show,
    hide
  }
}
export default {
  setup() {
    const { obj, add } = count();
    const {showDivFlag, show, hide} = vifDiv();
    return {
      obj,
      add,
      showDivFlag,
      show,
      hide
    }
  },
}
</script>

setup内对数据进行响应式,之后将datamethods返回,这样,模板中就可以使用这些。比起vue2中选项式API,使用setup能将各个模块逻辑单独放在一起,避免在methods中定义一大堆的方法。

以上例子使用vue2的选项式API

<template>
  <!-- 计数器 -->
  <p>{{obj.count}}</p>
  <button @click="add">+1</button>

  <!-- v-if展示区域 -->
  <button @click="show">显示</button>
  <button @click="hide">隐藏</button>
  <div v-if="showDivFlag">一个被控制显隐的div</div>
</template>
<script>
export default {
  data() {
    return {
      obj: {
        count: 0
      },
      showDivFlag: true
    }
  },
  methods: {
    add() {
      this.obj.count++;
    },
    show() {
      this.showDivFlag = true;
    },
    hide() {
      this.showDivFlag = false;
    }
  }
}
</script>

组合式 API 的核心思想是直接在函数作用域内定义响应式状态变量,并将从多个函数中得到的状态组合起来处理复杂问题。这种形式更加自由,也需要开发者对 Vue 的响应式系统有更深的理解才能高效使用。相应的,它的灵活性也使得组织和重用逻辑的模式变得更加强大。

两种 API 风格都能够覆盖大部分的应用场景。它们只是同一个底层系统所提供的两套不同的接口。实际上,选项式 API 是在组合式 API 的基础上实现的。关于 Vue 的基础概念和知识在它们之间都是通用的。

setup()

组件中的一个选项,处于生命周期函数 beforeCreate 钩子函数之前的函数,props 被解析之后执行。同时也是组合式API的入口。

接受两个参数:

  • props: 父组件的传值
    如果想在setup中使用props,必须在propsprop进行声明,未声明的prop的无法在setup中的props中解构出来(可以通过context.attrs获取)。声明了的prop即使父组件没有传,在setupprops仍然可以获取,但是值为undefined或者props中定义的默认值。
    props: {
      key: String,
      value: {
        type: String,
        default: 'default'
      }
    },
    setup(props) {
      console.log(props);
    }
    
    setup 函数中的 props 是响应式的,当传入新的 prop 时,它将被更新。但是,因为 props 是响应式的,你不能使用 ES6 解构,因为它会消除 prop 的响应性。可以使用toRefs来解构。
    setup(props) {
      const {key, value} = toRefs(props);
    }
    
  • context: 上下文, context 是一个普通的 JavaScript 对象,也就是说,它不是响应式的,这意味着你可以安全地对 context 使用 ES6 解构。
    context有4个属性:
    • attrs: 非响应式对象,跟vue2this.$attrs一样,用于接收父组件传的但是子组件中未声明的值。
    • emit: 跟vue2this.$emit一样 用于向父组件触发事件。
    • slots:非响应式对象,跟vue2this.$slots一样,用于获取插槽的信息。
    • expose:用于当在setup中使用渲染函数,而无法将组件中的data这些暴露给外部组件(外部组件通过ref获取)。
      // 子组件
      <script>
      import { h, ref } from 'vue'
      export default {
        setup(props, { expose }) {
          const count = ref(0)
          const increment = () => ++count.value
          expose({
            increment
          })
          return () => h('div', count.value)
        }
      }
      </script>
      
      <!-- 父组件 -->
      <son ref="son"></son>
      <script>
      export default {
        mounted() {
          console.log(this.$refs['son'].increment())
        }
      }
      </script>
      

执行 setup 时,组件实例尚未被创建(在 setup() 内部,this 不会是该活跃实例的引用,即不指向vue实例,Vue 为了避免开发者错误的使用,直接将 setup 函数中的this修改成了 undefined
setup() 中返回的对象上的 property 返回并可以在模板中被访问时,它将自动展开为内部值。不需要在模板中追加 .value
setup函数只能是同步的不能是异步的

<script setup>其实就是 setup 函数的一个语法糖。标签内的代码会被编译成setup()函数里的内容(这样也能使得组件实例在被创建的时候才会执行,而普通的<script>在组件引入时就执行)。

下面举几个例子看看语法糖的简便:

1. 变量和方法

<script setup>
  import {ref} from 'vue';
  import {minusNum} from 'xxx';
  const num = ref(1);
  function addNum() {
    num.value++;
  }
</script>

<script>
  import {ref} from 'vue';
  import {minusNum} from 'xxx';
  setup() {
    const num = ref(1);
    function addNum() {
      num.value++;
    }
    return {
      num,
      addNum,
      minusNum
    }
  }
</script>

使用语法糖不需要将变量和方法(包括外部文件中的方法)返回即可使用。

2. 组件注册、自定义指令

<template>
  <h1 v-direct>directive</h1>
</template>
<script setup>
  import { Component1 } from 'xxx';
  import { vDirect } from 'xxx';
</script>

<script>
  import { Component1 } from 'xxx';
  import { direct } from 'xxx';
  export default {
    components: {
      Component1
    },
    directives: {
      direct
    }
  }
</script>

使用语法糖组件就不需要在 component 注册,在组件中定义指令虽然也不需要在 directives 中注册,但是需要遵循 vNameOfDirective 这样的命名规范,否则并不会注册成指令。

3. 父子组件通信

<script setup>
  import { defineProps, defineEmits } from 'vue'
  const emit = defineEmits(['addNum']);
  const addNumEmit = () => {
    emit('addNum', 1);
  }

  const props = defineProps({
    num: {
      type: Number,
      default: 0
    }
  })

  console.log(props.num);
</script>

<script>
  export default {
    props: {
      num: {
        type: Number,
        default: 0
      }
    },
    setup (props, { emit }) {
      console.log(props.num)
      const addNumEmit = () => {
        emit('addNum', 1)
      }
      return {
        addNumEmit
      }
    }
  }
</script>

可以与普通script一起使用

<script setup> 可以和普通的 <script> 一起使用。

<script>
export default {
  mounted() {
    console.log('mounted');
  }
}
</script>
<script setup>
  import { onMounted } from 'vue';
  onMounted(() => {
    console.log('setup mounted')
  })
</script>

setup中定义的生命周期钩子比普通script先执行。

生命周期

setup中通过onXXX来注册生命周期:

选项式 API 的生命周期选项和组合式 API 之间的映射

因为 setup 是围绕 beforeCreatecreated 生命周期钩子运行的,因此setup内部的没有beforeCreatecreated对应的钩子映射。

  • beforeMount -> onBeforeMount
  • mounted -> onMounted
  • beforeUpdate -> onBeforeUpdate
  • updated -> onUpdated
  • beforeUnmount -> onBeforeUnmount
  • unmounted -> onUnmounted
  • errorCaptured -> onErrorCaptured
  • renderTracked -> onRenderTracked
  • renderTriggered -> onRenderTriggered
  • activated -> onActivated
  • deactivated -> onDeactivated

这些函数接受一个回调函数,当钩子被组件调用时将会被执行:

export default {
  setup() {
    // mounted
    onMounted(() => {
      console.log('Component is mounted!')
    })
  }
}

选项式API与组合式API混用

选项式API与组合式API是允许混用的(需要Vue3vue2.7后的版本)。如下:

<script>
  import { reactive } from 'vue';
  export default {
    data() {
      return {
        value: 1
      }
    },
    props: {
      num: {
        type: Number,
        default: 0
      }
    },
    setup() {
      let name = reactive('leo');
      return {
        name
      }
    },
    methods: {
      getName() {
        console.log(this.name);
      }
    }
  }
</script>

在混用的时候需要注意由于setup函数里的this指向undefined,因此不能使用选项式中定义的datamethods这些。

组件

组件注册

全局注册

vue2中是直接使用vue的原型中component进行注册.

import Vue from 'vue';
Vue.component('xxx', xxx);

vue3中,则是在实例中进行添加

import { createApp } from 'vue'

const app = createApp({})
app.component('xxx', xxx);

局部注册

前面提到了对于<script setup>不需要进行声明,直接引入即可使用,否则跟vue2一样也需要在components中注册。

<template>
  <Component1 />
</template>
<script setup>
  import { Component1 } from 'xxx';
</script>

<script>
  import { Component1 } from 'xxx';
  export default {
    components: {
      Component1
    }
  }
</script>

响应式API

reactive

reactive用于将数据变成响应式数据。调用reactive后返回的对象是响应式副本而非原始对象。其原理就是将传入的数据包装成一个Proxy对象。

import { reactive, watchEffect } from 'vue';
const data = {
  count: 0
};
const obj = reactive(data);
data === obj // false

watchEffect(() => {
  // 用于响应性追踪
  console.log(obj.count);
});

setTimeout(() => {
  obj.count++;
}, 2000);

响应式转换是“深层”的——它影响所有嵌套 property。在基于 ES2015 Proxy 的实现中,返回的 proxy 是不等于原始对象的。建议只使用响应式 proxy,避免依赖原始对象。

reactive用于复杂数据类型,比如对象和数组等,当传入基础数据类型时,默认情况下修改数据,界面不会自动更新,如果想更新,可以通过重新赋值的方式。

readonly

接受一个对象 (响应式或纯对象) 或 ref 并返回原始对象的只读代理。只读代理是深层的:任何被访问的嵌套 property 也是只读的,不能改变值,否则是报错。

import { reactive, watchEffect, readonly } from 'vue';
const data = {
  count: 0
};
const obj = reactive(data);
const copy = readonly(obj);

watchEffect(() => {
  // 用于响应性追踪
  console.log(obj.count)
});

setTimeout(() => {
  obj.count++;
  copy.count++;
}, 2000);

ref

接受一个内部值并返回一个响应式且可变的 ref 对象。ref 对象仅有一个 .value值,指向该内部值。跟reactive类似,也是将数据变成响应式。

import { ref, watchEffect } from 'vue';

const count = ref(0);
const obj = ref({
  count: 0
})
console.log(obj)
watchEffect(() => {
  console.log(count.value);
})
watchEffect(() => {
  console.log('obj.value.count: ', obj.value.count);
})

count.value++;
obj.value.count++;

ref和reactive的区别

  • ref是把值类型添加一层包装,使其变成响应式的引用类型的值。ref(0) --> reactive( { value:0 })
  • reactive 则是引用类型的值变成响应式的值。

两者的区别只是在于是否需要添加一层引用包装,对于对象而言,添加一层包装后会被reactive处理为深层的响应式对象,在调用unref后就能看到其实对象是一个Reactive对象

像上面的例子,使用ref同样可以将对象响应化,不过访问的时候需要调用value.去访问内部属性。所以对于对象而言,最好使用reative去响应化处理。

watch

vue3中组合式api watchvue2中选项式的watch完全等效。watch 需要监听特定的数据源,并在单独的回调函数中执行副作用。默认情况下,它也是惰性的——即回调仅在监听源发生变化时被调用。

watchEffect 相比,watch 允许我们:

  • 惰性地执行副作用;
  • 更具体地说明应触发监听器重新运行的状态;
  • 访问被监听状态的先前值和当前值。

监听单一源

源数据可以是一个reactive,也可以直接是一个 ref
语法

watch( name , callback, options )
  • name: 需要监听的属性或者返回值的getter函数
  • callback: 属性改变后执行的方法,接受两个参数
    • newVal: 新值
    • oldVal: 旧值
  • options: 配置项,可配置如下
    • deep: Boolean, 是否深度监听
    • immediate: Boolean,是否立即执行
// 侦听reactive
const state = reactive({ count: 0, value: 1 })
// 只监听对象中的count
watch(
  // 使用getter函数保证只监听了state中的count
  () => state.count,
  (count, prevCount) => {
    /* ... */
  }
)

// 侦听state中所有属性
watch(state, (newVal, oldVal) => {
  // ...
})

// 直接侦听一个 ref
const count = ref(0)
watch(count, (count, prevCount) => {
  /* ... */
})

监听多个源

使用数组来同时侦听多个源数据,当任一源数据发生改变都会触发watch

const state = reactive({ count: 0, value: 1 });
const count = ref(0)
watch([state, count], ([newState, newCount], [oldState, oldCount]) => {
  console.log(newState);
  console.log(newCount);
})

state.count++;
count.value++;

深度监听

watch的第三个参数中传入deep: true

const state = reactive({ count: 0, value: { status: false } });
watch(() => state, (newVal, oldVal) => {
  console.log('不会触发')
})

watch(() => state, (newVal, oldVal) => {
  console.log('触发深度监听')
}, {
  deep: true
})

state.value.status = true;

由于getter方法只返回state,对没有对内部的对象进行监听,因此内部对象的属性发生改变不会触发watch

当没有使用getter方法而是传入state这个reactive数据,则不需要设置deep: true都会进行深度监听。

立即执行

watch的第三个参数中传入immediate: true,当传入数据就会执行一次。

watchEffect

立即执行传入的一个函数,响应式追踪其依赖,在其依赖变更时重新运行该函数

它会监听引用数据类型的所有属性,不需要具体到某个属性,一旦运行就会立即监听,组件卸载的时候会停止监听

const count = ref(0)

watchEffect(() => console.log(count.value))
// -> logs 0

setTimeout(() => {
  count.value++
  // -> logs 1
}, 100)

停止监听

watchEffect 在组件的 setup() 函数或生命周期钩子被调用时,侦听器会被链接到该组件的生命周期,并在组件卸载时自动停止。

在一些情况下,也可以显式调用返回值以停止侦听

const stop = watchEffect(() => {
  /* ... */
})

// later
stop()

清除副作用

有时副作用函数会执行一些异步的副作用,这些响应需要在其失效时清除 (即完成之前状态已改变了) 。所以侦听副作用传入的函数可以接收一个 onInvalidate 函数作入参,用来注册清理失效时的回调。当以下情况发生时,这个失效回调会被触发:

  • 副作用即将重新执行时
  • 侦听器被停止 (如果在 setup() 或生命周期钩子函数中使用了 watchEffect,则在组件卸载时)
watchEffect(onInvalidate => {
  const token = performAsyncOperation(id.value)
  onInvalidate(() => {
    // id has changed or watcher is stopped.
    // invalidate previously pending async operation
    token.cancel()
  })
})

computed

接受一个 getter 函数,并根据 getter 的返回值返回一个不可变的响应式 ref 对象。

const count = ref(1)
const plusOne = computed(() => count.value + 1)

console.log(plusOne.value) // 2

plusOne.value++ // 错误

或者,接受一个具有 getset 函数的对象,用来创建可写的 ref 对象。

const count = ref(1)
const plusOne = computed({
  get: () => count.value + 1,
  set: val => {
    count.value = val - 1
  }
})

plusOne.value = 1
console.log(count.value) // 0

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

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

相关文章

谷歌推出下一代大型语言模型 PaLM 2

谷歌在 2023 年度 I/O 大会上宣布推出了其下一代大型语言模型 PaLM 2&#xff0c;擅长高级推理任务&#xff0c;包括代码和数学、分类和问答、翻译和多语言能力以及自然语言生成。 谷歌声称 PaLM 2 是一种最先进的语言模型&#xff0c;要优于其之前所有的 LLM&#xff0c;包括…

网络拓扑架构规划设计

小型组网架构 1.网络拓扑 终端用户接入到交换机,交换机直连防火墙构成的简单网络,防火墙连接internet,对内网的用户进行安全控制 2.特点 用户接入数量较少:小型网络应用于接入用户数量较少的场景,一般支持几个至几十个用户 网络覆盖范围小:网络覆盖范围一般也是一个…

网络拓扑图制作软件

1.亿图图示(EdrawMax) 最像VISIO的网络拓扑制作工具 亿图图示实际上就是国产的Visio,与Visio非常相似。支持B/S架构,查看拓扑非常方便,模版也比较丰富。缺点与Visio也类似,画简单的网络拓扑没有问题,但针对大型网络拓扑管理而言,既需要能制作拓扑,又需要具有管理分析…

Hystrix详解及实践---SpringCloud组件(四)

Hystrix详解及实践 1.Hystrix简介2.雪崩问题3.服务降级、线程隔离、原理3.1.服务降级实践&#xff08;在feign的基础上实现&#xff09;1 添加Hystrix依赖2.在yml中开启熔断功能3.编写降级逻辑4.重启测试 4.服务熔断(Circuit Breaker)、原理4.1. 熔断原理4.2.动手实践 5.Hystri…

5个模板非常多的免费样机素材分享

样机素材是设计行业的不可缺少的素材之一&#xff0c;设计师可以将自己的设计作品&#xff0c;应用到一个效果图中进行展示&#xff0c;让你的设计作品看起来更加形象逼真。 本文分享5个非常给力的样机素材网站 1.即时设计资源社区 即时设计是一款「专业UI设计工具」&#x…

【edusrc】某校园网登录系统存在的一个简单爆破

前言 红中(hong_zh0) CSDN内容合伙人、2023年新星计划web安全方向导师、 华为MindSpore截至目前最年轻的优秀开发者、IK&N战队队长、 阿里云专家博主、华为网络安全云享专家、腾讯云自媒体分享计划博主 本文主讲思路&#xff0c;可能有点废话。 该漏洞已修复 最开始的思…

华为OD机试真题 Java 实现【字符串重新排序】【2023Q1 100分】

一、题目描述 给定一个字串s&#xff0c;s包含以空格分隔的若干个单词&#xff0c;请对s进行如下处理后输出&#xff1a; 1、单词内部调整 对每个单词字母重新按字典序排序。 2、单词间顺序调整: 统计每个单词出现的次数&#xff0c;并按次数降序排列&#xff1b;次数相同…

存下吧!Spring高频面试题总结

Spring是什么&#xff1f; Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架。 Spring的优点 通过控制反转和依赖注入实现松耦合。支持面向切面的编程&#xff0c;并且把应用业务逻辑和系统服务分开。通过切面和模板减少样板式代码。声明式事务的支持。可以从单…

Windows系统下Chromedriver.exe安装及配置

Windows系统下Chromedriver.exe安装及配置 在利用selenium工具进行Web自动化测试时&#xff0c;必须先要安装浏览器驱动&#xff0c;通常比较常用的是谷歌浏览器和火狐浏览器。 一、浏览器驱动下载地址 1.浏览器驱动官网&#xff1a;http://chromedriver.storage.googleapis…

计算环境安全

计算环境安全 操作系统安全安全机制标识与鉴别访问控制权限管理信道保护安全审计内存保护与文件系统保护 安全部署原则操作系统安全配置密码远程暴力破解安全审计 针对系统的攻击信息收集公开信息收集-搜索引擎信息收集与分析的防范 缓冲区溢出缓冲区溢出基础-堆栈、指针、寄存…

Dubbo源码篇03---点点直连如何实现及背后原理

Dubbo源码篇03---从点点直连探究Complier编译的原理 什么是点点直连实际需求如何实现动态编译&#xff1f;如何发起调用?点点直连原理实现点点直连消费端提供端测试 点点直连小结 什么是点点直连 Dubbo正常的请求模型&#xff0c;都是消费端从注册中心拉取服务提供者列表&…

spring 命令执行 (CVE-2022-22947)

漏洞原理 该漏洞产生的原因是由于&#xff0c;当AddResponseHeaderGatewayFilterFactory传入的值进行计算(getValue())的时候&#xff0c;会逐一向上调用对应的方法&#xff0c;直到进入带有SpEL表达式解析器的位置进行最后的解析&#xff0c;从而触发了SpEL表达式注入漏洞。漏…

分组背包问题 java

&#x1f351; 算法题解专栏 &#x1f351; 分组背包问题 输入 3 5 2 1 2 2 4 1 3 4 1 4 5输出 8&#x1f468;‍&#x1f3eb; 参考题解 &#x1f351; 终极简化版 import java.util.Scanner;public class 分组背包极简版 {static int N 110;static int[] f new int[N]…

【大数据之Hadoop】三十、HDFS故障排除

使用3台服务器&#xff0c;恢复yarn快照。 1 NameNode故障处理 出现NameNode进程挂了并且存储的数据也丢失了&#xff0c;怎么恢复NameNode。 故障模拟&#xff1a; &#xff08;1&#xff09;kill掉NameNode的进程&#xff1a; kill -9 进程ID&#xff08;2&#xff09;删…

SpringCloud:微服务保护之授权规则

授权规则可以对请求方来源做判断和控制。 1.授权规则 1.1.基本规则 授权规则可以对调用方的来源做控制&#xff0c;有白名单和黑名单两种方式。 白名单&#xff1a;来源&#xff08;origin&#xff09;在白名单内的调用者允许访问 黑名单&#xff1a;来源&#xff08;origi…

【JAVAEE】阻塞队列的实现

目录 阻塞队列 生产者消费者模型 消息队列 消息队列的作用 1.解耦 2.削峰填谷 3.异步 演示JDK中的阻塞队列 实现一个阻塞队列 阻塞队列 队列&#xff0c;是一种先进先出&#xff08;FIFO&#xff09;数据结构。 阻塞队列也满足队列的特性&#xff1a; 入队元素时&am…

详细版易学版TypeScript - 泛型详解

一、泛型的基本使用 泛型&#xff1a;定义时不确定是什么类型&#xff0c;就先定义为泛型&#xff0c;等到使用时再去确定具体的类型 //实现需求&#xff1a;根据传入的数据value和数量count&#xff0c;返回有count个value值的数组 const myFuncTest1 (value: number, count:…

一文搞懂Bootloader跳转到APP 的方法和原理

一 跳转方法 1、检查栈顶地址是否合法 if (((*(uint32_t*)(NRF52840_APP_BASE)) & 0xffff0000 ) 0x20040000 ){nrf_bootloader_app_start();}在编译生成的APP.bin文件中,前4个字节存放的是__initial_sp&#xff0c;紧接着第二个地址存放的是Reset_Handler&#xff1b;这两…

【机器学习】决策树(基础篇)

决策树&#xff08;基础篇---分类问题&#xff0c;回归问题会另出一篇博客,但也是基础篇&#xff09; 思维导图前言了解决策树&#xff08;前后观看&#xff09;使用决策树&#xff08;感性的认识&#xff09;如何生成决策树&#xff08;原理部分&#xff0c;此部分有局限性&am…

一次失败的面试经历:我只想找个工作,你却用面试题羞辱我

金三银四就要来了&#xff0c;即将又是一波求职月&#xff0c;面对跳槽的高峰期&#xff0c;很多软件测试人员都希望能拿一个满意的高薪offer&#xff0c;但是随着招聘职位的不断增多&#xff0c;面试的难度也随之加大&#xff0c;而面试官更是会择优录取 小王最近为面试已经焦…