Vue插槽与作用域插槽
title: Vue插槽与作用域插槽
date: 2024/6/1 下午9:07:52
updated: 2024/6/1 下午9:07:52
categories:
- 前端开发
tags: - VueSlot
- ScopeSlot
- 组件通信
- Vue2/3插槽
- 作用域API
- 动态插槽
- 插槽优化

第1章:插槽的概念与原理
插槽的定义
在Vue.js中,插槽(Slots)是一种强大的功能,它允许你将内容分发到组件的各个部分。简单来说,插槽是组件内部预留的一个位置,用于放置组件使用者提供的HTML结构。这样,组件的使用者可以根据自己的需求,灵活地填充或替换组件的某些部分,而不需要修改组件的内部实现。
插槽的工作原理
插槽的工作原理基于Vue的模板编译机制。当一个组件包含插槽时,它会在模板中定义一个或多个插槽的位置。这些位置可以是默认插槽,也可以是命名插槽。当组件被渲染时,父组件传递给子组件的内容会被插入到这些插槽位置中。
例如,一个简单的插槽定义如下:
<!-- 子组件模板 -->
<template><div><slot></slot></div>
</template>
在父组件中使用这个子组件时,可以这样传递内容:
<!-- 父组件模板 -->
<template><child-component><p>这是插入到子组件插槽中的内容</p></child-component>
</template>
当父组件渲染时,<p>这是插入到子组件插槽中的内容</p>这段内容会被插入到子组件的<slot></slot>位置。
插槽与组件复用的关系
插槽与组件复用有着密切的关系。通过使用插槽,组件可以变得更加灵活和可复用。组件的开发者可以定义一个通用的结构,而组件的使用者则可以根据自己的需求,通过插槽来填充特定的内容。这样,同一个组件可以在不同的场景下被复用,同时保持其内容的多样性和定制性。
例如,一个通用的卡片组件可以定义一个插槽,用于放置卡片的内容。这样,无论卡片是用于展示文章、产品还是其他任何信息,都可以通过插槽来填充相应的内容,而不需要为每种情况单独创建一个组件。
第2章:默认插槽
默认插槽的使用场景
默认插槽主要适用于那些希望在组件内部保留一些默认内容,同时允许用户自定义内容的场景。例如,一个简单的导航栏组件,它通常包含一个主导航条和一个可自定义的附加区域,如搜索框或用户信息。这时,就可以使用默认插槽来包含默认的主导航条,并允许使用者填充附加区域。
默认插槽的语法与实现
默认插槽在Vue组件模板中使用<slot>标签,没有指定名称,就是默认插槽。默认插槽通常放在组件的主体部分,这样任何传入的内容都会被插入到这个位置。
<!-- 子组件模板 -->
<template><div><header>默认导航</header><slot></slot> <!-- 这里就是默认插槽 --></div>
</template>
在父组件中使用时,可以像这样插入内容:
<!-- 父组件模板 -->
<template><my-component><!-- 自定义内容 --><div>搜索框</div></my-component>
</template>
当渲染时,<div>搜索框</div>会被插入到<my-component>的默认插槽位置。
默认插槽示例
<template><div class="parent"><h1>父组件</h1><my-component><p>这是默认插槽的内容</p></my-component></div>
</template><script>
import MyComponent from './MyComponent.vue';export default {components: {MyComponent}
};
</script>
在这个例子中,my-component组件的默认插槽会被<p>这是默认插槽的内容</p>替换,而<h1>父组件</h1>会被保留在外层,不会影响到插槽内的内容。
第3章:命名插槽
命名插槽的概念
命名插槽是Vue组件中另一种插槽的形式,它允许在组件模板中为不同的插槽指定不同的名称。这有助于组件开发者更好地控制组件的内容,并使组件更加灵活和易于使用。
命名插槽的语法与实现
命名插槽在Vue组件模板中使用<template>标签和v-slot指令,并在<template>标签上指定v-slot的值作为插槽的名称。
<!-- 子组件模板 -->
<template><div><header>默认导航</header><slot name="header"></slot> <!-- 这里是一个命名插槽 --><slot></slot> <!-- 这里是默认插槽 --><slot name="footer"></slot> <!-- 这里是另一个命名插槽 --></div>
</template>
在父组件中使用时,可以像这样为插槽提供内容:
<!-- 父组件模板 -->
<template><my-component><template #header><h1>自定义标题</h1></template><p>这是默认插槽的内容</p><template #footer><p>自定义页脚</p></template></my-component>
</template>
当渲染时,<h1>自定义标题</h1>会被插入到<my-component>的header命名插槽位置,<p>这是默认插槽的内容</p>
会被插入到默认插槽位置,<p>自定义页脚</p>会被插入到footer命名插槽位置。
命名插槽示例
<template><div class="parent"><h1>父组件</h1><my-component><template #header><h1>自定义标题</h1></template><p>这是默认插槽的内容</p><template #footer><p>自定义页脚</p></template></my-component></div>
</template><script>
import MyComponent from './MyComponent.vue';export default {components: {MyComponent}
};
</script>
在这个例子中,my-component组件的三个插槽分别被赋予了不同的内容。header命名插槽被<h1>自定义标题</h1>
替换,默认插槽被<p>这是默认插槽的内容</p>替换,footer命名插槽被<p>自定义页脚</p>替换。这种灵活的插槽机制使得组件的定制化能力得到了大大的提升。
第4章:插槽属性与作用域
插槽属性的传递
在Vue中,插槽本身并不支持传递属性,但可以通过父组件传递数据到子组件的插槽。这通常是通过v-bind或prop
属性实现的。例如,如果你有一个父组件想要传递一个对象到子组件的插槽:
<!-- 父组件 -->
<template><my-component><template #customSlot><div :data="slotData"></div></template></my-component>
</template><script>
export default {components: {MyComponent},data() {return {slotData: { key: 'value' }}}
};
</script>
子组件中的插槽会接收到这个slotData对象。
作用域插槽的原理
作用域插槽(Scoped Slot)是Vue
2.6版本引入的一个特性,用于在子组件内部定义插槽,这样可以在父组件中动态地插入内容。作用域插槽的主要原理是,子组件定义了一个特殊的插槽,父组件可以通过<slot>
标签以及v-slot指令的#符号来引用这个插槽。
作用域插槽的语法与实现
<!-- 子组件 -->
<template><div><slot name="customSlot">默认内容</slot> <!-- 定义一个作用域插槽 --><p>子组件内容</p></div>
</template><script>
export default {name: 'ScopedChild'
};
</script>
在父组件中,你可以这样使用作用域插槽:
<!-- 父组件 -->
<template><ScopedChild><template v-slot:customSlot><h2>自定义内容</h2></template></ScopedChild>
</template><script>
import ScopedChild from './ScopedChild.vue';export default {components: {ScopedChild}
};
</script>
在这个例子中,<h2>自定义内容</h2>会被插入到ScopedChild组件的customSlot
作用域插槽中,如果父组件没有提供内容,那么默认内容“默认内容”会被显示。作用域插槽让内容的传递更加灵活,父组件可以根据需要动态地改变传递给子组件的内容。
第5章:作用域插槽的进阶应用
作用域插槽与渲染函数
在Vue中,你可以使用渲染函数(Render Function)来更加灵活地生成虚拟DOM。当你使用渲染函数时,你可以使用this.$slots
来获取插槽内容,并将它们与渲染函数中的内容结合起来。例如:
<!-- 子组件 -->
<template><div><slot name="customSlot"></slot></div>
</template><script>
export default {name: 'ScopedChild',render(h) {return h('div', [h('h2', '子组件标题'),this.$slots.customSlot // 获取插槽内容]);}
};
</script>
作用域插槽与计算属性
在父组件中,你可以使用计算属性来动态地生成插槽内容,并将其传递给子组件的作用域插槽。例如:
<!-- 父组件 -->
<template><ScopedChild><template v-slot:customSlot><h2>{{ computedTitle }}</h2></template></ScopedChild>
</template><script>
import ScopedChild from './ScopedChild.vue';export default {components: {ScopedChild},computed: {computedTitle() {return '父组件计算的标题';}}
};
</script>
作用域插槽与组件状态管理
在使用Vuex进行组件状态管理时,你可以使用作用域插槽来动态地显示状态数据。例如:
<!-- 子组件 -->
<template><div><slot name="customSlot" :data="stateData"></slot></div>
</template><script>
import { mapState } from 'vuex';export default {name: 'ScopedChild',computed: {...mapState(['stateData'])}
};
</script>
在父组件中,你可以这样使用作用域插槽:
<!-- 父组件 -->
<template><ScopedChild><template v-slot:customSlot="{ data }"><h2>{{ data }}</h2></template></ScopedChild>
</template><script>
import ScopedChild from './ScopedChild.vue';export default {components: {ScopedChild},computed: {...mapState(['stateData'])}
};
</script>
在这个例子中,<h2>{{ data }}</h2>会被插入到ScopedChild组件的customSlot作用域插槽中,并显示stateData
的值。这样,你可以在父组件中动态地显示子组件的状态数据。
第6章:作用域插槽的实际应用案例
动态表单组件
使用作用域插槽可以很方便地创建一个动态表单组件,该组件可以根据传入的数据动态地生成表单元素。例如:
<!-- 子组件:DynamicForm.vue -->
<template><form @submit.prevent="onSubmit"><div v-for="(field, index) in fields" :key="index"><slot :name="field.name" v-bind="field"><input v-if="field.type === 'text'" v-model="field.value" :name="field.name" type="text"><input v-else-if="field.type === 'number'" v-model.number="field.value" :name="field.name" type="number"><select v-else-if="field.type === 'select'" v-model="field.value" :name="field.name"><option v-for="(option, index) in field.options" :key="index" :value="option.value">{{ option.label }}</option></select></slot></div><button type="submit">提交</button></form>
</template><script>
export default {name: 'DynamicForm',props: {fields: {type: Array,required: true}},methods: {onSubmit() {this.$emit('submit', this.fields);}}
};
</script>
在父组件中,你可以这样使用动态表单组件:
<!-- 父组件 -->
<template><DynamicForm :fields="fields" @submit="onSubmit"><template v-slot:name="{ value }"><input v-model="value" type="text" placeholder="请输入姓名"></template><template v-slot:age="{ value }"><input v-model.number="value" type="number" placeholder="请输入年龄"></template><template v-slot:gender="{ value }"><select v-model="value"><option value="male">男</option><option value="female">女</option></select></template></DynamicForm>
</template><script>
import DynamicForm from './DynamicForm.vue';export default {components: {DynamicForm},data() {return {fields: [{ name: 'name', type: 'text', value: '' },{ name: 'age', type: 'number', value: null },{ name: 'gender', type: 'select', value: null, options: [{ label: '男', value: 'male' }, { label: '女', value: 'female' }] }]};},methods: {onSubmit(fields) {console.log(fields);}}
};
</script>
在这个例子中,DynamicForm组件会根据fields数组动态地生成表单元素,并将它们传递给父组件的插槽。父组件可以在插槽中自定义表单元素的样式和行为。
博客文章组件
使用作用域插槽可以很方便地创建一个博客文章组件,该组件可以根据传入的数据动态地生成文章元素。例如:
<!-- 子组件:BlogPost.vue -->
<template><article><header><h1>{{ post.title }}</h1><p>作者:{{ post.author }}</p></header><section><slot name="content" :post="post"></slot></section><footer><slot name="footer" :post="post"></slot></footer></article>
</template><script>
export default {name: 'BlogPost',props: {post: {type: Object,required: true}}
};
</script>
在父组件中,你可以这样使用博客文章组件:
<!-- 父组件 -->
<template><BlogPost :post="post"><template v-slot:content="{ post }"><p v-for="(paragraph, index) in post.paragraphs" :key="index">{{ paragraph }}</p></template><template v-slot:footer="{ post }"><p>发布日期:{{ post.publishDate }}</p><p>更新日期:{{ post.updateDate }}</p></template></BlogPost>
</template><script>
import BlogPost from './BlogPost.vue';export default {components: {BlogPost},data() {return {post: {title: '标题',author: '作者',paragraphs: ['内容1', '内容2'],publishDate: '2022-01-01',updateDate: '2022-01-02'}};}
};
</script>
在这个例子中,BlogPost组件会根据post对象动态地生成文章元素,并将它们传递给父组件的插槽。父组件可以在插槽中自定义文章元素的样式和行为。
AD:漫画首页
商品列表组件
使用作用域插槽可以很方便地创建一个商品列表组件,该组件可以根据传入的数据动态地生成商品元素。例如:
<!-- 子组件:ProductList.vue -->
<template><ul><li v-for="(product, index) in products" :key="index"><slot name="item" :product="product"></slot></li></ul>
</template><script>
export default {name: 'ProductList',props: {products: {type: Array,required: true}}
};
</script>
在父组件中,你可以这样使用商品列表组件:
<!-- 父组件 -->
<template><ProductList :products="products"><template v-slot:item="{ product }"><img :src="product.imageUrl" alt="商品图片"><h2>{{ product.name }}</h2><p>价格:{{ product.price }}</p></template></ProductList>
</template><script>
import ProductList from './ProductList.vue';export default {components: {ProductList},data() {return {products: [{name: '商品1',price: 100,imageUrl: 'https://example.com/product1.jpg'},{name: '商品2',price: 200,imageUrl: 'https://example.com/product2.jpg'}]};}
};
</script>
在这个例子中,ProductList组件会根据products数组动态地生成商品元素,并将它们传递给父组件的插槽。父组件可以在插槽中自定义商品元素的样式和行为。
第7章:作用域插槽的最佳实践
作用域插槽的设计模式
在使用作用域插槽时,可以采用以下几种设计模式:
- 单个插槽模式:在子组件中定义一个单一的插槽,并将所有需要传递给父组件的数据都放在该插槽的
scope属性中。
<!-- 子组件:SingleSlot.vue -->
<template><slot name="default" :data="data"></slot>
</template><script>
export default {data() {return {data: {title: '标题',content: '内容'}};}
};
</script>
<!-- 父组件 -->
<template><SingleSlot><template v-slot="{ data }"><h1>{{ data.title }}</h1><p>{{ data.content }}</p></template></SingleSlot>
</template>
- 多个插槽模式:在子组件中定义多个插槽,并将每个插槽的数据分别放在对应的
scope属性中。
<!-- 子组件:MultipleSlots.vue -->
<template><header><slot name="header" :title="title"></slot></header><main><slot name="content" :data="data"></slot></main><footer><slot name="footer" :footer="footer"></slot></footer>
</template><script>
export default {data() {return {title: '标题',data: {content: '内容'},footer: '页脚'};}
};
</script>
<!-- 父组件 -->
<template><MultipleSlots><template v-slot:header="{ title }"><h1>{{ title }}</h1></template><template v-slot:content="{ data }"><p>{{ data.content }}</p></template><template v-slot:footer="{ footer }"><p>{{ footer }}</p></template></MultipleSlots>
</template>
- 函数插槽模式:在父组件中定义一个函数插槽,并将子组件的数据作为参数传递给该函数。
<!-- 子组件:FunctionSlot.vue -->
<template><slot :data="data" :footer="footer"></slot>
</template><script>
export default {data() {return {data: {title: '标题',content: '内容'},footer: '页脚'};}
};
</script>
<!-- 父组件 -->
<template><FunctionSlot><template v-slot="{ data, footer }"><h1>{{ data.title }}</h1><p>{{ data.content }}</p><p>{{ footer }}</p></template></FunctionSlot>
</template>
作用域插槽的性能优化
在使用作用域插槽时,可以采用以下几种方式进行性能优化:
- 缓存渲染函数:可以使用
<template>元素的v-once指令来缓存渲染函数,以避免在每次渲染时都重新创建函数。
<template v-slot:default="slotProps"><p v-once>{{ slotProps.data }}</p>
</template>
- 使用
v-if和v-for的优先级规则:可以在v-if和v-for指令中使用作用域插槽,但需要注意它们的优先级规则。
<template v-slot:default="slotProps"><ul><li v-for="item in slotProps.items" v-if="item.visible">{{ item.name }}</li></ul>
</template>
- 使用
v-for的key属性:可以在使用v-for指令时为每个元素添加一个唯一的key属性,以提高渲染性能。
<template v-slot:default="slotProps"><ul><li v-for="(item, index) in slotProps.items" :key="index">{{ item.name }}</li></ul>
</template>
作用域插槽的安全性与兼容性
在使用作用域插槽时,需要注意以下几点:
- 避免不必要的数据暴露:在子组件中,只需要暴露必要的数据给父组件,避免暴露敏感信息或不必要的数据。
- 避免不必要的渲染函数:在父组件中,只需要渲染必要的内容,避免渲染不必要的内容。
- 兼容性考虑:在使用作用域插槽时,需要注意兼容性问题,可以使用
babel-plugin-transform-vue-slot-scope插件来转换作用域插槽的语法。 - 安全性考虑:在使用作用域插槽时,需要注意安全问题,可以使用
v-once指令来缓存渲染函数,避免潜在的安全风险。
第8章:自定义插槽组件
创建自定义插槽组件的方法
创建自定义插槽组件通常涉及以下几个步骤:
- 定义插槽:在自定义组件的模板中定义一个或多个
<slot>元素,这些<slot>元素将作为可被父组件填充的位置。
AD:专业搜索引擎 - 编写插槽内容:在组件的
<template>部分,你可以为每个插槽编写默认内容。这些内容将在没有父组件提供内容时显示。 - 传递插槽数据:使用
v-slot或slot-scope(在Vue 2中)指令将数据从子组件传递到插槽中。 - 使用插槽:在父组件中,使用
<component>标签并结合v-slot指令来使用自定义插槽组件,并传递所需的数据。
自定义插槽组件的示例
下面是一个简单的自定义插槽组件示例:
<!-- 自定义插槽组件 ExampleSlot.vue -->
<template><div class="example-slot"><slot name="default"></slot><slot name="footer"></slot></div>
</template><script>
export default {// 组件逻辑...
};
</script>
在父组件中使用该插槽组件:
<template><div><example-slot><template v-slot:default="slotProps"><h1>标题: {{ slotProps.title }}</h1><p>内容: {{ slotProps.content }}</p></template><template v-slot:footer="slotProps"><p>页脚: {{ slotProps.footerText }}</p></template></example-slot></div>
</template><script>
import ExampleSlot from './ExampleSlot.vue';export default {components: {ExampleSlot},data() {return {title: '我是标题',content: '我是内容',footerText: '我是页脚文本'};}
};
</script>
自定义插槽组件的最佳实践
- 明确插槽用途:确保插槽的用途清晰,避免在单个插槽中混合不同的数据或功能。
- 使用命名插槽:通过命名插槽,可以更明确地表示插槽的用途,并且可以同时在同一个组件中使用多个插槽。
- 避免全局插槽:尽量避免使用全局插槽,因为这可能会导致组件的复用性降低。
- 提供默认内容:为插槽提供默认内容,可以在父组件没有提供内容时提供一个占位符。
- 优化性能:如果插槽内容复杂,考虑使用
v-if或v-for指令来条件渲染内容,以提高性能。 - 使用
v-slot:使用v-slot指令来传递数据到插槽,而不是使用已经废弃的slot-scope。 - 文档说明:为你的自定义插槽组件提供清晰的文档,说明每个插槽的用途和接受的参数。
第9章:复杂组件中的插槽与作用域插槽
处理复杂组件中的插槽逻辑
在处理复杂组件中的插槽逻辑时,需要考虑以下几个关键点:
- 插槽的命名和组织:为插槽命名,并合理组织它们,以便于理解和使用。使用命名插槽可以避免逻辑混乱,并提高组件的可维护性。
- 作用域插槽的数据传递:使用作用域插槽来传递数据,使得父组件能够访问子组件的数据和方法。这通常通过在
<slot>
标签上绑定属性来实现。 - 插槽内容的条件渲染:根据不同的条件渲染不同的插槽内容。这可以通过
v-if、v-else-if和v-else或者v-show来实现。 - 插槽内容的循环渲染:如果插槽内容需要基于数组数据进行渲染,可以使用
v-for指令。 - 插槽内容的动态绑定:使用动态绑定来根据不同的上下文渲染不同的内容。
高级作用域插槽的使用技巧
高级作用域插槽的使用技巧包括:
- 使用解构赋值:在父组件中使用解构赋值来简化作用域插槽的代码,例如
<template v-slot="{ user }">。 - 传递复杂数据结构:在作用域插槽中传递复杂的数据结构,如对象或数组,并允许父组件访问这些数据。
- 传递方法:在作用域插槽中传递子组件的方法,使得父组件可以调用这些方法。
- 使用具名作用域插槽:为作用域插槽命名,以便在父组件中更精确地控制内容的渲染。
- 插槽的默认内容:为作用域插槽提供默认内容,当父组件没有提供内容时显示。
复杂组件的示例项目
假设我们正在构建一个复杂的列表组件,该组件可以显示不同类型的数据,并且允许用户自定义列表项的显示方式。
<!-- ComplexList.vue -->
<template><div class="complex-list"><slot v-for="item in items" :item="item" :key="item.id" name="item"></slot></div>
</template><script>
export default {props: {items: {type: Array,required: true}}
};
</script>
在父组件中使用这个复杂列表组件:
<!-- ParentComponent.vue -->
<template><div><complex-list :items="data"><template v-slot:item="{ item }"><div class="list-item"><h2>{{ item.title }}</h2><p>{{ item.description }}</p></div></template></complex-list></div>
</template><script>
import ComplexList from './ComplexList.vue';export default {components: {ComplexList},data() {return {data: [{ id: 1, title: 'Item 1', description: 'Description 1' },{ id: 2, title: 'Item 2', description: 'Description 2' },// ...]};}
};
</script>
在这个示例中,ComplexList组件接受一个items
数组作为属性,并为每个列表项提供了一个作用域插槽。父组件通过作用域插槽自定义了列表项的显示方式。AD:首页 | 一个覆盖广泛主题工具的高效在线平台
通过这种方式,我们可以构建出高度可定制和可复用的复杂组件,同时保持代码的清晰和可维护性。
附录:Vue.js插槽与作用域插槽API参考
-
插槽(Slot) :
<template slot="name">内容</template>:声明插槽<slot>或<slot name="name">默认内容</slot>:插入插槽内容v-slot:在Vue 2.6及更高版本中,使用更简洁的语法<template v-slot:name="props">内容</template>
-
作用域插槽(Scoped Slot) :
<template v-slot="{ prop1, prop2 }">...</template>:声明作用域插槽,可以接收父组件传递的属性<template v-slot:child slot-scope="{ item }">...</template>:插入作用域插槽,item是父组件传递的属性
-
API特性:
v-bind slot="name":绑定插槽v-bind slot-scope:绑定作用域插槽slot或slot-scope的name属性是可选的,如果没有提供,插槽将默认插入到第一个匹配的插槽位置。
资源与学习材料
- Vue.js官方文档:官方插槽和作用域插槽的详细指南
- Vue.js插槽教程:基础到高级的教程
- Vue Mastery - Vue.js 2.0 Slots:深入讲解插槽的课程
- Vue.js插槽实战:实战项目案例分析
常见问题解答
-
插槽和作用域插槽的区别是什么?
- 插槽是组件间内容传递的一种方式,而作用域插槽允许父组件更精确地控制子组件中插槽内容的渲染,可以接收和传递属性。
-
如何在父组件中使用插槽?
- 在模板中使用
<template slot="name">...</template>声明插槽,然后在父组件中使用<slot>
或<slot name="name">...</slot>插入插槽内容。
- 在模板中使用
-
如何传递数据到插槽?
- 可以使用
v-bind slot="name"或v-bind slot-scope绑定数据,或者通过slot-scope接收父组件传递的属性。
- 可以使用
-
插槽的默认内容如何设置?
- 使用
<slot>标签,不提供内容,则会插入默认内容(如果有)。
- 使用
-
如何处理插槽的动态内容?
- 可以使用条件渲染(
v-if、v-show)或循环渲染(v-for)来动态决定插入哪些内容。
- 可以使用条件渲染(
相关文章:
Vue插槽与作用域插槽
title: Vue插槽与作用域插槽 date: 2024/6/1 下午9:07:52 updated: 2024/6/1 下午9:07:52 categories: 前端开发 tags:VueSlotScopeSlot组件通信Vue2/3插槽作用域API动态插槽插槽优化 第1章:插槽的概念与原理 插槽的定义 在Vue.js中,插槽(…...
类和对象(一)(C++)
类和对象: 类的引入: C语言结构体中只能定义变量,在C中,结构体内不仅可以定义变量,也可以定义函数。比如: 之前在数据结构初阶中,用C语言方式实现的栈,结构体中只能定义变量&#…...
【免费Web系列】JavaWeb实战项目案例六
这是Web第一天的课程大家可以传送过去学习 http://t.csdnimg.cn/K547r 员工信息-删除&修改 前面我们已经实现了员工信息的条件分页查询以及新增操作。 关于员工管理的功能,还有两个需要实现: 删除员工 修改员工 除了员工管理的功能之外&#x…...
git分布式版本控制系统(四)
目前世界上最先进的分布式版本控制系统 官方网址:https://git-scm.com 学习目标: 1 了解 git 前世今生 2 掌握 git 基础概念、基础操作 3 各种 git 问题处理 4 互联网常用 gitflow(工作流程规范) 5 git 代码提交规范 6 git 分支管理及命名规范 常见问…...
【React篇】简述React-Router 的实现原理及工作方式
React Router 路由的基础实现原理分为两种,如果是切换 Hash 的方式,那么依靠浏览器 Hash 变化即可;如果是切换网址中的 Path,就要用到 HTML5 History API 中的 pushState、replaceState 等。在使用这个方式时,还需要在…...
Django里多app
在 Django 里的某一个项目,里面得包含很多 App (功能),那么如何在该项目里管理这么多App呢? 先说明下背景:未先创建 apps 文件夹来存各个app文件夹,直接在项目文件目录里创建各个app。为了便于管理,得将各…...
Prime1 - 信息收集和分析能力的试炼
主机发现 nmap扫描与分析 端口22、80 详细扫描;linux、ubuntu、 udp扫描 端口都是关闭的 脚本扫描 web渗透 打开只有一张图片;源码有图片和一个alt:hnp security不知道有啥用,先记录下来吧 继续web渗透思路走吧,目录…...
3.location的写法
location的写法 一、location的写法1、 精确匹配2、~ 以正则表达式匹配请求,区分大小写3、~* 以正则匹配请求,不区分大小写4、^~ 不以正则的方式匹配请求 二、stub_status模块显示工作状态三、url地址重写 rewrite模块1、语法2、针对项目结构有变化3、网…...
AndroidStudio设置允许APP获取定位权限
1. 在AndroidManifest.xml中声明权限 常用的定位权限有以下两种: <uses-permission android:name"android.permission.ACCESS_FINE_LOCATION"/> <uses-permission android:name"android.permission.ACCESS_COARSE_LOCATION"/>2. …...
Spring Boot 统一数据返回格式
在 Spring Boot 项目中,统一的数据格式返回是一种良好的实践,它提高了代码的可维护性和一致性,并改善了客户端与服务端之间的通信。本文将介绍如何在 Spring Boot 中实现统一的数据格式返回。 1 为什么需要统一数据返回格式 ⽅便前端程序员更…...
Android 项目Gradle文件讲解(Groovy和Kotlin)
Android 项目Gradle文件讲解(Groovy和Kotlin) 前言正文一、Gradle的作用二、Gradle的种类① 工程build.gradle② 项目build.gradle③ settings.gradle④ gradle.properties⑤ gradle-wrapper.properties⑥ local.properties 三、Groovy和Kotlin的语言对比…...
python-最接近target的值
【问题描述】:给定一个数组,在数组中找到两个数,使它们的和最接近目标值的值但不超过目标值,然后返回它们的和。 【问题示例】:输入target15,array[1,3,5,11,7],输出14,31114。 完整代码如下: …...
转换张量形状:`nlc_to_nchw` 函数详解
在深度学习和计算机视觉领域,张量的形状转换是一个常见的操作。本文将详细讲解一个用于形状转换的函数 nlc_to_nchw,它能够将形状为 [N, L, C] 的张量转换为 [N, C, H, W] 的张量。 函数定义 def nlc_to_nchw(x, hw_shape):"""Convert …...
「架构」云上自动化运维及其应用
随着云计算的普及,自动化运维成为企业提升运营效率和降低成本的关键。本文通过分析一家中型企业实施云上自动化运维(CloudOps)的案例,探讨了自动化监控、配置管理和持续集成/持续部署(CI/CD)三个核心模块的实际应用。文章详细阐述了每个模块的技术选型、实施原因、优缺点…...
分布式和集群的区别
分布式系统(Distributed System)和集群(Cluster)是两个经常被提及的计算机科学概念,它们在提高系统性能和可靠性方面都扮演着重要角色,很多同学会觉得这俩个是同一种东西,但事实上它们之间有着本…...
最新h5st(4.7.2)参数分析与纯算法还原(含算法源码)
文章目录 1. 写在前面2. 加密分析3. 算法还原 【🏠作者主页】:吴秋霖 【💼作者介绍】:擅长爬虫与JS加密逆向分析!Python领域优质创作者、CSDN博客专家、阿里云博客专家、华为云享专家。一路走来长期坚守并致力于Python…...
Spark大数据 掌握RDD的创建
在Apache Spark中,弹性分布式数据集(Resilient Distributed Dataset,简称RDD)是一个核心的数据结构,用于表示不可变、可分区、可并行操作的元素集合。理解并掌握RDD的创建是使用Spark进行大数据处理的关键步骤之一。 …...
Chrome谷歌浏览器如何打开不安全页面的禁止权限?
目录 一、背景二、如何打开不安全页面被禁止的权限?2.1 第一步,添加信任站点2.2 第二步,打开不安全页面的权限2.3 结果展示 一、背景 在开发过程中,由于测试环境没有配置 HTTPS 请求,所以谷歌浏览器的地址栏会有这样一…...
3D目标检测入门:探索OpenPCDet框架
前言 在自动驾驶和机器人视觉这两个飞速发展的领域中,3D目标检测技术扮演着核心角色。随着深度学习技术的突破性进展,3D目标检测算法的研究和应用正日益深入。OpenPCDet,这个由香港中文大学OpenMMLab实验室精心打造的开源工具箱,…...
JS异步编程
目录 概念定时器Promise对象概念 单线程模型指的是,JavaScript 只在一个线程上运行。也就是说,JavaScript 同时只能执行一个任务,其他任务都必须在后面排队等待。JavaScript 只在一个线程上运行,不代表 JavaScript 引擎只有一个线程。事实上,JavaScript 引擎有多个线程,…...
深入浅出Asp.Net Core MVC应用开发系列-AspNetCore中的日志记录
ASP.NET Core 是一个跨平台的开源框架,用于在 Windows、macOS 或 Linux 上生成基于云的新式 Web 应用。 ASP.NET Core 中的日志记录 .NET 通过 ILogger API 支持高性能结构化日志记录,以帮助监视应用程序行为和诊断问题。 可以通过配置不同的记录提供程…...
Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件
今天呢,博主的学习进度也是步入了Java Mybatis 框架,目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学,希望能对大家有所帮助,也特别欢迎大家指点不足之处,小生很乐意接受正确的建议&…...
Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)
参考官方文档:https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java(供 Kotlin 使用) 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...
SiFli 52把Imagie图片,Font字体资源放在指定位置,编译成指定img.bin和font.bin的问题
分区配置 (ptab.json) img 属性介绍: img 属性指定分区存放的 image 名称,指定的 image 名称必须是当前工程生成的 binary 。 如果 binary 有多个文件,则以 proj_name:binary_name 格式指定文件名, proj_name 为工程 名&…...
Qemu arm操作系统开发环境
使用qemu虚拟arm硬件比较合适。 步骤如下: 安装qemu apt install qemu-system安装aarch64-none-elf-gcc 需要手动下载,下载地址:https://developer.arm.com/-/media/Files/downloads/gnu/13.2.rel1/binrel/arm-gnu-toolchain-13.2.rel1-x…...
C语言中提供的第三方库之哈希表实现
一. 简介 前面一篇文章简单学习了C语言中第三方库(uthash库)提供对哈希表的操作,文章如下: C语言中提供的第三方库uthash常用接口-CSDN博客 本文简单学习一下第三方库 uthash库对哈希表的操作。 二. uthash库哈希表操作示例 u…...
掌握 HTTP 请求:理解 cURL GET 语法
cURL 是一个强大的命令行工具,用于发送 HTTP 请求和与 Web 服务器交互。在 Web 开发和测试中,cURL 经常用于发送 GET 请求来获取服务器资源。本文将详细介绍 cURL GET 请求的语法和使用方法。 一、cURL 基本概念 cURL 是 "Client URL" 的缩写…...
零知开源——STM32F103RBT6驱动 ICM20948 九轴传感器及 vofa + 上位机可视化教程
STM32F1 本教程使用零知标准板(STM32F103RBT6)通过I2C驱动ICM20948九轴传感器,实现姿态解算,并通过串口将数据实时发送至VOFA上位机进行3D可视化。代码基于开源库修改优化,适合嵌入式及物联网开发者。在基础驱动上新增…...
Python 训练营打卡 Day 47
注意力热力图可视化 在day 46代码的基础上,对比不同卷积层热力图可视化的结果 import torch import torch.nn as nn import torch.optim as optim from torchvision import datasets, transforms from torch.utils.data import DataLoader import matplotlib.pypl…...
【深度学习新浪潮】什么是credit assignment problem?
Credit Assignment Problem(信用分配问题) 是机器学习,尤其是强化学习(RL)中的核心挑战之一,指的是如何将最终的奖励或惩罚准确地分配给导致该结果的各个中间动作或决策。在序列决策任务中,智能体执行一系列动作后获得一个最终奖励,但每个动作对最终结果的贡献程度往往…...
