【学习笔记70】数据劫持

news2025/1/21 11:57:18

一、 数据驱动视图

  • 多次渲染页面,多的时候,比较麻烦和繁琐
        const box = document.querySelector('.box')
        const obj = {
            name: 'QF666',
            age: 18
        }
        box.innerHTML = `名字: ${obj.name};    年龄: ${obj.age}`;

        obj.age = 99;
        box.innerHTML = `名字: ${obj.name};    年龄: ${obj.age}`;


        obj.name = 'QF999';
        box.innerHTML = `名字: ${obj.name};    年龄: ${obj.age}`;
        console.log(obj);

二、数据劫持

  • 将原始数据,劫持出一份一摸一样, 听起来有点像浅拷贝
  • 劫持出来的数据, 默认是不可以修改的
语法: Object.defineProperty(那个对象, '对象的key', {配置项})
    配置项:
    1. value        访问这个值 之后, 得到结果
    2. writable     决定当前这个属性能否被修改, 默认是 false
    3. enumerable   决定当前这个属性能否被枚举, 决定当前这个属性能否被遍历到
    4. getter       是一个函数, 是一个获取器, 当访问这个属性时, 会执行这个函数
                    getter不能和value、writable一起使用

    5. setter      是一个函数, 是一个设置器, 当设置这个属性是, 会执行这个函数

1、初级版 

        const obj = {}
        obj.name = 'QF666'
        console.log(obj)

        // Object.defineProperty(obj, str, {配置项})
        Object.defineProperty(obj, 'age', {
            // value: 'QF999',
            // writable: true,
            enumerable: true,
            get() {
                // console.log('你当前访问了这个age属性, 触发了get函数')
                return 'qwerty'
            },
            set(val) {
                console.log('你当前想要修改这个age属性, 修改的值是: ', val)
            }
        })
        obj.age = 999
        console.log(obj.age)

2、升级版

    <div id="box"></div>
    <script>
        // 实现数据劫持
        const box = document.querySelector('#box')
        const obj = {
            name: 'QF666',
            age: 18
        }
        console.log('原始对象obj:', obj)
        const res = {}

        Object.defineProperty(res, 'name', {
            get() {
                return obj.name
            },
            set(val) {
                // console.log('你想要修改这个属性的值, 新值为: ', val)
                obj.name = val
                box.innerHTML = `名字: ${res.name}; 年龄: ${res.age}`
            }
        })

        Object.defineProperty(res, 'age', {
            get() {
                return obj.age
            },
            set(val) {
                // console.log('你想要修改这个属性的值, 新值为: ', val)
                obj.age = val
                box.innerHTML = `名字: ${res.name}; 年龄: ${res.age}`
            }
        })
        box.innerHTML = `名字: ${res.name}; 年龄: ${res.age}`
        res.age = 100;
    </script>

 三、封装数据劫持

    <input type="text" name="" id="inp">
    <div id="box"></div>

函数的封装 :

        const box = document.querySelector('#box')
        const obj = {
            name: 'QF666',
            age: 18
        }
        function observer(origin, callback) {
            const target = {}

            for (let k in origin) {
                Object.defineProperty(target, k, {
                    get() {
                        return origin[k]
                    },
                    set(val) {
                        // console.log('你现在想要修改的key是' , k, '修改的值为', val)
                        origin[k] = val
                        callback(target)
                    }
                })
            }
            callback(target)
            return target
        }

调用的 

        function fn(res) {
            box.innerHTML = `名字: ${res.name}; 年龄: ${res.age}`
        }

        const app = observer(obj, fn)


        document.querySelector('#inp').oninput = function (e) {
            app.age = e.target.value
        }

四、封装数据劫持+渲染

  • 模拟一个其他人写的代码, 框架的源码
  • 把需要渲染的代码, 都放到HTML 中, 然后由我这段代码帮助我们去渲染
    <div id="app">
        <h1> {{msg}} </h1>
        <h1> {{ name }} </h1>
        <h1> {{ age }} </h1>
        <h1> {{ abc }} </h1>
    </div>

封装的

    <script>
        function observer(options) {
            // 1. 验证root有没有传递
            if (options.root === undefined) {
                // console.log('您没有传递root属性, 请重新传递')
                // 手动返回一个错误
                throw new Error('您没有传递root属性, 请重新传递') 
            }

            // 2. 验证root能否正确获取到节点
            const rootHtml = document.querySelector(options.root)
            if (rootHtml == null) {
                // 手动返回一个错误
                throw new Error('您传递 root 的属性, 有问题, 请检查后重新传递') 
            }

            // 3. 验证是否传递data
            if (options.data == undefined) {
                throw new Error('您没有传递 data 属性, 请重新传递')
            }

            // 4. 验证data是否为一个对象
            if (options.data.constructor !== Object) {
                throw new Error('您传递的 data 不是一个对象, 请重新传递')
            }

            // 5. 数据劫持
            // 存储一份最初的html文本
            const rootHtmlStr = rootHtml.innerHTML
            //  这里存储最开始的具有{{}}的HTML文本,
            //  如果不写, 第一次渲染完毕, HTML中就没有花括号, 第二次渲染正则没有办法正确匹配
            const _data = {}
            for (let k in options.data) {
                Object.defineProperty(_data, k, {
                    get() {
                        return options.data[k]
                    },
                    set(val) {
                        options.data[k] = val
                        // 5.2 数据更新后, 重新渲染页面
                        randers(rootHtml, _data, rootHtmlStr)
                    }
                })
            }

            // 5.1 首次执行, 渲染页面
            randers(rootHtml, _data, rootHtmlStr)

            // 6. 返回一个劫持后的数据
            return _data
        }

        function randers(root, _data, str) {
            // 准备一个正则
            const reg = /{{ *(\w+) *}}/g

            // 匹配到节点中所有的花括号, 存放在一个数组中
            const res = str.match(reg)

            // 遍历数组, 拿到每一个花括号
            res.forEach(item => {

                // 拿到花括号内的文本, 因为这是对象中的 key
                const key = reg.exec(str)[1]

                // 将原本花括号与内部文本, 全部替换为对象中实际的值
                str = str.replace(/{{ *(\w+) *}}/, _data[key])
            })

            // 重新将修改完毕的字符串, 渲染到页面
            root.innerHTML = str
        }
    </script>

调用的

    <script>
        // 我们后续使用, 就在这里
        const app = observer({
            root: '#app',
            data: {
                msg: '你好',
                name: 'QF001',
                age: 18,
                abc: '醒醒, 别睡了, 说你呢, 还睡'
            }
        })
    </script>

 五、数据劫持升级

    <div id="root"></div>

1、基本写法

        const obj = {
            name: 'QF001',
            age: 18
        }
        const res = {}
        Object.defineProperty(res, 'name', {
            get() {
                return obj.name;
            },
            set(val) {
                obj.name = val;
            }
        })

        obj.newName = 'QF999';
        console.log(obj);

        document.querySelector('#root').innerHTML = `name: ${obj.name}`;

2、升级版

语法: Object.defineProperties(到那个对象, {

        属性1: 配置项,

        属性2: 配置项

  })

        const obj = {
            name: 'QF001',
            age: 18
        }
        const res = {}
        Object.defineProperties(res, {
            name: {
                get() {
                    return obj.name
                },
                set(val) {
                    obj.name = val
                }
            },
            age: {
                get() {
                    return obj.age
                },
                set(val) {
                    obj.age = val
                }
            }
        })

        obj.newName = 'QF999';
        console.log(obj);
        document.querySelector('#root').innerHTML = `name: ${obj.name}`;

3、升级版(plus) 

        const obj = {
            name: 'QF001',
            age: 18
        }
        const res = {};
        // 升级版(plus)
        for (let k in obj) {
            // k === 'name'    2. k === age
            console.log(k);
            Object.defineProperties(res, {
                // 对象内部直接写 k 会帮当成字符串, 所以可以写成 [k], 将他识别为变量
                [k]: {
                    get() {
                        return obj[k];
                    },
                    set(val) {
                        obj[k] = val;
                    }
                },
            })
        }

        obj.newName = 'QF999';
        console.log(obj);
        document.querySelector('#root').innerHTML = `name: ${obj.name}`;

3、升级版(super plus),   自己劫持自己

        const obj = {
            name: 'QF001',
            age: 18
        }
   
        for (let k in obj) {
            Object.defineProperties(obj, {
                ['_' + k]: {    
                    // 我在我这个对象内把所有属性复制一份, 放在自己这个对象内部
                    value: obj[k],
                    writable: true
                },
                [k]: {
                    get () {
                        return obj['_' + k]
                    },
                    set(val) {
                        obj['_' + k] = val

                        document.querySelector('#root').innerHTML = `name: ${obj.name}`
                    }
                }
            })
        }
        obj.newName = 'QF999'
        console.log(obj)

        document.querySelector('#root').innerHTML = `name: ${obj.name}`

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

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

相关文章

RabbitMQ系列【16】AmqpTemplate接口详解

有道无术&#xff0c;术尚可求&#xff0c;有术无道&#xff0c;止于术。 文章目录前言AmqpTemplateAPIsendconvertAndSendreceivereceiveAndConvertreceiveAndReplysendAndReceiveconvertSendAndReceive前言 RabbitTemplate 是spring-amqp提供的一个 RabbitMQ 消息操作模板类…

【Git】rebase 和 merge 的区别

前言 今天想把本地的两个提交压缩成一个提交&#xff0c;再推送到远程。用的是rebase命令解决的&#xff0c;于是乎又捡起了之前的遗留问题&#xff1a;rebase和 merge 有什么区别&#xff1f; 用的是idea内置的git插件&#xff0c;先把idea官网对 “update project” 选择 “…

postgresql使用pg_basebackup备份与恢复

postgresql可以使用pg_dump,pg_restore等命令来进行备份与恢复&#xff0c;那种情况不用停止pgsql服务&#xff0c;只需要执行备份恢复命令即可。 今天介绍的这种备份方式&#xff0c;类似于文件系统的备份与恢复&#xff0c;它需要使用pg_basebackup命令来进行备份&#xff0c…

C#医院门诊会员管理系统源码 通用会员系统源码

C#通用医院会员管理系统源码 源码分享&#xff01; 本系统使用的技术为NhibernateEF,底层完全封装&#xff0c;可二次使用快速开发。 本技术具有以下特点&#xff1a; 1.面向对象方式访问数据库&#xff0c;摆脱SQL&#xff1b; 2.可移植性强&#xff0c;支持所有流行的数据…

光格科技递交科创板上会稿:拟募资6亿 预计年营收3亿

雷递网 雷建平 11月29日苏州光格科技股份有限公司&#xff08;简称&#xff1a;“光格科技”&#xff09;日前递交上会稿&#xff0c;准备在科创板上市。光格科技计划募资6亿&#xff0c;其中&#xff0c;3.1亿元用于分布式光纤传感系统升级研发及量产项目&#xff0c;8000万元…

Java EE|多线程基本操作

文章目录一、一个简单的线程程序及运行二、线程的创建三、线程类——Thread详解常见构造方法常见几个属性线程的启动——start()线程的中断线程的等待——join()线程引用的获取线程的休眠四、多线程编程效率举例一、一个简单的线程程序及运行 在写这样一个代码之前&#xff0c…

web框架

目录 1 左右分割窗口 2 上下分割窗口 3 嵌套分割窗口 4 内联框架 框架的作用是把浏览器窗口划分成若干个小窗口&#xff0c;每个小窗口可以分别显示不同的网页。 框架的基本结构主要分为框架集和框架两个部分&#xff0c;在网页中分别用<frameset>和<frame>标记…

Netty进阶——粘包与半包(代码示例)

目录一、消息粘包和消息半包的概述1.1、消息粘包1.2、消息半包二、粘包现象代码示例2.1、粘包现象服务端示例代码2.2、粘包现象客户端示例代码2.3、分别启动服务端&#xff0c;客户端&#xff0c;查看服务端结果输出三、半包现象代码示例3.1、半包现象服务端示例代码3.2、半包现…

【JavaSE】学习异常

前言&#xff1a; 作者简介&#xff1a;爱吃大白菜1132 人生格言:纸上得来终觉浅&#xff0c;绝知此事要躬行 如果文章知识点有错误的地方不吝赐教&#xff0c;和大家一起学习&#xff0c;一起进步&#xff01; 如果觉得博主文章还不错的话&#xff0c;希望三连支持&#xff01…

D-020 SPI FLASH硬件电路设计

SPI FLASH硬件电路设计1 简介2 EEPROM 和SPI Flash的区别3 电路设计实战4 电路设计要点1 简介 SPI FLASH(Serial Peripheral Interface)是串行外设接口的缩写&#xff0c;是一种高度、全双工、同步的通信总线。一般应用在MCU与外围设备之间通讯&#xff0c;广泛应用在FLASH&am…

从模型容量的视角看监督学习

这几天看离线强化学习瞎想的&#xff0c;不一定正确&#xff0c;仅记录个人想法 文章目录1. 监督学习的本质2. 容量视角下的模型选择、正则化和归纳偏置3. 几点启发1. 监督学习的本质 我认为监督学习的本质在于在过拟合和欠拟合之间取得平衡&#xff0c;捋一下逻辑 我们知道&a…

基于JAVA+SpringMVC+Mybatis+Vue+MYSQL的医药销售管理系统

项目介绍 药品一直以来在人类生活中扮演着非常重要的角色&#xff0c;随着时代的飞速发展&#xff0c;人们基本已经告别了那个缺医少药的年代&#xff0c;各大药房基本随处都可以购买&#xff0c;但是很多时候因为没有时间或者在药店很难找到自己想要购买的药品&#xff0c;所…

[附源码]计算机毕业设计springboot个人博客系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

Windows远程连接centos7图形化界面,安装xrdp

Windows远程连接centos7图形化界面&#xff0c;安装xrdp写在最前面准备工作查看ubuntu系统的版本信息和gcc版本尝试进入图形化界面更新yum下载安装图形化界面查询本地是否有Server with GUI group安装Server with GUI设置系统启动默认加入GUI界面安装epel库&#xff08;epel是社…

Djiango 模版系统详解(ORM数据模型-使用mysql数据库增删改查)

djiango模版系统&#xff1a; 用于自动渲染一个文本文件&#xff0c;一般用于HTML页面&#xff0c;模版引擎渲染的最终HTML内容返回给客户端浏览器 模版系统分成两部分 静态部分&#xff1a; 例如html css .js 动态部分 djiango 模版语言&#xff0c;类似于jinja语法变量定义&…

ArcGIS QGIS学习二:图层如何只显示需要的部分几何面数据(附最新坐标边界下载全国省市区县乡镇)

文章目录前言准备SHP数据ArcMap 的筛选QGIS 的筛选如何编写查询条件前言 当我们用GIS软件打开一个SHP文件的时候&#xff0c;会显示出里面全部的几何图形&#xff0c;假如我只想要其中的一部分数据显示出来&#xff0c;其他的均不要显示&#xff0c;有那么几种操作方法。 我们…

UE4使用蓝图实现角色冲刺

又学了几天&#xff0c;前面记录了如何使用蓝图实现开关门&#xff0c;这次来实现一下角色的冲刺、瞬移的操作 一般玩游戏的时候&#xff0c;可能都会有按左shift键让角色从行走变成奔跑的状态&#xff0c;又或者双击回避键角色瞬移躲避等操作 那就先实现一下加速奔跑吧&…

Python毕业设计必备案例:【学生信息管理系统】

嗨害大家好鸭&#xff01;我是小熊猫~ 最近看大家好像都快放假啦~ 但是放假之前有的小朋友要做毕业设计~ 有的要期末考试~ 那么今天来给大家整一个小的毕业管理系统案例 康康整题思路是怎么样的~ 源码、资料电子书点击这里获取 功能模块 基本信息管理和学生成绩管理。 基…

【自用】VUE 获取登录用户名 显示在其他页面上

大步骤一、准备工作 步骤1&#xff1a; 安装 js-cookie 依赖 npm install js-cookie --save步骤2&#xff1a; 在登录页面中引入 js-cookie 依赖 <script> import jsCookie from js-cookie; </script>大步骤二、在 登录页面 的vue文件 中使用它&#xff01; …

[附源码]计算机毕业设计springboot基于JAVA技术的旅游信息交互系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…