关于前端指令

在前端开发中,指令(Directives)通常指在框架中使用的一种特殊的语法或机制,用于扩展 HTML 的功能。常见的指令主要存在于前端框架中,如 Vue.js、Angular 等。下面我们将分别介绍 Vue.js 和 Angular 中的常用指令,并通过代码示例分析它们的作用。最后再讨论自定义指令的常见使用场景。


Vue.js 常用指令

Vue.js 提供了多种内置指令,用于与 DOM 元素交互。以下是常用指令及示例:

1. v-bind

用于动态绑定 HTML 元素的属性。

<img v-bind:src="imageUrl">

作用:将 imageUrl 数据绑定到 img 标签的 src 属性,当 imageUrl 变化时,src 会自动更新。

2. v-model

用于实现表单元素的双向数据绑定。

<input v-model="username">

作用:将 input 的值与 username 数据绑定,当用户输入时,username 会自动更新,反之亦然。

3. v-if / v-else / v-else-if

用于条件渲染。

<p v-if="score >= 90">优秀</p>
<p v-else-if="score >= 60">及格</p>
<p v-else>不及格</p>

作用:根据 score 的值动态渲染不同的内容。

4. v-for

用于列表渲染。

<ul><li v-for="item in items" :key="item.id">{{ item.name }}</li>
</ul>

作用:遍历 items 数组,渲染列表。

5. v-on

用于绑定事件监听器。

<button v-on:click="handleClick">点击我</button>

作用:为按钮绑定 click 事件,触发 handleClick 方法。

6. v-show

用于控制元素的显示/隐藏。

<div v-show="isVisible">显示内容</div>

作用:根据 isVisible 的值控制 div 的显示或隐藏(通过 display: none)。

Angular 常用指令

Angular 也提供了丰富的内置指令,以下是一些常见的:

1. *ngIf

用于条件渲染。

<p *ngIf="isLoggedIn">欢迎回来!</p>

作用:根据 isLoggedIn 的值决定是否渲染 p 元素。

2. *ngFor

用于列表渲染。

<ul><li *ngFor="let item of items">{{ item.name }}</li>
</ul>

作用:遍历 items 数组,渲染列表。

3. ngModel

用于双向数据绑定。

<input [(ngModel)]="username">

作用:将 input 的值与 username 双向绑定。

4. ngClass

用于动态添加 CSS 类。

<div [ngClass]="{'active': isActive, 'error': hasError}">内容</div>

作用:根据 isActivehasError 的值动态添加 activeerror 类。

5. ngStyle

用于动态添加样式。

<div [ngStyle]="{'color': textColor, 'font-size': fontSize}">内容</div>

作用:根据 textColorfontSize 的值动态设置样式。


常用自定义指令

在 Vue.js 和 Angular 中,可以创建自定义指令来扩展功能。

Vue.js 自定义指令
Vue.directive('focus', {inserted: function (el) {el.focus();}
});

使用:

<input v-focus>

作用:当页面加载时,自动聚焦到 input 元素。

Vue.js 提供了 Vue.directive 方法来创建自定义指令。以下是一个监听点击事件并执行自定义逻辑的示例

Vue.directive('custom-click', {bind(el, binding) {// 绑定点击事件el.addEventListener('click', () => {// 执行传入的回调函数if (typeof binding.value === 'function') {binding.value();}});},unbind(el) {// 解绑点击事件el.removeEventListener('click', () => {});}
});
使用示例
<div id="app"><button v-custom-click="handleClick">点击我</button>
</div><script>
new Vue({el: '#app',methods: {handleClick() {alert('按钮被点击了!');}}
});
</script>
解析
  1. bind:在指令绑定到元素时调用。这里我们监听了 click 事件。
  2. unbind:在指令解绑时调用。这里我们移除了 click 事件监听器。
  3. binding.value:获取指令绑定的值(即回调函数 handleClick)。
  4. 当按钮被点击时,会触发传入的回调函数 handleClick
Angular 自定义指令
@Directive({selector: '[appHighlight]'
})
export class HighlightDirective {constructor(private el: ElementRef) {}@HostListener('mouseenter') onMouseEnter() {this.el.nativeElement.style.backgroundColor = 'yellow';}@HostListener('mouseleave') onMouseLeave() {this.el.nativeElement.style.backgroundColor = '';}
}

使用:

<p appHighlight>鼠标悬停我</p>

作用:当鼠标悬停在 p 元素上时,背景色变为黄色。

自定义指令的常见应用

  1. DOM 操作

    • 场景:自动聚焦、滚动到某个元素、动态修改元素属性等。
    • 示例:v-focus 指令用于输入框自动聚焦。
  2. 事件处理

    • 场景:监听特定事件,执行自定义逻辑。             
    • 示例:appHighlight 指令用于鼠标悬停时动态改变背景色。
  3. 表单验证

    • 场景:自定义表单验证逻辑。
    • 示例:创建 v-validate 指令,实时验证输入内容是否符合规则。
  4. 第三方库集成

    • 场景:将第三方库(如 jQuery 插件)集成到框架中。
    • 示例:创建 v-datepicker 指令,将日期选择器插件与 Vue 或 Angular 集成。
  5. 性能优化

    • 场景:懒加载图片、组件或数据。
    • 示例:创建 v-lazy 指令,实现图片懒加载。

全局注册指令

全局注册的指令可以在应用的任何组件中使用。通常在 main.jsapp.js 中注册。

代码实现

import Vue from 'vue';// 定义自定义指令
Vue.directive('focus', {inserted(el) {el.focus();}
});// 或者使用简写形式
Vue.directive('focus', (el) => {el.focus();
});// 启动应用
new Vue({el: '#app'
});

使用示例

<input v-focus>

局部注册指令

局部注册的指令仅在特定组件中可用。

代码实现

export default {directives: {focus: {inserted(el) {el.focus();}}}
};

使用示例

<template><input v-focus>
</template><script>
export default {directives: {focus: {inserted(el) {el.focus();}}}
};
</script>

复杂指令的应用场景

1. DOM 操作和交互

场景:实现拖拽、缩放、滚动监听等复杂交互。

  • 示例
    • Vue.js 实现拖拽指令:
      • Vue.directive('drag', {bind(el) {let isDragging = false;el.addEventListener('mousedown', () => {isDragging = true;});document.addEventListener('mousemove', (e) => {if (isDragging) {el.style.left = `${e.clientX}px`;el.style.top = `${e.clientY}px`;}});document.addEventListener('mouseup', () => {isDragging = false;});}
        });
        
    • Angular 实现拖拽指令:​​​​​
      • @Directive({selector: '[appDrag]'
        })
        export class DragDirective {@HostListener('mousedown', ['$event']) onMouseDown(event: MouseEvent) {const element = this.el.nativeElement;const startX = event.clientX - element.offsetLeft;const startY = event.clientY - element.offsetTop;const onMouseMove = (e: MouseEvent) => {element.style.left = `${e.clientX - startX}px`;element.style.top = `${e.clientY - startY}px`;};const onMouseUp = () => {document.removeEventListener('mousemove', onMouseMove);document.removeEventListener('mouseup', onMouseUp);};document.addEventListener('mousemove', onMouseMove);document.addEventListener('mouseup', onMouseUp);}constructor(private el: ElementRef) {}
        }
        
2. 事件绑定和解绑

场景:监听复杂的用户交互事件(如长按、双击、滚动等)。

  • 示例
    • Vue.js 长按指令:
      • Vue.directive('longpress', {bind(el, binding) {let timeout;const handler = () => binding.value();el.addEventListener('mousedown', () => {timeout = setTimeout(handler, 1000);});el.addEventListener('mouseup', () => {clearTimeout(timeout);});}
        });
        
    • Angular 实现拖拽指令: 
      • @Directive({selector: '[appLongPress]'
        })
        export class LongPressDirective {@HostListener('mousedown') onMouseDown() {this.timeout = setTimeout(() => {this.callback();}, 1000);}@HostListener('mouseup') onMouseUp() {clearTimeout(this.timeout);}private timeout: any;@Input('appLongPress') callback: () => void;
        }
        

3. 动态样式和类名

场景:根据条件动态添加样式或类名。

  • 示例
Vue.directive('dynamic-class', {update(el, binding) {el.className = binding.value;}
});
4. 性能优化

场景:懒加载图片、虚拟滚动等。

  • 示例
    • Vue.js 图片懒加载指令:
      • Vue.directive('lazy', {inserted(el, binding) {const observer = new IntersectionObserver((entries) => {if (entries[0].isIntersecting) {el.src = binding.value;observer.unobserve(el);}});observer.observe(el);}
        });
        

 

设计复杂指令的注意事项

1. 指令的生命周期
  • Vue.js:需要了解 bindinsertedupdateunbind 等钩子函数。
  • Angular:需要了解 ngOnInitngAfterViewInitngOnDestroy 等生命周期钩子。
  • 注意:在指令解绑时(如 unbind 或 ngOnDestroy),一定要清除事件监听器、定时器或观察者,避免内存泄漏。
2. 指令的复用性
  • 设计指令时应尽量保持通用性,避免硬编码。
  • 通过参数化(如 binding.value 或 @Input)来实现灵活配置。
3. 性能优化
  • 避免在指令中进行频繁的 DOM 操作,减少重绘和回流。
  • 使用事件委托或防抖/节流技术优化高频事件(如滚动、拖拽)。
4. 指令的测试
  • 编写单元测试或端到端测试,确保指令在不同场景下的行为正确。
  • 使用工具(如 Jest、Karma)进行测试。
5. 指令的兼容性
  • 确保指令在不同浏览器和设备上的兼容性。
  • 对旧版浏览器使用 polyfill 或降级方案。

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

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

相关文章

虚拟地址空间(下)进程地址空间(上)

一.关于页表组成 1.权限&#xff08;rwx) 作用&#xff1a;如1.让代码区变成只读的 2.写时拷贝的实现&#xff1a;子进程创建时其页表指向的父进程代码和数据权限都是只读的&#xff0c;子进程试图修改&#xff0c;触发错误&#xff0c;系统开始写时拷贝。 来源&#xff1a;…

【区块链 + 航运物流】丰溯 - 区块链溯源平台 | FISCO BCOS 应用案例

丰溯是顺丰科技推出的区块链溯源平台&#xff0c; 采用 FISCO BCOS 底层开源框架&#xff0c; 为农副食品、 冷链生鲜等企业客户及消费 者提供关键流通节点的溯源信息服务&#xff0c;形成从源头到消费者端全链路透明的信息链。 在商贸消费领域&#xff0c; 溯源一直是保障产品…

iwebsec-SQL数字型注入

1.判断是否存在漏洞 添加and 11发现正常显示&#xff0c;添加and 12无回显条目&#xff0c;则存在sql注入漏洞 2.因为有回显&#xff0c;尝试union联合注入&#xff0c;使用order by判断出有3个字段 3.使用union联合注入查看回显位&#xff0c;发现3三个字段均有回显&#xff…

蓝桥杯每日五题第一日

蓝桥杯每日5题 问题一 班级活动 1.班级活动 - 蓝桥云课 问题描述 小明的老师准备组织一次班级活动。班上一共有 nn 名 (nn 为偶数) 同学&#xff0c;老师想把所有的同学进行分组&#xff0c;每两名同学一组。为了公平&#xff0c;老师给每名同学随机分配了一个 nn 以内的正…

STM32 —— 嵌入式系统、通用计算机系统、物联网三层架构

目录 一、嵌入式系统的概念 二、通用计算机系统与嵌入式系统的比较 用途 硬件 软件 性能与功耗 开发与维护 三、嵌入式系统与物联网的关系 四、物联网的三层架构 1. 感知层&#xff08;Perception Layer&#xff09; 2. 网络层&#xff08;Network Layer&#xff09; …

卡码网25题——掌握ACM输入输出方式(15 至 18)

刷题小记&#xff1a; 本期涉及ACM模式下栈和链表的构建与使用&#xff0c;值得学习。 卡玛网15.神秘字符&#xff08;卡玛网15.神秘字符&#xff09; 题目分析&#xff1a; 若给定2行字符串&#xff0c;其中第一个串的长度为偶数&#xff0c;现要求把第二个串插入到第一个…

前端字段名和后端不一致?解锁 JSON 映射的“隐藏规则” !!!

&#x1f680; 前端字段名和后端不一致&#xff1f;解锁 JSON 映射的“隐藏规则” &#x1f31f; 嘿&#xff0c;技术冒险家们&#xff01;&#x1f44b; 今天我们要聊一个开发中常见的“坑”&#xff1a;前端传来的 JSON 参数字段名和后端对象字段名不一致&#xff0c;会发生…

python中使用单例模式在整个程序中只创建一个数据库连接,节省资源

示例代码&#xff1a; from loguru import logger from pymongo import MongoClient from pymongo.errors import ConnectionFailurefrom llm_engineering.settings import settingsclass MongoDatabaseConnector:_instance: MongoClient | None Nonedef __new__(cls, *args,…

AI小白的第六天:必要的数学知识(一)

在学习的过程中&#xff0c;不管是上代码还是理论学习&#xff0c;其中都掺杂了一些数学知识。俗话说“磨刀不误砍柴工”&#xff0c;而我已经“误了砍柴功”了&#xff0c;现在变成了“亡羊补牢&#xff0c;为时不晚”。 线性代数 线性代数是数学的一个分支&#xff0c;主要…

【Linux】Bash是什么?怎么使用?

李升伟 整理 什么是 Bash&#xff1f; Bash&#xff08;Bourne Again Shell&#xff09;是一种 命令行解释器&#xff08;Shell&#xff09;&#xff0c;广泛用于 Unix 和 Linux 操作系统。它是 Bourne Shell&#xff08;sh&#xff09; 的增强版&#xff0c;提供了更多的功能…

Qt Creator入门

1.创建项目 选择创建项目-Application&#xff08;Qt&#xff09;-Qt Widgets Application-修改名称即可 默认创建有窗口类&#xff0c;myWidget,基类有三种选择&#xff1a;QWidget&#xff0c;QMainWindow&#xff0c;QDialog 注意&#xff1a; 名称和创建路径不能有中文、…

C语言经典代码练习题

1.输入一个4位数&#xff1a;输出这个输的个位 十位 百位 千位 #include <stdio.h> int main(int argc, char const *argv[]) {int a;printf("输入一个&#xff14;位数&#xff1a;");scanf("%d",&a);printf("个位&#xff1a;%d\n"…

cls(**dict(data, id=id))灵活地从一个字典生成实例,同时确保某些关键字段(如 id)被正确设置或覆盖

示例代码&#xff1a; classmethoddef from_mongo(cls: Type[T], data: dict) -> T:"""Convert "_id" (str object) into "id" (UUID object)."""if not data:raise ValueError("Data is empty.")id data.pop…

MyBatis XMLMapperBuilder 是如何将 SQL 语句解析成可执行的对象? 如何将结果映射规则解析成对应的处理器?

1. XMLMapperBuilder 如何将 SQL 语句解析成可执行对象 (MappedStatement): XMLMapperBuilder 解析 <select>, <insert>, <update>, <delete> 等 SQL 语句元素时&#xff0c;并不仅仅是简单地读取 SQL 文本&#xff0c;而是要将 SQL 语句和相关的配置…

咖啡点单小程序毕业设计(JAVA+SpringBoot+微信小程序+完整源码+论文)

✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取项目下载方式&#x1f345; 一、项目背景介绍&#xff1a; 随着社会的快速发展和…

003-掌控命令行-CLI11-C++开源库108杰

首选的现代C风格命令行参数解析器! &#xff08;本课程包含两段教学视频。&#xff09; 以文件对象监控程序为实例&#xff0c;五分钟实现从命令行读入多个监控目标路径&#xff1b;区分两大时机&#xff0c;学习 CLI11 构建与解析参数两大场景下的异常处理&#xff1b;区分三…

【leetcode hot 100 124】二叉树中的最大路径和

解法一&#xff1a;&#xff08;递归&#xff09;考虑实现一个简化的函数 maxGain(node)&#xff0c;该函数计算二叉树中的一个节点的最大贡献值&#xff0c;具体而言&#xff0c;就是在以该节点为根节点的子树中寻找以该节点为起点的一条路径&#xff0c;使得该路径上的节点值…

谱分析方法

前言 本文隶属于专栏《机器学习数学通关指南》&#xff0c;该专栏为笔者原创&#xff0c;引用请注明来源&#xff0c;不足和错误之处请在评论区帮忙指出&#xff0c;谢谢&#xff01; 本专栏目录结构和参考文献请见《机器学习数学通关指南》 ima 知识库 知识库广场搜索&#…

在图像/视频中裁剪出人脸区域

1. 在图像中裁剪人脸区域 import face_alignment import skimage.io import numpy from argparse import ArgumentParser from skimage import img_as_ubyte from skimage.transform import resize from tqdm import tqdm import os import numpy as np import warnings warni…

【软考-架构】11.3、设计模式-新

✨资料&文章更新✨ GitHub地址&#xff1a;https://github.com/tyronczt/system_architect 文章目录 项目中的应用设计模式创建型设计模式结构型设计模式行为型设计模式 &#x1f4af;考试真题题外话 项目中的应用 在实际项目中&#xff0c;我应用过多种设计模式来解决不同…