首页 web前端 js教程 使用 Javascript 从头开始​​构建反应式商店

使用 Javascript 从头开始​​构建反应式商店

Nov 04, 2024 am 11:32 AM

反应式编程是一种巧妙的方法,允许您创建动态反映数据更改的应用程序。它是 React 和 Vue 等许多现代 JavaScript 框架背后的核心技术 - 它会根据用户操作或其他状态变化进行更新。理解反应性背后的内容可能感觉工作量太大,感觉就像框架所针对的“神奇”抽象之一。但是,如果您可以自己构建一个小型反应式系统来看看它是如何工作的呢?

本文将通过使用 JavaScript 从头开始​​构建一个简单的反应式存储来介绍反应式编程的基础知识。我们将以最小的实现方式介绍关键概念,包括依赖项跟踪和自动更新。最后,您应该能够了解如何创建反应式数据结构,以便在状态发生变化时自动跟踪依赖关系并触发更新。这种方法将帮助您理解反应性,并为您提供自行实验的工具,并可能将其应用到您的项目中。

让我们开始看看我们将要使用的反应式系统的核心组件:

  • 效果:自动运行以响应反应数据变化的函数。它们是使用效果函数注册的,该函数跟踪对任何访问信号的依赖关系。
  • 信号:反应性数据属性,每当其值发生变化时通知依赖效应。
  • 依赖性:信号和依赖于它们的效果之间的关系。跟踪依赖关系,以便信号的变化触发相关效果的更新。

现在我们已经了解了响应式编程定义,让我们也提一下我们将要使用的 Javascript API:

Proxy:Proxy 对象允许您为另一个对象创建代理,使您能够为基本操作(如属性访问和分配)定义自定义行为。在此代码中,它用于使反应式存储(状态对象)响应更改。

Reflect:Reflect API 提供可拦截 JavaScript 操作的方法。它用于在响应式函数中执行诸如 Reflect.get 和 Reflect.set 之类的操作,从而允许代理处理属性访问和赋值,同时保持对象的原始行为。

Map:Map 对象是一个保存键值对的集合,其中键可以是任何数据类型。在此实现中,它用于创建 dependencyMap,该依赖关系跟踪与每个信号关联的依赖关系。

现在,让我们开始定义我们的初始状态:

// Let's define a Map object to track our dependencies
const dependencyTrackerMap = new Map();
// The activeEffect variable will hold the currently executing 
// effect function. 
// It will be set when an effect is run and will be used
// to track which effects depend on specific reactive properties. 
let activeEffect = null

// This function will make an object reactive
function reactive(target) {
    return new Proxy(target, {
        get(obj, prop) {
            trackDependency(prop); // Track dependency
            return Reflect.get(obj, prop);
        },
        set(obj, prop, value) {
            const result = Reflect.set(obj, prop, value);
            triggerDependency(prop); // Trigger reactions
            return result;
        }
    });
}

// the effect function will register reactive functions
function effect(fn) {
    activeEffect = fn;
    fn(); // Run the function once to register dependencies
    activeEffect = null;
}

// this function will track dependencies
function trackDependency(key) {
    if (activeEffect) {
        if (!dependencyTrackerMap.has(key)) {
            dependencyTrackerMap.set(key, new Set());
        }
        dependencyTrackerMap.get(key).add(activeEffect);
    }
}

// this function will trigger dependencies
function triggerDependency(key) {
    const deps = dependencyTrackerMap.get(key);
    if (deps) {
        deps.forEach(effect => effect());
    }
}

// This will create a reactive object with an initial state
// count and message here are signals
const state = reactive({ count: 0, message: "Hello" });
登录后复制
登录后复制

所以,这就是我们所做的:

  1. 我们创建了一个响应式函数,它接受一个对象并返回一个代理,该代理将跟踪将来对该对象进行的更改
  2. 我们创建了一个效果函数,它注册依赖于状态的反应函数。定义效果后,它会立即运行以注册任何依赖项并相应地设置 activeEffect。
  3. 我们创建了一个依赖跟踪器,它由三部分组成:trackDependency 函数检查当访问响应式属性时是否有主动效果。如果是,它会将该效果添加到 dependencyTrackerMap 中相应属性的一组依赖项中。反过来,triggerDependency 函数从 dependencyTrackerMap 中检索与属性关联的依赖效果集,并在属性值更改时执行每个效果。

现在,让我们创建一个带有回调的效果并尝试触发它:

// Let's define a Map object to track our dependencies
const dependencyTrackerMap = new Map();
// The activeEffect variable will hold the currently executing 
// effect function. 
// It will be set when an effect is run and will be used
// to track which effects depend on specific reactive properties. 
let activeEffect = null

// This function will make an object reactive
function reactive(target) {
    return new Proxy(target, {
        get(obj, prop) {
            trackDependency(prop); // Track dependency
            return Reflect.get(obj, prop);
        },
        set(obj, prop, value) {
            const result = Reflect.set(obj, prop, value);
            triggerDependency(prop); // Trigger reactions
            return result;
        }
    });
}

// the effect function will register reactive functions
function effect(fn) {
    activeEffect = fn;
    fn(); // Run the function once to register dependencies
    activeEffect = null;
}

// this function will track dependencies
function trackDependency(key) {
    if (activeEffect) {
        if (!dependencyTrackerMap.has(key)) {
            dependencyTrackerMap.set(key, new Set());
        }
        dependencyTrackerMap.get(key).add(activeEffect);
    }
}

// this function will trigger dependencies
function triggerDependency(key) {
    const deps = dependencyTrackerMap.get(key);
    if (deps) {
        deps.forEach(effect => effect());
    }
}

// This will create a reactive object with an initial state
// count and message here are signals
const state = reactive({ count: 0, message: "Hello" });
登录后复制
登录后复制

当我们尝试更新我们创建的状态时,控制台日志将触发:

//We are using state from the previous snippet:
effect(() => {
    console.log(Count has changed: ${state.count});
});

effect(() => {
    console.log("Message has changed");
    console.log(The new message is: ${state.message});
});
登录后复制

以下是触发依赖项时发生的情况的一些可视化:

Build a reactive store from scratch using Javascript


在本文中,我们探讨了如何在 JavaScript 中创建基本的反应式系统,从而实现自动更新(或副作用)以响应数据的变化。此实现作为反应式编程概念的介绍,它是框架“魔法”的一部分。此外,我们还了解了 Proxy 和 Reflect API 的作用并使用了它们,以及 Map 对象。

总之,这个反应式系统管理依赖关系并在状态变化时自动更新效果。通过注册依赖于特定反应属性的函数,系统可以跟踪哪些函数依赖于哪些属性,并在需要时重新运行它们。这种方法允许创建响应式应用程序,其中状态更改会自动反映在 UI 中,无需额外代码,从而改善开发人员体验并使数据处理更轻松、更高效。

以上是使用 Javascript 从头开始​​构建反应式商店的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

<🎜>:泡泡胶模拟器无穷大 - 如何获取和使用皇家钥匙
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系统,解释
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆树的耳语 - 如何解锁抓钩
3 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

热门话题

Java教程
1673
14
CakePHP 教程
1428
52
Laravel 教程
1333
25
PHP教程
1277
29
C# 教程
1257
24
Python vs. JavaScript:学习曲线和易用性 Python vs. JavaScript:学习曲线和易用性 Apr 16, 2025 am 12:12 AM

Python更适合初学者,学习曲线平缓,语法简洁;JavaScript适合前端开发,学习曲线较陡,语法灵活。1.Python语法直观,适用于数据科学和后端开发。2.JavaScript灵活,广泛用于前端和服务器端编程。

JavaScript和Web:核心功能和用例 JavaScript和Web:核心功能和用例 Apr 18, 2025 am 12:19 AM

JavaScript在Web开发中的主要用途包括客户端交互、表单验证和异步通信。1)通过DOM操作实现动态内容更新和用户交互;2)在用户提交数据前进行客户端验证,提高用户体验;3)通过AJAX技术实现与服务器的无刷新通信。

JavaScript在行动中:现实世界中的示例和项目 JavaScript在行动中:现实世界中的示例和项目 Apr 19, 2025 am 12:13 AM

JavaScript在现实世界中的应用包括前端和后端开发。1)通过构建TODO列表应用展示前端应用,涉及DOM操作和事件处理。2)通过Node.js和Express构建RESTfulAPI展示后端应用。

了解JavaScript引擎:实施详细信息 了解JavaScript引擎:实施详细信息 Apr 17, 2025 am 12:05 AM

理解JavaScript引擎内部工作原理对开发者重要,因为它能帮助编写更高效的代码并理解性能瓶颈和优化策略。1)引擎的工作流程包括解析、编译和执行三个阶段;2)执行过程中,引擎会进行动态优化,如内联缓存和隐藏类;3)最佳实践包括避免全局变量、优化循环、使用const和let,以及避免过度使用闭包。

Python vs. JavaScript:社区,图书馆和资源 Python vs. JavaScript:社区,图书馆和资源 Apr 15, 2025 am 12:16 AM

Python和JavaScript在社区、库和资源方面的对比各有优劣。1)Python社区友好,适合初学者,但前端开发资源不如JavaScript丰富。2)Python在数据科学和机器学习库方面强大,JavaScript则在前端开发库和框架上更胜一筹。3)两者的学习资源都丰富,但Python适合从官方文档开始,JavaScript则以MDNWebDocs为佳。选择应基于项目需求和个人兴趣。

Python vs. JavaScript:开发环境和工具 Python vs. JavaScript:开发环境和工具 Apr 26, 2025 am 12:09 AM

Python和JavaScript在开发环境上的选择都很重要。1)Python的开发环境包括PyCharm、JupyterNotebook和Anaconda,适合数据科学和快速原型开发。2)JavaScript的开发环境包括Node.js、VSCode和Webpack,适用于前端和后端开发。根据项目需求选择合适的工具可以提高开发效率和项目成功率。

C/C在JavaScript口译员和编译器中的作用 C/C在JavaScript口译员和编译器中的作用 Apr 20, 2025 am 12:01 AM

C和C 在JavaScript引擎中扮演了至关重要的角色,主要用于实现解释器和JIT编译器。 1)C 用于解析JavaScript源码并生成抽象语法树。 2)C 负责生成和执行字节码。 3)C 实现JIT编译器,在运行时优化和编译热点代码,显着提高JavaScript的执行效率。

从网站到应用程序:JavaScript的不同应用 从网站到应用程序:JavaScript的不同应用 Apr 22, 2025 am 12:02 AM

JavaScript在网站、移动应用、桌面应用和服务器端编程中均有广泛应用。1)在网站开发中,JavaScript与HTML、CSS一起操作DOM,实现动态效果,并支持如jQuery、React等框架。2)通过ReactNative和Ionic,JavaScript用于开发跨平台移动应用。3)Electron框架使JavaScript能构建桌面应用。4)Node.js让JavaScript在服务器端运行,支持高并发请求。

See all articles