作为前端面试官我面试必须问一下面试者:描述一下你对MVVM的理解?

接下来,我将从零实现一套完整的基于Vue的MVVM,提供给来年“金三银四”跳槽高峰期的小伙伴们阅读也详细梳理一下自己对MVVM的理解。

图片alt

MVVM是什么

在了解MVVM之前,我们来对MVC说明一下。MVC架构起初以及现在一直存在于后端。MVC分别代表后台的三层,M代表模型层、V代表视图层、C代表控制器层,这三层架构完全可以满足于绝大分部的业务需求开发。
图片alt

MVC & 三层架构

下面以Java为例,分别阐述下MVC和三层架构中各层代表的含义以及职责:

  1. Model:模型层,代表着每一个JavaBean。其分为两类,一类称为数据承载Bean,一类称为业务处理Bean。
  2. View:视图层,代表着对应的视图页面,与用户直接进行交互。
  3. Controller:控制层,该层是Model和View的“中间人”,用于将用户请求转发给相应的Model进行处理,并处理Model的计算结果向用户提供相应响应。

以登录为例,介绍一下三层之间的逻辑关系。当用户点击View视图页面的登录按钮时,系统会调取Controller控制层里的登录接口。一般在Controller层中不会写很多具体的业务逻辑代码,只会写一个接口方法,该方法具体的逻辑在Service层进行实现,然后service层里的具体逻辑就会调用DAO层里的Model模型,从而达到动态化的效果。

MVVM 的描述

MVVM 设计模式,是由 MVC(最早来源于后端)、MVP 等设计模式进化而来。

  1. M - 数据模型(Model),简单的JS对象
  2. VM - 视图模型(ViewModel),连接Model与View
  3. V - 视图层(View),呈现给用户的DOM渲染界面
    图片alt
    通过以上的MVVM模式图,我们可以看出最核心的就是ViewModel,它主要的作用:对View中DOM元素的监听和对Model中的数据进行绑定,当View变化会引起Modal中数据的改动,Model中数据的改动会触发View视图重新渲染,从而达到数据双向绑定的效果,该效果也是Vue最为核心的特性。

常见库实现数据双向绑定的做法:

  • 发布订阅模式(Backbone.js)
  • 脏值检查(Angular.js)
  • 数据劫持(Vue.js)

面试者在回答Vue的双向数据绑定原理时,几乎所有人都会说:Vue是采用数据劫持结合发布订阅模式,通过Object.defineProperty()来劫持各个属性的getter,setter, 在数据变动时发布消息给订阅者,触发相应的回调函数,从而实现数据双向绑定。但当继续深入问道:

  • 实现一个MVVM里面需要那些核心模块?
  • 为什么操作DOM要在内存上进行?
  • 各个核心模块之间的关系是怎样的?
  • Vue中如何对数组进行数据劫持?
  • 你自己手动完整的实现过一个MVVM吗?

  • 图片alt

接下来,我将一步一步的实现一套完整的MVVM,当再次问道MVVM相关问题,完全可以在面试过程中脱颖而出。在开始编写MVVM之前,我们很有必要对核心API和发布订阅模式熟悉一下:

介绍一下 Object.defineProperty 的使用

Object.defineProperty(obj, prop, desc) 的作用就是直接在一个对象上定义一个新属性,或者修改一个已经存在的属性

  1. obj: 需要定义属性的当前对象
  2. prop: 当前需要定义的属性名
  3. desc: 属性描述符

注意:一般通过为对象的属性赋值的情况下,对象的属性可以修改也可以删除,但是通过Object.defineProperty()定义属性,通过描述符的设置可以进行更精准的控制对象属性。
图片alt

let obj = {}
Object.defineProperty(obj, 'name', {configurable: true,   // 默认为false,可配置的【删除】writable: true,       // 默认为false, 是否可写【修改】enumerable: true,     // 默认为false, 是否可枚举【for in 遍历】value: 'sfm',         // name属性的值get() {// 获取obj.name的值时会调用get函数},set(val) {// val就是重新赋值的值// 重新给obj.name赋值时会调用set函数}
})

注意:当出现get,set函数时,不能同时出现writable, enumerable属性,否则系统报错。并且该API不支持IE8以下的版本,也就是Vue不兼容IE8以下的浏览器。

DocumentFragment - 文档碎片

DocumentFragment 表示文档片段,它不属于 DOM 树,但是它可以存储 DOM,并且可以将所存储的 DOM 加入到指定的 DOM 节点中去。那么有人要问了,那要它何用,直接把元素加入到 DOM 中不就可以了吗?用它的原因在于,使用它操作 DOM 要比直接操作 DOM 性能要高很多。

介绍一下 发布订阅模式

发布者-订阅者模式定义了一种一对多的依赖关系,即当一个对象的状态发生改变时,所有依赖于他的对象都会得到通知并自动更新,解决了主体对象与观察者之间功能的耦合。以下是一个发布订阅模式的小例子,实际上可以理解为靠的就是数组关系,订阅就是放入函数,发布就是让数组里的函数执行。

// 发布订阅模式  先有订阅后有发布
function Dep() {this.subs = [];
}
// 订阅
Dep.prototype.addSub = function(sub) {this.subs.push(sub);
}
Dep.prototype.notify = function() {this.subs.forEach(sub => sub.update());
}
// Watcher类,通过这个类创建的实例都有update方法
function Watcher(fn) {this.fn = fn;
}
Watcher.prototype.update = function() {this.fn();
}
let watcher1 = new Watcher(function() {console.log(123);
})
let watcher2 = new Watcher(function() {console.log(456);
})
let dep = new Dep();
dep.addSub(watcher1); // 将watcher放到了数组中
dep.addSub(watcher2);
dep.notify();// 控制台输出:
// 123 456

图片alt

实现自己的 MVVM

要实现mvvm的双向绑定,就必须要实现以下几点:

  1. 实现一个数据劫持 - Observer,能够对数据对象的所有属性进行监听,如有变动可拿到最新值并通知订阅者
  2. 实现一个模板编译 - Compiler,对每个元素节点的指令进行扫描和解析,根据指令模板替换数据,以及绑定相应的更新函数
  3. 实现一个 - Watcher,作为连接Observer和Compile的桥梁,能够订阅并收到每个属性变动的通知,执行指令绑定的相应回调函数,从而更新视图
  4. MVVM 作为入口函数,整合以上三者

图片alt

数据劫持 - Observer

Observer 类主要目的就是给 data 数据内的所有层级的数据都进行数据劫持,让其具备监听对象属性变化的能力

【重点】:

  1. 当对象的属性值也是对象时,也要对其值进行劫持 — 递归
  2. 当对象赋值与旧值一样,则不需要后续操作 — 防止重复渲染
  3. 当模板渲染获取对象属性会调用get添加target,对象属性改动通知订阅者更新 — 数据变化,视图更新
// 数据劫持
class Observer {constructor(data) {this.observer(data);}observer(data) {if(data && typeof data == 'object') {// 判断data数据存在 并 data是对象  才观察for(let key in data) {this.defineReactive(data, key, data[key]);}}}defineReactive(obj, key, value) {let dep = new Dep();this.observer(value); // 如果value还是对象,还需要观察Object.defineProperty(obj, key, {get() {Dep.target && dep.addSub(Dep.target);return value;},set:(newVal) => { // 设置新值if(newVal != value) { // 新值和就值如果一致就不需要替换了this.observer(newVal); // 如果赋值的也是对象的话  还需要观察value = newVal;dep.notify(); // 通知所有订阅者更新了}}})}
}

注意:该类只会对对象进行数据劫持,并不会对数组的监听。

模板编译 - Compiler

Compiler 是解析模板指令,将模板中的变量替换成数据,然后初始化渲染页面视图,并将每个指令对应的节点绑定更新函数,添加监听数据的订阅者,一旦数据有变动,收到通知,更新视图

Compiler 主要做了三件事:

  • 将当前根节点所有子节点遍历放到内存中
  • 编译文档碎片,替换模板(元素、文本)节点中属性的数据
  • 将编译的内容回写到真实DOM上

【重点】:

  1. 先把真实的 dom 移入到内存中操作 — 文档碎片
  2. 编译 元素节点 和 文本节点
  3. 给模板中的表达式和属性添加观察者
// 模板编译
class Compiler {/*** @param {*} el 元素 注意:el选项中有可能是‘#app’字符串也有可能是document.getElementById('#app')* @param {*} vm 实例*/constructor(el, vm) {// 判断el属性  是不是一个元素  如果不是元素就获取this.el = this.isElementNode(el) ? el : document.querySelector(el);// console.log(this.el);拿到当前的模板this.vm = vm;// 把当前节点中的元素获取到  放到内存中  防止页面重绘let fragment = this.node2fragment(this.el);// console.log(fragment);内存中所有的节点// 1. 编译模板 用data中的数据编译this.compile(fragment);// 2. 把内存中的内容进行替换this.el.appendChild(fragment);// 3. 再把替换后的内容回写到页面中}/*** 判断是含有指令* @param {*} attrName 属性名 type v-modal*/isDirective(attrName) {return attrName.startsWith('v-'); // 是否含有v-}/*** 编译元素节点* @param {*} node 元素节点*/compileElement(node) {// 获取当前元素节点的属性;【类数组】NamedNodeMap; 也存在没有属性,则NamedNodeMap{length: 0}let attributes = node.attributes;// Array.from()、[...xxx]、[].slice.call 等都可以将类数组转化为真实数组[...attributes].forEach(attr => {// attr格式:type="text"  v-modal="obj.name"let {name, value: expr} = attr;// 判断是不是指令if(this.isDirective(name)) { // v-modal v-html v-bind// console.log('element', node); 元素let [, directive] = name.split('-'); // 获取指令名// 需要调用不同的指令来处理CompilerUtil[directive](node, expr, this.vm);}});}/*** 编译文本节点 判断当前文本节点中的内容是否含有 {{}}* @param {*} node 文本节点*/compileText(node) {let content = node.textContent;// console.log(content, ‘内容’); 元素里的内容if(/\{\{(.+?)\}\}/.test(content)) { // 通过正则去匹配只需要含有{{}}大括号的,空的不需要 获取大括号中间的内容// console.log(content, ‘内容’); 只包含{{}} 不需要空的 和其他没有{{}}的子元素CompilerUtil['text'](node, content, this.vm);}}/*** 编译内存中的DOM节点* @param {*} fragmentNode 文档碎片*/compile(fragmentNode) {// 从文档碎片中拿到子节点  注意:childNodes【之包含第一层,不包含{{}}等】let childNodes = fragmentNode.childNodes; // 获取的是类数组NodeLis[...childNodes].forEach(child => {// 是否是元素节点if (this.isElementNode(child)) {this.compileElement(child);// 如果是元素的话  需要把自己传进去  再去遍历子节点   递归this.compile(child);} else {// 文本节点// console.log('text', child);this.compileText(child);}});}/*** 将节点中的元素放到内存中* @param {*} node 节点*/node2fragment(node) {// 创建一个稳定碎片;目的是为了将这个节点中的每个孩子都写到这个文档碎片中let fragment = document.createDocumentFragment();let firstChild; // 这个节点中的第一个孩子while (firstChild = node.firstChild) {// appendChild具有移动性,每移动一个节点到内存中,页面上就会少一个节点fragment.appendChild(firstChild);}return fragment;}/*** 判断是不是元素* @param {*} node 当前这个元素的节点*/isElementNode(node) {return node.nodeType === 1;}
}
// 编译功能
CompilerUtil = {/*** 根据表达式取到对应的数据* @param {*} vm * @param {*} expr */getVal(vm, expr) {return expr.split('.').reduce((data, current) => {return data[current];}, vm.$data);},setVal(vm, expr, value) {expr.split('.').reduce((data, current, index, arr) => {if (index === arr.length - 1) {return data[current] = value;}return data[current]}, vm.$data)},/*** 处理v-modal* @param {*} node 对应的节点* @param {*} expr 表达式* @param {*} vm 当前实例*/modal(node, expr, vm) {// 给输入框赋予value属性 node.value = xxxlet fn = this.updater['modalUpdater'];new Watcher(vm, expr, (newValue) => {//给输入框加一个观察者 数据更新会触发此方法 会拿新值给 输入框赋值fn(node, newValue)})node.addEventListener('input', e => {let value = e.target.value; // 获取用户输入的内容this.setVal(vm, expr, value);})let value = this.getVal(vm, expr); // 返回tmcfn(node, value);},text(node, expr, vm) {let fn = this.updater['textUpdater'];let content = expr.replace(/\{\{(.+?)\}\}/g, (...args) => {// 给表达式 每个{{}} 加上观察者new Watcher(vm, args[1], (newValue) => {fn(node, this.getContentValue(vm, expr)); // 返回了一个新的字符串})return this.getVal(vm, args[1].trim());});fn(node, content);},updater: {// 把数据插入到节点中modalUpdater(node, value) {node.value = value;},// 处理文本节点textUpdater(node, value) {node.textContent = value;}}
}

Complier 具备将 HTML 模版解析成 Document Fragment 的能力,并且会创建响应的 Watcher,让视图中绑定的数据产生变化。

发布订阅 - Watcher

Watcher 订阅者作为 Observer 和 Compile 之间通信的桥梁,主要做的事情是:

  1. 在自身实例化时往属性订阅器(dep)里面添加自己
  2. 自身必须有一个update()方法
  3. 待属性变动dep.notice()通知时,能调用自身的update()方法,并触发Compile中绑定的回调,则功成身退。
// 发布订阅
function Dep() { this.subs = []
}
Dep.prototype.addSub = function(sub) {this.subs.push(sub)
}
Dep.prototype.notify = function() {this.subs.forEach(sub => sub.update())
}
/*** watcher* @param {*} vm 当前实例* @param {*} exp 表达式* @param {*} fn 监听函数*/
function Watcher(vm, exp, fn) { this.fn = fn;this.vm = vm;this.exp = exp; // 添加到订约中Dep.target = this;let val = vm;let arr = exp.split('.');arr.forEach(function (k) { val = val[k];})Dep.target = null; // 保证watcher不会重复添加
}
Watcher.prototype.update = function() {let val = this.vm;let arr = this.exp.split('.');arr.forEach(function (k) { val = val[k];})this.fn(val)
}

Dep 和 Watcher 是简单的观察者模式的实现,Dep 即订阅者,它会管理所有的观察者,并且有给观察者发送消息的能力。Watcher 即观察者,当接收到订阅者的消息后,观察者会做出自己的更新操作。

整合 - MVVM

MVVM作为数据绑定的入口,整合Observer、Compile和Watcher三者,通过Observer来监听自己的model数据变化,通过Compile来解析编译模板指令,最终利用Watcher搭起Observer和Compile之间的通信桥梁,达到数据变化 -> 视图更新;视图交互变化(input) -> 数据model变更的双向绑定效果。

class MVVM {constructor(options) {// 当new该类时,参数就会传到构造函数中 options就是el  data computed ...this.$el = options.el; // 创建一个当前实例$elthis.$data = options.data;// 判断根元素是否存在 <div id='app'></div> =>  编译模板if (this.$el) {// 把data里的数据 全部转化成用Object.defineProperty来定义new Observer(this.$data);new Compiler(this.$el, this);}}
}
注意:这样有个问题? 
在开发中是能通过实例+属性(vm.a)来获取数据,而我们实现的MVVM获取数据要通过myMvvm.$data.xxx来获取到数据,中间多了一个$data,这样显然不是我们想要的样子。接下来实现让实例this来代理$data数据,即可实现myMvvm.xxx获取数据和真实场景一样的操作。

数据代理

在MVVM实例上添加一个属性代理的方法,使访问myMvvm的属性代理为访问myMvvm.$data的属性。其实还是利用了Object.defineProperty()方法来劫持了myMvvm实例对象的属性。添加的代理方法如下:

// this 代理 $datafor (let key in data) {Object.defineProperty(this, key, {enumerable: true,get() {return this.$data[key]; // this.xxx == {}},set(newVal) {this.$data[key] = newVal;}})}

图片alt

扩展 - 实现computed

computed 具有缓存功能,当依赖的属性发送变化,才会更新视图变化

function initComputed() {let vm = this; // 将当前this挂载到vm上let computed = this.$options.computed;  // 从options上拿到computed属性// 得到的都是对象的key可以通过Object.keys转化为数组Object.keys(computed).forEach(key => {Object.defineProperty(vm, key, { // 映射到this实例上// 判断是computed里的key是对象还是函数// 若是函数,则直接就调get方法// 若是对象,则需要手动调一下get方法// 因为computed只根据依赖的属性进行触发,当获取依赖属性时,系统会自动的去调用get方法,所以就不要用Watcher去监听变化了get: typeof computed[key] === 'function' ? computed[key] : computed[key].get,set() {}});});
}

项目git地址

https://github.com/tangmengcheng/mvvm.git 欢迎小伙伴点赞、评论加关注哦🤭 star~

相关问题

  • Objest.defineProperty() 有那些缺点?
  • 实现数组的一个监听?
  • Vue3 中是如何用 Proxy 实现的?
  • Vue2.x 源码中数据双向绑定大致的实现流程?

图片alt

总结

通过以上描述和核心代码的演示,相信小伙伴们对MVVM有重新的认识,面试中对面面试官的提问可以对答如流。希望同行的小伙伴手动敲一遍,实现一个自己MVVM,这样对其原理理解更加深入。

随着日益月薪需求的不断增加,jQuery操作DOM的时代已经满足不了企业项目快速迭代的进度了。 MVVM 模式对于前端领域有着重大的意义,其核心原理:实时保证View层与Model层的数据同步,实现了双向数据绑定。减少了频繁的DOM操作,提高了页面渲染的性能,也让开发者把更多的时间放到数据的处理以及业务功能的开发上。

最后

如果本文对你有帮助得话,给本文点个赞❤️❤️❤️

欢迎大家加入,一起学习前端,共同进步!
cmd-markdown-logo
cmd-markdown-logo

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

相关文章

  1. java Swing教程

    swing入门教程UI 组件简介在开始学习 Swing 之前,必须回答针对真正初学者的一个问题:什么是 UI?初学者的答案是“用户界面”。但是因为本教程的目标是要保证您不再只是个初学者,所以我们需要比这个定义更高级的定义。所以,我再次提出这个问题:什么是 UI?您可能把它定义成…...

    2024/5/2 12:12:36
  2. Angular模块/服务/MVVM

    angular 与 jquery 共同点 都是一个前端的JS文件而已 不同点&#xff1a; angular 是一个框架 我们写的代码由框架调用&#xff0c;我们必须要按照特定的规则编写代码 jquery是一个库 我们调用库预制的方法或对象实现我们想要功能 在思想上&#xff1a;jquery提高了操作DOM…...

    2024/4/21 3:08:50
  3. mui.ajax ie8,IE8+MVVM的适配方案尝试

    概述你是不是还在为客户不肯升级浏览器而烦恼&#xff1f;你是不是还在为IE8调试代码而抓狂&#xff1f;你是不是也在吐槽IE8稀烂的性能&#xff1f;如果你也碰到以上问题&#xff0c;恭喜你&#xff01;&#xff01;&#xff01;看了这篇文章其实也没啥用... T_T首先&#xff…...

    2024/4/24 21:31:01
  4. 第 1 章: Vue 核心

    1.1. Vue的基本认识 1.1.1. 官网 英文官网:https://vuejs.org/ 中文官网:https://cn.vuejs.org/ 1.1.2. 介绍描述 渐进式 JavaScript 框架 作者: 尤雨溪(一位华裔前 Google 工程师) 作用: 动态构建用户界面 1.1.3. Vue 的特点 遵循 MVVM 模式 编码简洁, 体积小, 运行效…...

    2024/4/21 3:08:49
  5. MVVM框架热门常用的组件库收集整理

    在前端项目的开发过程中&#xff0c;少不了需要选择一款组件库作为项目开发的基础&#xff0c;而只要选择对了一款的合适的组件库就能极大的提升开发效率&#xff0c;直接让人体会到飞一般的感觉&#xff0c;简直爽的不要不要的。减少很多冗余的操作&#xff0c;和一些基础的代…...

    2024/4/21 3:08:47
  6. 一篇让你学懂AngularJS

    AngularJS是什么? AngularJS是一个开源Web应用程序框架。它最初是由MISKO Hevery和Adam Abrons于2009年开发。现在是由谷歌维护。它的最新版本是1.3.14。 AngularJS在它的官方文档 中定义如下&#xff1a; AngularJS is a structural framework for dynamic web apps. It le…...

    2024/4/21 3:08:46
  7. (翻译)Angular.js为什么如此火呢?

    在本文中让我们来逐步发掘angular为什么如此火: Angular.js 是一个MV*&#xff08;Model-View-Whatever&#xff0c;不管是MVC或者MVVM&#xff0c;统归MDV(model Drive View)&#xff09;JavaScript框架&#xff0c;其是Google推出的SPA&#xff08;single-page-application&a…...

    2024/4/21 3:08:46
  8. angular.js,backbone.js,ember.js区别

    2019独角兽企业重金招聘Python工程师标准>>> 1 backbone. 对于初学者来说,我非常建议首先学习backbone 而不是jQuery. 因为jQuery提供的功能是操作DOM和Ajax数据传输. 而Backbone的功能和目的仅仅是提供一个javascript的MVC 实际针对Web就是 MVR. (Model, View, Ro…...

    2024/4/21 3:08:46
  9. 【Angular】Can‘t have multiple template bindings on one element. Use only one attribute...

    目录 一. 场景再现&#xff08;一个元素上同时使用 *ngIf 和 *ngFor&#xff09; 二. 解决方案&#xff08;使用 包裹组件&#xff09;​​​​​​​ 一. 场景再现&#xff08;一个元素上同时使用 *ngIf 和 *ngFor&#xff09; 下面是 Angular 中的模板代码&#xff1a;<th…...

    2024/4/20 15:22:00
  10. Angular 作用域 scope

    文章目录什么是作用域如何创建作用域作用作用域作为数据模型使用作用域分层结构在DOM中抓取作用域作用域的事件传播scope的生命周期创建&#xff08;creation&#xff09;注册监听&#xff08;Watcher registration&#xff09;数据模型变化&#xff08;Model mutation&#xf…...

    2024/4/21 3:08:42
  11. html中scope的作用,AngularJS 作用域(Scope)

    AngularJS 作用域(Scope)作用域(Scope)是一个存储应用数据模型的对象为 表达式 提供了一个执行上下文作用域的层级结构对应于 DOM 树结构作用域可以监听 表达式 的变化并传播事件作用域有什么作用域提供了 ($watch) 方法监听数据模型的变化作用域提供了 ($apply) 方法把不是由A…...

    2024/4/20 20:12:12
  12. angular.4 引入百度地图

    1、在index.html中引入 bmap.min.js <script type"text/javascript" src"./assets/echarts/bmap.min.js"></script><script type"text/javascript" src"http://api.map.baidu.com/api?v2.0&ak自己申请的密钥">…...

    2024/4/20 20:12:11
  13. angular.element方法汇总以及AngularJS 动态添加元素和删除元素

    addClass()-为每个匹配的元素添加指定的样式类名after()-在匹配元素集合中的每个元素后面插入参数所指定的内容&#xff0c;作为其兄弟节点append()-在每个匹配元素里面的末尾处插入参数内容attr() - 获取匹配的元素集合中的第一个元素的属性的值bind() - 为一个元素绑定一个事…...

    2024/4/20 20:12:10
  14. angular.json

    newProjectRoot ng g lib ** 的根目录 projects 项目 cli defaultCollectionpackageManagerwarningsschematics schematics/angular:componentschematics/angular:directiveschematics/angular:moduleschematics/angular:serviceschematics/angular:pipeschematics/angular:cla…...

    2024/4/20 4:14:20
  15. angular.run 妙用

    **1.浏览器判断**在angular做微信应用的时候&#xff0c;有时候我们也想把相同一份代码运行在非微信的浏览器上&#xff0c;这时候我们可以在angular的run上写点东西实现~例如asw.run函数里执行定义一个$rootScope.isWeiXinLogin的函数 .run([$rootScope, $route, $window, $lo…...

    2024/4/21 3:08:42
  16. angular.element的常用方法

    addClass()-为每个匹配的元素添加指定的样式类名after()-在匹配元素集合中的每个元素后面插入参数所指定的内容&#xff0c;作为其兄弟节点append()-在每个匹配元素里面的末尾处插入参数内容attr() - 获取匹配的元素集合中的第一个元素的属性的值bind() - 为一个元素绑定一个事…...

    2024/4/23 8:30:44
  17. AngularJs angular.element

    angular.element 将DOM元素或者HTML字符串一包装成一个jQuery元素。 格式&#xff1a;angular.element(element); element&#xff1a;包装成jquery对象的html字符串或者dom元素 jqLite提供的方法&#xff1a; addClass()after()append()attr()bind() – 不支持命名空间,选择器…...

    2024/4/21 3:08:39
  18. angularjs获取元素以及angular.element()用法

    addClass()-为每个匹配的元素添加指定的样式类名 after()-在匹配元素集合中的每个元素后面插入参数所指定的内容&#xff0c;作为其兄弟节点 append()-在每个匹配元素里面的末尾处插入参数内容 attr() - 获取匹配的元素集合中的第一个元素的属性的值 bind() - 为一个元素绑定一…...

    2024/4/21 3:08:38
  19. angular element()

    使用angular.element()获取一个dom的方法。 1.可以使用jquery的选择器 2.可以使用javascript的原生查找元素的方法 下面是angular.element()提供的方法 <input type"checkbox" class"input" /><input type"text" class"input1&quo…...

    2024/4/21 3:08:38
  20. Angular 学习系列 - - angular.element

    angular.element 将DOM元素或者HTML字符串一包装成一个jQuery元素。 格式&#xff1a;angular.element(element); element&#xff1a;包装成jquery对象的html字符串或者dom元素 jqLite提供的方法&#xff1a; addClass()after()append()attr()bind() – 不支持命名空间,选择器…...

    2024/4/21 3:08:36

最新文章

  1. [Python基础知识]05函数和模块

    一、函数的定义 格式&#xff1a;def 函数名&#xff08;参数列表&#xff09;: 注&#xff1a; 函数代码块以 def 关键词开头&#xff0c;后接函数标识符名称和圆括号()。即使该函数不需要接收任何参数&#xff0c;也必须保留一对空的圆括号 函数形参不需要声明其类型&#x…...

    2024/5/2 18:37:30
  2. 梯度消失和梯度爆炸的一些处理方法

    在这里是记录一下梯度消失或梯度爆炸的一些处理技巧。全当学习总结了如有错误还请留言&#xff0c;在此感激不尽。 权重和梯度的更新公式如下&#xff1a; w w − η ⋅ ∇ w w w - \eta \cdot \nabla w ww−η⋅∇w 个人通俗的理解梯度消失就是网络模型在反向求导的时候出…...

    2024/3/20 10:50:27
  3. Redis Stack十部曲之三:理解Redis Stack中的数据类型

    文章目录 前言String字符串作为计数器限制 List限制列表阻塞列表自动创建和删除聚合类型键限制 Set限制 Hash限制 Sorted Set范围操作字典操作更新分数 JSON路径限制 BitMapBitfieldProbabilisticHyperLogLogBloom filterCuckoo filtert-digestTop-KCount-min sketchConfigurat…...

    2024/5/2 2:37:49
  4. Verilog基础【二】

    3.1 Verilog 连续赋值 关键词&#xff1a;assign&#xff0c; 全加器 连续赋值语句是 Verilog 数据流建模的基本语句&#xff0c;用于对 wire 型变量进行赋值。&#xff1a; assign LHS_target RHS_expression &#xff1b;LHS&#xff08;left hand side&#xff09;…...

    2024/5/1 5:48:25
  5. 【外汇早评】美通胀数据走低,美元调整

    原标题:【外汇早评】美通胀数据走低,美元调整昨日美国方面公布了新一期的核心PCE物价指数数据,同比增长1.6%,低于前值和预期值的1.7%,距离美联储的通胀目标2%继续走低,通胀压力较低,且此前美国一季度GDP初值中的消费部分下滑明显,因此市场对美联储后续更可能降息的政策…...

    2024/5/1 17:30:59
  6. 【原油贵金属周评】原油多头拥挤,价格调整

    原标题:【原油贵金属周评】原油多头拥挤,价格调整本周国际劳动节,我们喜迎四天假期,但是整个金融市场确实流动性充沛,大事频发,各个商品波动剧烈。美国方面,在本周四凌晨公布5月份的利率决议和新闻发布会,维持联邦基金利率在2.25%-2.50%不变,符合市场预期。同时美联储…...

    2024/5/2 16:16:39
  7. 【外汇周评】靓丽非农不及疲软通胀影响

    原标题:【外汇周评】靓丽非农不及疲软通胀影响在刚结束的周五,美国方面公布了新一期的非农就业数据,大幅好于前值和预期,新增就业重新回到20万以上。具体数据: 美国4月非农就业人口变动 26.3万人,预期 19万人,前值 19.6万人。 美国4月失业率 3.6%,预期 3.8%,前值 3…...

    2024/4/29 2:29:43
  8. 【原油贵金属早评】库存继续增加,油价收跌

    原标题:【原油贵金属早评】库存继续增加,油价收跌周三清晨公布美国当周API原油库存数据,上周原油库存增加281万桶至4.692亿桶,增幅超过预期的74.4万桶。且有消息人士称,沙特阿美据悉将于6月向亚洲炼油厂额外出售更多原油,印度炼油商预计将每日获得至多20万桶的额外原油供…...

    2024/5/2 9:28:15
  9. 【外汇早评】日本央行会议纪要不改日元强势

    原标题:【外汇早评】日本央行会议纪要不改日元强势近两日日元大幅走强与近期市场风险情绪上升,避险资金回流日元有关,也与前一段时间的美日贸易谈判给日本缓冲期,日本方面对汇率问题也避免继续贬值有关。虽然今日早间日本央行公布的利率会议纪要仍然是支持宽松政策,但这符…...

    2024/4/27 17:58:04
  10. 【原油贵金属早评】欧佩克稳定市场,填补伊朗问题的影响

    原标题:【原油贵金属早评】欧佩克稳定市场,填补伊朗问题的影响近日伊朗局势升温,导致市场担忧影响原油供给,油价试图反弹。此时OPEC表态稳定市场。据消息人士透露,沙特6月石油出口料将低于700万桶/日,沙特已经收到石油消费国提出的6月份扩大出口的“适度要求”,沙特将满…...

    2024/4/27 14:22:49
  11. 【外汇早评】美欲与伊朗重谈协议

    原标题:【外汇早评】美欲与伊朗重谈协议美国对伊朗的制裁遭到伊朗的抗议,昨日伊朗方面提出将部分退出伊核协议。而此行为又遭到欧洲方面对伊朗的谴责和警告,伊朗外长昨日回应称,欧洲国家履行它们的义务,伊核协议就能保证存续。据传闻伊朗的导弹已经对准了以色列和美国的航…...

    2024/4/28 1:28:33
  12. 【原油贵金属早评】波动率飙升,市场情绪动荡

    原标题:【原油贵金属早评】波动率飙升,市场情绪动荡因中美贸易谈判不安情绪影响,金融市场各资产品种出现明显的波动。随着美国与中方开启第十一轮谈判之际,美国按照既定计划向中国2000亿商品征收25%的关税,市场情绪有所平复,已经开始接受这一事实。虽然波动率-恐慌指数VI…...

    2024/4/30 9:43:09
  13. 【原油贵金属周评】伊朗局势升温,黄金多头跃跃欲试

    原标题:【原油贵金属周评】伊朗局势升温,黄金多头跃跃欲试美国和伊朗的局势继续升温,市场风险情绪上升,避险黄金有向上突破阻力的迹象。原油方面稍显平稳,近期美国和OPEC加大供给及市场需求回落的影响,伊朗局势并未推升油价走强。近期中美贸易谈判摩擦再度升级,美国对中…...

    2024/4/27 17:59:30
  14. 【原油贵金属早评】市场情绪继续恶化,黄金上破

    原标题:【原油贵金属早评】市场情绪继续恶化,黄金上破周初中国针对于美国加征关税的进行的反制措施引发市场情绪的大幅波动,人民币汇率出现大幅的贬值动能,金融市场受到非常明显的冲击。尤其是波动率起来之后,对于股市的表现尤其不安。隔夜美国股市出现明显的下行走势,这…...

    2024/5/2 15:04:34
  15. 【外汇早评】美伊僵持,风险情绪继续升温

    原标题:【外汇早评】美伊僵持,风险情绪继续升温昨日沙特两艘油轮再次发生爆炸事件,导致波斯湾局势进一步恶化,市场担忧美伊可能会出现摩擦生火,避险品种获得支撑,黄金和日元大幅走强。美指受中美贸易问题影响而在低位震荡。继5月12日,四艘商船在阿联酋领海附近的阿曼湾、…...

    2024/4/28 1:34:08
  16. 【原油贵金属早评】贸易冲突导致需求低迷,油价弱势

    原标题:【原油贵金属早评】贸易冲突导致需求低迷,油价弱势近日虽然伊朗局势升温,中东地区几起油船被袭击事件影响,但油价并未走高,而是出于调整结构中。由于市场预期局势失控的可能性较低,而中美贸易问题导致的全球经济衰退风险更大,需求会持续低迷,因此油价调整压力较…...

    2024/4/26 19:03:37
  17. 氧生福地 玩美北湖(上)——为时光守候两千年

    原标题:氧生福地 玩美北湖(上)——为时光守候两千年一次说走就走的旅行,只有一张高铁票的距离~ 所以,湖南郴州,我来了~ 从广州南站出发,一个半小时就到达郴州西站了。在动车上,同时改票的南风兄和我居然被分到了一个车厢,所以一路非常愉快地聊了过来。 挺好,最起…...

    2024/4/29 20:46:55
  18. 氧生福地 玩美北湖(中)——永春梯田里的美与鲜

    原标题:氧生福地 玩美北湖(中)——永春梯田里的美与鲜一觉醒来,因为大家太爱“美”照,在柳毅山庄去寻找龙女而错过了早餐时间。近十点,向导坏坏还是带着饥肠辘辘的我们去吃郴州最富有盛名的“鱼头粉”。说这是“十二分推荐”,到郴州必吃的美食之一。 哇塞!那个味美香甜…...

    2024/4/30 22:21:04
  19. 氧生福地 玩美北湖(下)——奔跑吧骚年!

    原标题:氧生福地 玩美北湖(下)——奔跑吧骚年!让我们红尘做伴 活得潇潇洒洒 策马奔腾共享人世繁华 对酒当歌唱出心中喜悦 轰轰烈烈把握青春年华 让我们红尘做伴 活得潇潇洒洒 策马奔腾共享人世繁华 对酒当歌唱出心中喜悦 轰轰烈烈把握青春年华 啊……啊……啊 两…...

    2024/5/1 4:32:01
  20. 扒开伪装医用面膜,翻六倍价格宰客,小姐姐注意了!

    原标题:扒开伪装医用面膜,翻六倍价格宰客,小姐姐注意了!扒开伪装医用面膜,翻六倍价格宰客!当行业里的某一品项火爆了,就会有很多商家蹭热度,装逼忽悠,最近火爆朋友圈的医用面膜,被沾上了污点,到底怎么回事呢? “比普通面膜安全、效果好!痘痘、痘印、敏感肌都能用…...

    2024/4/27 23:24:42
  21. 「发现」铁皮石斛仙草之神奇功效用于医用面膜

    原标题:「发现」铁皮石斛仙草之神奇功效用于医用面膜丽彦妆铁皮石斛医用面膜|石斛多糖无菌修护补水贴19大优势: 1、铁皮石斛:自唐宋以来,一直被列为皇室贡品,铁皮石斛生于海拔1600米的悬崖峭壁之上,繁殖力差,产量极低,所以古代仅供皇室、贵族享用 2、铁皮石斛自古民间…...

    2024/4/28 5:48:52
  22. 丽彦妆\医用面膜\冷敷贴轻奢医学护肤引导者

    原标题:丽彦妆\医用面膜\冷敷贴轻奢医学护肤引导者【公司简介】 广州华彬企业隶属香港华彬集团有限公司,专注美业21年,其旗下品牌: 「圣茵美」私密荷尔蒙抗衰,产后修复 「圣仪轩」私密荷尔蒙抗衰,产后修复 「花茵莳」私密荷尔蒙抗衰,产后修复 「丽彦妆」专注医学护…...

    2024/4/30 9:42:22
  23. 广州械字号面膜生产厂家OEM/ODM4项须知!

    原标题:广州械字号面膜生产厂家OEM/ODM4项须知!广州械字号面膜生产厂家OEM/ODM流程及注意事项解读: 械字号医用面膜,其实在我国并没有严格的定义,通常我们说的医美面膜指的应该是一种「医用敷料」,也就是说,医用面膜其实算作「医疗器械」的一种,又称「医用冷敷贴」。 …...

    2024/5/2 9:07:46
  24. 械字号医用眼膜缓解用眼过度到底有无作用?

    原标题:械字号医用眼膜缓解用眼过度到底有无作用?医用眼膜/械字号眼膜/医用冷敷眼贴 凝胶层为亲水高分子材料,含70%以上的水分。体表皮肤温度传导到本产品的凝胶层,热量被凝胶内水分子吸收,通过水分的蒸发带走大量的热量,可迅速地降低体表皮肤局部温度,减轻局部皮肤的灼…...

    2024/4/30 9:42:49
  25. 配置失败还原请勿关闭计算机,电脑开机屏幕上面显示,配置失败还原更改 请勿关闭计算机 开不了机 这个问题怎么办...

    解析如下&#xff1a;1、长按电脑电源键直至关机&#xff0c;然后再按一次电源健重启电脑&#xff0c;按F8健进入安全模式2、安全模式下进入Windows系统桌面后&#xff0c;按住“winR”打开运行窗口&#xff0c;输入“services.msc”打开服务设置3、在服务界面&#xff0c;选中…...

    2022/11/19 21:17:18
  26. 错误使用 reshape要执行 RESHAPE,请勿更改元素数目。

    %读入6幅图像&#xff08;每一幅图像的大小是564*564&#xff09; f1 imread(WashingtonDC_Band1_564.tif); subplot(3,2,1),imshow(f1); f2 imread(WashingtonDC_Band2_564.tif); subplot(3,2,2),imshow(f2); f3 imread(WashingtonDC_Band3_564.tif); subplot(3,2,3),imsho…...

    2022/11/19 21:17:16
  27. 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机...

    win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”问题的解决方法在win7系统关机时如果有升级系统的或者其他需要会直接进入一个 等待界面&#xff0c;在等待界面中我们需要等待操作结束才能关机&#xff0c;虽然这比较麻烦&#xff0c;但是对系统进行配置和升级…...

    2022/11/19 21:17:15
  28. 台式电脑显示配置100%请勿关闭计算机,“准备配置windows 请勿关闭计算机”的解决方法...

    有不少用户在重装Win7系统或更新系统后会遇到“准备配置windows&#xff0c;请勿关闭计算机”的提示&#xff0c;要过很久才能进入系统&#xff0c;有的用户甚至几个小时也无法进入&#xff0c;下面就教大家这个问题的解决方法。第一种方法&#xff1a;我们首先在左下角的“开始…...

    2022/11/19 21:17:14
  29. win7 正在配置 请勿关闭计算机,怎么办Win7开机显示正在配置Windows Update请勿关机...

    置信有很多用户都跟小编一样遇到过这样的问题&#xff0c;电脑时发现开机屏幕显现“正在配置Windows Update&#xff0c;请勿关机”(如下图所示)&#xff0c;而且还需求等大约5分钟才干进入系统。这是怎样回事呢&#xff1f;一切都是正常操作的&#xff0c;为什么开时机呈现“正…...

    2022/11/19 21:17:13
  30. 准备配置windows 请勿关闭计算机 蓝屏,Win7开机总是出现提示“配置Windows请勿关机”...

    Win7系统开机启动时总是出现“配置Windows请勿关机”的提示&#xff0c;没过几秒后电脑自动重启&#xff0c;每次开机都这样无法进入系统&#xff0c;此时碰到这种现象的用户就可以使用以下5种方法解决问题。方法一&#xff1a;开机按下F8&#xff0c;在出现的Windows高级启动选…...

    2022/11/19 21:17:12
  31. 准备windows请勿关闭计算机要多久,windows10系统提示正在准备windows请勿关闭计算机怎么办...

    有不少windows10系统用户反映说碰到这样一个情况&#xff0c;就是电脑提示正在准备windows请勿关闭计算机&#xff0c;碰到这样的问题该怎么解决呢&#xff0c;现在小编就给大家分享一下windows10系统提示正在准备windows请勿关闭计算机的具体第一种方法&#xff1a;1、2、依次…...

    2022/11/19 21:17:11
  32. 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”的解决方法...

    今天和大家分享一下win7系统重装了Win7旗舰版系统后&#xff0c;每次关机的时候桌面上都会显示一个“配置Windows Update的界面&#xff0c;提示请勿关闭计算机”&#xff0c;每次停留好几分钟才能正常关机&#xff0c;导致什么情况引起的呢&#xff1f;出现配置Windows Update…...

    2022/11/19 21:17:10
  33. 电脑桌面一直是清理请关闭计算机,windows7一直卡在清理 请勿关闭计算机-win7清理请勿关机,win7配置更新35%不动...

    只能是等着&#xff0c;别无他法。说是卡着如果你看硬盘灯应该在读写。如果从 Win 10 无法正常回滚&#xff0c;只能是考虑备份数据后重装系统了。解决来方案一&#xff1a;管理员运行cmd&#xff1a;net stop WuAuServcd %windir%ren SoftwareDistribution SDoldnet start WuA…...

    2022/11/19 21:17:09
  34. 计算机配置更新不起,电脑提示“配置Windows Update请勿关闭计算机”怎么办?

    原标题&#xff1a;电脑提示“配置Windows Update请勿关闭计算机”怎么办&#xff1f;win7系统中在开机与关闭的时候总是显示“配置windows update请勿关闭计算机”相信有不少朋友都曾遇到过一次两次还能忍但经常遇到就叫人感到心烦了遇到这种问题怎么办呢&#xff1f;一般的方…...

    2022/11/19 21:17:08
  35. 计算机正在配置无法关机,关机提示 windows7 正在配置windows 请勿关闭计算机 ,然后等了一晚上也没有关掉。现在电脑无法正常关机...

    关机提示 windows7 正在配置windows 请勿关闭计算机 &#xff0c;然后等了一晚上也没有关掉。现在电脑无法正常关机以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容&#xff0c;让我们赶快一起来看一下吧&#xff01;关机提示 windows7 正在配…...

    2022/11/19 21:17:05
  36. 钉钉提示请勿通过开发者调试模式_钉钉请勿通过开发者调试模式是真的吗好不好用...

    钉钉请勿通过开发者调试模式是真的吗好不好用 更新时间:2020-04-20 22:24:19 浏览次数:729次 区域: 南阳 > 卧龙 列举网提醒您:为保障您的权益,请不要提前支付任何费用! 虚拟位置外设器!!轨迹模拟&虚拟位置外设神器 专业用于:钉钉,外勤365,红圈通,企业微信和…...

    2022/11/19 21:17:05
  37. 配置失败还原请勿关闭计算机怎么办,win7系统出现“配置windows update失败 还原更改 请勿关闭计算机”,长时间没反应,无法进入系统的解决方案...

    前几天班里有位学生电脑(windows 7系统)出问题了&#xff0c;具体表现是开机时一直停留在“配置windows update失败 还原更改 请勿关闭计算机”这个界面&#xff0c;长时间没反应&#xff0c;无法进入系统。这个问题原来帮其他同学也解决过&#xff0c;网上搜了不少资料&#x…...

    2022/11/19 21:17:04
  38. 一个电脑无法关闭计算机你应该怎么办,电脑显示“清理请勿关闭计算机”怎么办?...

    本文为你提供了3个有效解决电脑显示“清理请勿关闭计算机”问题的方法&#xff0c;并在最后教给你1种保护系统安全的好方法&#xff0c;一起来看看&#xff01;电脑出现“清理请勿关闭计算机”在Windows 7(SP1)和Windows Server 2008 R2 SP1中&#xff0c;添加了1个新功能在“磁…...

    2022/11/19 21:17:03
  39. 请勿关闭计算机还原更改要多久,电脑显示:配置windows更新失败,正在还原更改,请勿关闭计算机怎么办...

    许多用户在长期不使用电脑的时候&#xff0c;开启电脑发现电脑显示&#xff1a;配置windows更新失败&#xff0c;正在还原更改&#xff0c;请勿关闭计算机。。.这要怎么办呢&#xff1f;下面小编就带着大家一起看看吧&#xff01;如果能够正常进入系统&#xff0c;建议您暂时移…...

    2022/11/19 21:17:02
  40. 还原更改请勿关闭计算机 要多久,配置windows update失败 还原更改 请勿关闭计算机,电脑开机后一直显示以...

    配置windows update失败 还原更改 请勿关闭计算机&#xff0c;电脑开机后一直显示以以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容&#xff0c;让我们赶快一起来看一下吧&#xff01;配置windows update失败 还原更改 请勿关闭计算机&#x…...

    2022/11/19 21:17:01
  41. 电脑配置中请勿关闭计算机怎么办,准备配置windows请勿关闭计算机一直显示怎么办【图解】...

    不知道大家有没有遇到过这样的一个问题&#xff0c;就是我们的win7系统在关机的时候&#xff0c;总是喜欢显示“准备配置windows&#xff0c;请勿关机”这样的一个页面&#xff0c;没有什么大碍&#xff0c;但是如果一直等着的话就要两个小时甚至更久都关不了机&#xff0c;非常…...

    2022/11/19 21:17:00
  42. 正在准备配置请勿关闭计算机,正在准备配置windows请勿关闭计算机时间长了解决教程...

    当电脑出现正在准备配置windows请勿关闭计算机时&#xff0c;一般是您正对windows进行升级&#xff0c;但是这个要是长时间没有反应&#xff0c;我们不能再傻等下去了。可能是电脑出了别的问题了&#xff0c;来看看教程的说法。正在准备配置windows请勿关闭计算机时间长了方法一…...

    2022/11/19 21:16:59
  43. 配置失败还原请勿关闭计算机,配置Windows Update失败,还原更改请勿关闭计算机...

    我们使用电脑的过程中有时会遇到这种情况&#xff0c;当我们打开电脑之后&#xff0c;发现一直停留在一个界面&#xff1a;“配置Windows Update失败&#xff0c;还原更改请勿关闭计算机”&#xff0c;等了许久还是无法进入系统。如果我们遇到此类问题应该如何解决呢&#xff0…...

    2022/11/19 21:16:58
  44. 如何在iPhone上关闭“请勿打扰”

    Apple’s “Do Not Disturb While Driving” is a potentially lifesaving iPhone feature, but it doesn’t always turn on automatically at the appropriate time. For example, you might be a passenger in a moving car, but your iPhone may think you’re the one dri…...

    2022/11/19 21:16:57