JavaScript getter setter金字塔​​​​​​​

目录
  • 引言
  • 函数
  • getters
  • setters
  • getter-getters
  • setter-setter
  • iterables
  • promises
  • Observables
    • GetterSetterFunctionValue
  • async iterables
  • operators

引言

英文链接: staltz.com/javascript-…

函数是JavaScript的基石。 它是一种灵活的、抽象的,可作为其他抽象的基础, 如:Promise、Iterables、Observables等。 之前我在会议或研讨会中传递这些概念,随着时间的推移,我发现这些抽象的概念可以抽象为金字塔状排列的模型,在本文中,将介绍金字搭中的各层。

函数

JavaScript的基础是一级值,如: numbers、strings、objects、booleans等。 虽然,您可以只使用这些值和控制流(if/else/for等)来编写程序,但很快您可能需要编写一个新的函数来改进您的程序。

JavaScript中必须通过函数进行抽象,像异步I/O操作等通常都必须使用callback回调函数。 JavaScript中的函数,与【函数式编程】中的纯函数不同,建议最好将它们简单理解为“流程”,因为它只得惰性的可重用代码块,具有可选的输入(参数)和可选的输出(返回值)

与硬编码相比,函数有以下几个重要的好处:

  • 懒惰/可重用性, 函数内的代码必须是惰性的(即除非被调用,否则不会执行)才能使其可重用
  • 实现的灵活性,函数的使用者并不关心函数内部是如何实现的,这意味着可以各种方式灵活的实现函数。

getters

getter是一种不用传递任何参数但需要有返回值的函数。
函数定义: () => X

getter是一种不用传递任何参数但需要有返回值的函数。 JavaScript中有许多这样的getter, 例如: Math.random()、Date.now()等

getter作为值的抽象也很有用,例如:user 与 getUser

const user = {name: 'Alice', age:30};
console.log(user.name); // Alice
function getUser(){
	return {name: 'Alice', age:30};
}
console.log(getUser().name); // Alice

通过使用getter, 我们可以利用函数的一个好处: 惰性. 即我们不调用 getUser()user对象就不会被白白创建。 同时,我们利用了函数的另一好处:实现的灵活性, 因为我们可以通过多种不同的方式来返回对象。

getter还允许我们对副作用有一个钩子,当执行getter时,我们可以触发有用的副作用函数,比如console.log输出日志 或是 触发Analytics事件等,例如:

function getUser(){
	Analytics.sendEvent('User object is now being accessed');
	return {name:'Alice', age:30};
}

getter上的计算,也是可以抽象的。例如:

function add(getX, getY){
	return function getZ(){
		const x = getX();
		const y = getY();
		return x + y;
	}
}

当getter返回不可预测的值时,这种抽象计算的好处更加明显,例如:使用getter添加Math.random:

const getTen = () => 10;
const getTenPlusRandom = add(getTen, Math.random);

console.log(getTenPlusRandom()); // 10.948117215055046
console.log(getTenPlusRandom()); // 10.796721274448556
console.log(getTenPlusRandom()); // 10.15350303918338
console.log(getTenPlusRandom()); // 10.829703269933633

比较常见的是getter与Promise一起搭配使用,因为Promise是不可重用的计算,因此,将Promise的构造函数包装在一个getter中(如我们熟知的"工厂函数"或"thunk")使其可重用。

setters

setters是有一个或多个输入参数,但没有返回值的函数。
函数定义: X => ()

setters是有一个或多个输入参数,但没有返回值的函数。 在JavaScript运行时或DOM中有许多这样的setters, 比如: console.log(x), document.write(x)等。

与getter不同,setter通常不用抽象。 因为函数没有返回值,这意味着函数仅用于发送数据或执行JavaScript命令等。 例如, 上文中的getter getTen是数字10的抽象,我们可以将它作为值进行传递,但如果将函数setTen作为值传递是没有意义的,因为它没有返回值。

换句话说,setter可以是其他setter的简单包装器。例如:简单包装下setter console.log:

function fancyConsoleLog(str) {
  console.log(' ' + str + ' ');
}

getter-getters

getter-getters是一种不需要输入参数且返回一个getter的函数。
函数定义: () => (() => X)

“getter-getter”: 一种特殊类型的getter,其返回值是一个getter.

对getter的原始需求是使用getter的返回值进行迭代。
例如,我们想显示二次幂的数字序列,我们可以使用getter getNextPowerOfTwo()

let i = 2;
function getNextPowerOfTwo() {
  const next = i;
  i = i * 2;
  return next;
}
console.log(getNextPowerOfTwo()); // 2
console.log(getNextPowerOfTwo()); // 4
console.log(getNextPowerOfTwo()); // 8
console.log(getNextPowerOfTwo()); // 16
console.log(getNextPowerOfTwo()); // 32
console.log(getNextPowerOfTwo()); // 64
console.log(getNextPowerOfTwo()); // 128

在上面示例代码中,变量i是全局声明的,若我们想重新迭代序列,则必须要重置变量i, 从而泄漏了getter的实现细节。

若想将上述代码变为可重用且不包含全局变量,我们需要将getter封装在另一个函数中,这个包装函数也是一个getter.

function getGetNext() {
  let i = 2;
  return function getNext() {
    const next = i;
    i = i * 2;
    return next;
  }
}

let getNext = getGetNext();
console.log(getNext()); // 2
console.log(getNext()); // 4
console.log(getNext()); // 8
getNext = getGetNext(); //  restart!
console.log(getNext()); // 2
console.log(getNext()); // 4
console.log(getNext()); // 8
console.log(getNext()); // 16
console.log(getNext()); // 32

可以看出,getter-getter是一种特殊类型的getter, 它继承了getter的所有优点,例如:

  • 实现的灵活性
  • 用于副作用的钩子(hook)
  • 惰性,其惰性体现在函数在初始化时,外部函数启用延迟初始化,而内部函数启用值时延迟初始化。
function getGetNext() {
  //  LAZY INITIALIZATION
  let i = 2;

  return function getNext() {
    //  LAZY ITERATION
    const next = i;
    i = i * 2;
    return next;
  }
}

setter-setter

setter-setter是一种以setter作为输入且无返回值的函数。 函数定义: (X => ()) => ()

setter-setter是一种特殊的setter函数,其中传递的参数也是setter。虽然基本的setter不是抽象函数,但setter-setter是能够表示可在代码库中传递的值的抽象。 例如下列中,通过这个setter来表示数字10:

function setSetTen(setTen) {
  setTen(10)
}

注意: setter没有返回值。

通过简单地重构,让上述示例代码的可读性更强:

function setTenListener(cb) {
  cb(10)
}

cb,即callback缩写,名为“回调”,它表明在JavaScript中如何使用setter-setter

在JavaScript中,有大量回调的案例。 顾名思义,cb代表“回调”,通过上述示例代码来说明,cb在setter-setter如何使用。

下面两个示例,在功能上是等价的,但是调用方式不同。

setSetTen(console.log);

// compare with...

console.log(getTen());

setter-setter具有与getter-getter相同的好处:

  • 实现的灵活性
  • 用于副作用的钩子(hook)
  • 惰性,其惰性体现在函数在初始化时,外部函数启用延迟初始化,而内部函数启用值时延迟初始化。

同时,它还有两个getter-getter所没有的新特性:

  • 控制反转
  • 异步性

在上面的示例中,使用getter的代码指示何时使用“console.log”使用getter。
然而,当使用setter-setter时,setter-seter本身决定何时调用“console.log”。 这种控制反转允许setter-setter拥有比getter更大的能力,例如通过向回调函数传递许多值, 例如:

function setSetTen(setTen) {
  setTen(10)
  setTen(10)
  setTen(10)
  setTen(10)
}

控制反转还可以使setter决定何时向回调传递值,例如: 异步传递。 试想一下,若将“setSetTen”的更换为“setTenListener”呢?

function setTenListener(cb) {
  setTimeout(() => { cb(10); }, 1000);
}

setter-setter在JavaScript异步编程中的很常见,但回调函数不一定是异步的。
例如,下面的“setSetTen”示例中,它是同步的:

function setSetTen(setTen) {
  setTen(10)
}

console.log('before');
setSetTen(console.log);
console.log('after');

// (Log shows:)
// before
// 10
// after

iterables

iterable是省略了实现细节的getter-getter, 其返回值是一个描述型的对象值或完成时的对象。

An iterable is (with some details omitted:) a getter-getter of an object that describes either a value or completion

函数定义: () => (() => ({done, value}))

getter-getter可以重启序列值,但它没有通知序列何时结束的约定。
Iterables是一种特殊的getter-getter,它的返回值中始终有2个属性对象:

  • done,布尔值,表示是否完成
  • value,任意值,实际传递的值(在done的值不为true时)

完成指示符done,能够使iterable的代码在执行时,知道后续GET将返回无效数据,因此iterable知道何时停止迭代。
在下面的示例中,我们可以根据完成指示符生成一个偶数范围为40到48的有限序列:

function getGetNext() {
  let i = 40;
  return function getNext() {
    if (i <= 48) {
      const next = i;
      i += 2;
      return {done: false, value: next};
    } else {
      return {done: true};
    }
  }
}

let getNext = getGetNext();
for (let result = getNext(); !result.done; result = getNext()) {
  console.log(result.value);
}

ES6 Iterables除了简单的() => (() => ({done, value}))模式之外,还有更多的约定。它们在每个getter上添加了一个包装器对象:

  • 外部getter包含对象:{[Symbol.iterator]:f}
  • 内部getter包含对象: {next:g}

以下是与上一个示例代码功能等价的ES6 Iterable代码:

const oddNums = {
  [Symbol.iterator]: () => {
    let i = 40;
    return {
      next: () => {
        if (i <= 48) {
          const next = i;
          i += 2;
          return {done: false, value: next};
        } else {
          return {done: true};
        }
      }
    }
  }
}
let iterator = oddNums[Symbol.iterator]();
for (let result = iterator.next(); !result.done; result = iterator.next()) {
  console.log(result.value);
}

注意,两个示例间实现的差异点:

-function getGetNext() {
+const oddNums = {
+  [Symbol.iterator]: () => {
     let i = 40;
-  return function getNext() {
+    return {
+      next: () => {
         if (i <= 48) {
           const next = i;
           i += 2;
           return {done: false, value: next};
         } else {
           return {done: true};
         }
       }
+    }
   }
+}

-let getNext = getGetNext();
-for (let result = getNext(); !result.done; result = getNext()) {
+let iterator = oddNums[Symbol.iterator]();
+for (let result = iterator.next(); !result.done; result = iterator.next()) {
  console.log(result.value);
}

ES6提供了语法糖for let以便快速迭代对象:

for (let x of oddNums) {
  console.log(x);
}

为了方便创建Iterables,ES6还提供了生成器函数语法糖 function*

function* oddNums() {
  let i = 40;
  while (true) {
    if (i <= 48) {
      const next = i;
      i += 2;
      yield next;
    } else {
      return;
    }
  }
}

使用生产端语法糖和**消费端语法糖*,自2015年以来,可迭代函数是JavaScript中可完成值序列的易于使用的抽象。请注意,调用生成器函数将返回可迭代函数,生成器函数本身不是可迭代函数:

自2015年后,更易于使用iterables来抽象可序列化对象值。 需要注意的是,调用生成器函数将返回一个可迭代对象,生成器函数本身不可迭代。

function* oddNums() {
  let i = 40;
  while (true) {
    if (i <= 48) {
      yield i;
      i += 2;
    } else {
      return;
    }
  }
}

for (let x of oddNums()) {
  console.log(x);
}

promises

Promise是一个(省略了一些细节)特殊的setter, 它包含2个setter, 并且还有额外的保障。
函数定义: (X => (), Err => ()) => ()

虽然setter-setter功能强大,但由于控制反转,它们可能变得非常不可预测。

  • 它们可以是同步或异步的
  • 也可以随时间传递零或一个或多个值。

Promise是一种特殊的setter,它为返回值提供一些特殊的保证:

  • 内部setter(“回调”)从不同步调用
  • 内部setter最多调用一次
  • 提供了一个可选的第二setter,用于传递错误值

将下面的setter与等价的Promise代码比较,可以看出:

  • Promise只提供一次值,且不会在两个console.log输出
  • Promise的返回值是异步返回的:

setter-setter的实现:

function setSetTen(setTen) {
  setTen(10)
  setTen(10)
}

console.log('before setSetTen');
setSetTen(console.log);
console.log('after setSetTen');

// (Log shows:)
// before setSetTen
// 10
// 10
// after setSetTen

promise的实现:

const tenPromise = new Promise(function setSetTen(setTen) {
  setTen(10);
  setTen(10);
});

console.log('before Promise.then');
tenPromise.then(console.log);
console.log('after Promise.then');

// (Log shows:)
// before Promise.then
// after Promise.then
// 10

Promise可以方便的用于表示:一个异步且只返回一次的值。 在ES2017以后,可以使用async - await语法糖来编写Promise。 在函数前使用async关键字,在需要使用Promise值的位置使用await来接收值。

async function main() {
  console.log('before await');
  const ten = await new Promise(function setSetTen(setTen) {
    setTen(10);
  });
  console.log(ten);
  console.log('after await');
}

main();

// (Log shows:)
// before await
// 10
// after await

语法糖async - await也可用于创建Promise, 因为async函数将返回一个Promise对象,该Promise对象将返回return的值。

async function getTenPromise() {
  return 10;
}
const tenPromise = getTenPromise();

console.log('before Promise.then');
tenPromise.then(console.log);
console.log('after Promise.then');

// (Log shows:)
// before Promise.then
// after Promise.then
// 10

Observables

observable是(省略了一些细节:)一个包含有3个setter的setter函数,并带有额外的保障。
函数定义: (X => (), Err => (), () => ()) => ()

GetterSetterFunctionValue

像Iterables一样,GetterSetterFunctionValue是一种特殊类型的getter-getter,它增加了发送完成信号的能力,Observable也是一种setter-setter,它也增加了完成能力。
JavaScript中的典型setter-setter,如: element.addEventListener不会通知事件流是否完成,因此很难连接事件流或执行其他与完成相关的逻辑。

与JavaScript规范中标准化的可伸缩性不同,可观测性是在几个库中找到的松散约定,如:

  • RxJS
  • most.js
  • xstream
  • Bacon.js

尽管正在考虑将proposal-observable作为一个TC39方案,但该方案仍在不断修改。
因此在本文中,让我们假设遵循Fantasy Observable规范(其中RxJS、most.js和xstream等最受欢迎库都遵循这一规范)。

Observables the dual of Iterables,有以下特性:

可迭代:

  • 它是一个对象
  • 可迭代,有一个“iterate”属性方法,即:Symbol.iterator
  • “iterate” 方法是迭代器对象的 getter
  • 迭代器对象有一个名为next的 getter

可观察:

  • 它是一个对象
  • 有可观察method,即:subscribe
  • 可观察method是Observer对象的setter
  • Observer有一个名为nextsetter

Observer对象还可以包含另外两个方法:

  • complete,表示成功完成,相当于iterator中的“done”指示符
  • error,表示失败完成,相当iterator中执行时引发异常

与Promise一样,Observable为返回值增加了一些保障:

  • 一旦调用了complete,则不会调用error
  • 一旦调用了error,就不会调用complete
  • 一旦调用了complete或error,则不会调用next

在下面的示例中,Observable用于异步返回有限的数字序列

const oddNums = {
  subscribe: (observer) => {
    let x = 40;
    let clock = setInterval(() => {
      if (x <= 48) {
        observer.next(x);
        x += 2;
      } else {
        observer.complete();
        clearInterval(clock);
      }
    }, 1000);
  }
};

oddNums.subscribe({
  next: x => console.log(x),
  complete: () => console.log('done'),
});

// (Log shows:)
// 40
// 42
// 44
// 46
// 48
// done

与setter-setter一样,Observable具有控制反转的能力,因此消费端(oddNums.subscribe)无法暂停或取消传入的数据流。
大多数Observable的实现都添加了一个重要的细节,允许消费者取消订阅: unsubscribe

“subscribe”函数将返回一个对象subscription,即“unsubscribe”。 消费端可以使用该方法取消订阅。
因此,“subscribe”不再是setter,因为它是一个具有输入(观察者)和输出(订阅)的函数。

下面,我们在前面的示例中添加了一个订阅对象:

const oddNums = {
  subscribe: (observer) => {
    let x = 40;
    let clock = setInterval(() => {
      if (x <= 48) {
        observer.next(x);
        x += 2;
      } else {
        observer.complete();
        clearInterval(clock);
      }
    }, 1000);
    //  Subscription:
    return {
      unsubscribe: () => {
        clearInterval(clock);
      }
    };
  }
};

const subscription = oddNums.subscribe({
  next: x => console.log(x),
  complete: () => console.log('done'),
});

//  Cancel the incoming flow of data after 2.5 seconds
setTimeout(() => {
  subscription.unsubscribe();
}, 2500);

// (Log shows:)
// 40
// 42

async iterables

异步可迭代函数(省略了一些细节)类似于yield promise的可迭代函数
函数定义: () => (() => Promise<{done, value}>)

Iterables可以表示任何无限或有限的值序列,但它们有一个限制:

  • 只要使用者调用next()方法,返回值就必须是同步的。

AsyncIterables扩展了Iterables的功能,它通过允许延后返回值,而不是在调用时立即返回。
AsyncIterables通过使用Promise实现值的异步传递,因为Promise表示单个异步值。每次调用迭代器的next()(内部getter函数),都会创建并返回一个Promise。 在下面的示例中,我们以oddNums可迭代为例,其在迭代时每次返回一个Promise对象:

function slowResolve(val) {
  return new Promise(resolve => {
    setTimeout(() => resolve(val), 1000);
  });
}

function* oddNums() {
  let i = 40;
  while (true) {
    if (i <= 48) {
      yield slowResolve(i); //  yield a Promise
      i += 2;
    } else {
      return;
    }
  }
}

要使用Asynciteable,我们可以使用async - await语法糖,来遍历&获取异步迭代器的值:

async function main() {
  for (let promise of oddNums()) {
    const x = await promise;
    console.log(x);
  }
  console.log('done');
}
main();
// (Log shows:)
// 40
// 42
// 44
// 46
// 48
// done

虽然上例中通过async - await语法糖(ES6语法)能够编写出可读性较好的代码。但是,在ES2018中,异步迭代是通过{done,value}(Promise)对象来迭代的。

比较一下ES6&ES2018异步迭代器的函数定义:

  • ES6异步迭代器: () => (() => {done, value: Promise<X>})
  • ES2018异步迭代器: () => (() => Promise<{done, value}>)

可以看出,ES2018异步可迭代函数返回的不是一个可迭代函数,它将返回一个Promise对象,但在许多方面与ES6异步迭代器相似。
这一细节的原因是,异步可伸缩性还需要允许异步发送完成(done),因此Promise必须包裹{done,value}对象。

因为AsyncIterables不是Iterables,所以它们使用不同的符号。

  • Iterables 使用Symbol.iterator
  • AsyncIterables 使用'Symbol.asyncItrator'

在下面的示例中,我们使用ES2018 AsyncIterable实现了一个前例等价的功能:

const oddNums = {
  [Symbol.asyncIterator]: () => {
    let i = 40;
    return {
      next: () => {
        if (i <= 48) {
          const next = i;
          i += 2;
          return slowResolve({done: false, value: next});
        } else {
          return slowResolve({done: true});
        }
      }
    };
  }
};

async function main() {
  let iter = oddNums[Symbol.asyncIterator]();
  let done = false;
  for (let promise = iter.next(); !done; promise = iter.next()) {
    const result = await promise;
    done = result.done;
    if (!done) console.log(result.value);
  }
  console.log('done');
}
main();

像Iterables有语法糖function*for–let–of,以及Promises有async–await语法糖一样,ES2018中的AsyncIterable有两个语法糖功能:

  • 生产端:async function*
  • 消费端:for–await–let–of

在下面的示例中,我们使用这两种功能创建异步数字序列,并使用for-await循环遍历它们:

function sleep(period) {
  return new Promise(resolve => {
    setTimeout(() => resolve(true), period);
  });
}

//  Production side can use both `await` and `yield`
async function* oddNums() {
  let i = 40;
  while (true) {
    if (i <= 48) {
      await sleep(1000);
      yield i;
      i += 2;
    } else {
      await sleep(1000);
      return;
    }
  }
}

async function main() {
  //  Consumption side uses the new syntax `for await`
  for await (let x of oddNums()) {
    console.log(x);
  }
  console.log('done');
}

main();

尽管它们是新特性,但Babel、TypeScript、Firefox、Chrome、Safari 和 Node.js中都已经支持了。

AsyncIterables可以方便地结合基于Promise的API(如fetch)来创建异步序列。 例如在数据库中列出用户,每次请求一个用户:

async function* users(from, to) {
  for (let x = from; x <= to; x++) {
    const res = await fetch('http://jsonplaceholder.typicode.com/users/' + x);
    const json = await res.json();
    yield json;
  }
}

async function main() {
  for await (let x of users(1, 10)) {
    console.log(x);
  }
}

main();

operators

在本文中,抽象的列举了一些特殊的JavaScript函数示例。 根据定义,它们的功能的不可能比函数更强大。因此,函数是最强大、最灵活的抽象。完全灵活性的缺点是不可预测性,这些抽象提供的是“保证”,基于这些“保证”,让您可以编写更有组件、更可预测的代码。

另一方向, 函数可以作为JavaScript的值,允许将它进行传递和操纵。
在本文中,我们将Iterable、Observable、AsyncIterable作为值进行传递,并在传递过程中对其进行操作。

最常见的操作之一是map映射,它在数组中经常使用到,但也与其他抽象相关。 在下面的示例中,我们为AsyncIterables创建map操作符,并使用它用来创建异步迭代器,以获取数据库的用户信息。

async function* users(from, to) {
  for (let i = from; i <= to; i++) {
    const res = await fetch('http://jsonplaceholder.typicode.com/users/' + i);
    const json = await res.json();
    yield json;
  }
}

//  Map operator for AsyncIterables
async function* map(inputAsyncIter, f) {
  for await (let x of inputAsyncIter) {
    yield f(x);
  }
}

async function main() {
  const allUsers = users(1, 10);
  //  Pass `allUsers` around, create a new AsyncIterable `names`
  const names = map(allUsers, user => user.name);
  for await (let name of names) {
    console.log(name);
  }
}

main();

上面的示例代码,没有被抽象至Getter - Setter金字塔中。
若使用getter - setter来实现,需要更多的代码,可读性也不好。
因此,在不牺牲可读性的情况下,使用运算符和新的语法糖,用更少的代码编写函数做更多的事情,以处理这种特殊的场景。

到此这篇关于JavaScript getter setter金字塔的文章就介绍到这了,更多相关JavaScript getter-setter金字塔内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • JavaScript对象访问器Getter及Setter原理解析

    在ES5中引入了set和get关键字,允许访问和设置对象的属性. set关键字: set关键字接收一个参数,用于修改对象的属性值:个人觉得类似于在对象内部的每一个方法前都隐藏一个set. get关键字是用来在对象外部获取对象的属性值或方法:在对象外部可以使用set关键字修改对象的属性值. 使用原型方法和set.get关键字添加或修改对象的属性或方法: 以上观点是自己不成熟的一些看法,不当之处请大家批评指正. 以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们.

  • js中getter和setter用法实例分析

    本文实例讲述了js中getter和setter用法.分享给大家供大家参考,具体如下: 在学习Vue计算属性时,有一句"计算属性默认只有 getter ,不过在需要时你也可以提供一个 setter". getter和setter到底是什么?于是我查找了资料: 在Es5中可以使用getter和setter部分改写默认操作,但是只能应用在单个属性上,无法应用在整个对象上.getter是一个隐藏函数,会在获取属性值时调用.setter也是一个隐藏属性,会在设置属性值时调用. 例子: var m

  • 浅谈JS对象添加getter与setter的5种方法

    定义 getter 与 setter 1.通过对象初始化器在创建对象的时候指明(也可以称为通过字面值创建对象时声明) (function () { var o = { a : 7, get b(){return this.a +1;},//通过 get,set的 b,c方法间接性修改 a 属性 set c(x){this.a = x/2} }; console.log(o.a); console.log(o.b); o.c = 50; console.log(o.a); })(); 在 chro

  • JavaScript中setter和getter方法介绍

    javascript中的setter.getter是平时接触比较少的方法,其本身也并不是标准方法,只在非ie浏览器里支持(ie内核也许有其他方法可以做到呢?暂时不知其解),但是加以利用可以做许多事情,比如: 1.对数据的访问限制: a.value是对value变量的getter方法调用,如果在getter方法实现中抛出异常,可以阻止对value变量的访问 2.对dom变量进行监听: window.name是一个跨域非常好用的dom属性(大名鼎鼎,详见百度),如果覆盖window.name的set

  • 谈谈因Vue.js引发关于getter和setter的思考

    起因 当我打印出Vue实例下的data对象里的属性时,发现了一个有趣的事情: 它的每个属性都有两个相对应的get和set方法,我觉的这是多此一举的,于是去网上查了查Vue双向绑定的实现原理,才发现它和Angular.js双向绑定的实现原理完全不同,Angular是用的数据脏检测,当Model发生变化,会检测所有视图是否绑定了相关数据,再更改视图.而Vue使用的发布订阅模式,是点对点的绑定数据. Vue的数据绑定只有两个步骤,compile=>link. 我一直在想,vue是通过什么去监听用户对M

  • JavaScript之Getters和Setters 平台支持等详细介绍

    来自John Resig早年的文章,大致翻译了一下,以作备忘. 令人高兴的是,我想我终于可以说,"现在,JavaScript的Getters和Setters使用非常广泛,它和每个JavaScript开发者的切身利益息息相关".靠,我为了说这句话已经等了很久了. 首先,我们先来快速了解什么是Getters和Setters,以及它们为什么很有用.然后,我们来看看现在都有哪些平台支持Gettets和Setters. Getters和Setters Getters和Setters使你可以快速获

  • JavaScript的setter与getter方法

    以前在写项目过程一直都没有使用过Javascript的setter与getter方法,所以对其是一种要懂不懂的概念:今天看书看到这个知识点,还是模模糊糊的,于是就打算研究研究: Javascript对象的属性是由名字,值和一组特性构成的.那么首先,来了解一下对象的两种属性: 数据属性,我们经常使用,应该很熟悉 访问器属性,也称存取器属性 何为存取器属性?就是一组获取和设置值的函数.在ECMAScript5中,属性值可以用一个或两个方法设置,这两个方法就是getter和setter:因此gette

  • Javascript中的getter和setter初识

    前言 本文主要给大家介绍的关于Javascript中getter和setter的相关内容,第一次听说这个东西的时候是vue.js里面的数据绑定,只要绑定了数据,修改对象属性可以自动反馈到dom上,很神奇,后面也看到了文档里面实现是对对象定义了getter和setter并覆盖原属性,索性就来总结这两者的用法,下面话不多说了,来一起看看详细的介绍吧. 原理 利用Object.defineProperty来重写对象属性为getter和setter,通过getter和setter顺便改变绑定DOM节点的

  • javascript中的__defineGetter__和__defineSetter__介绍

    Getter是一种获取一个属性的值的方法,Setter是一种设置一个属性的值的方法.可以为任何预定义的核心对象或用户自定义对象定义getter和setter方法,从而为现有的对象添加新的属性. 有两种方法来定义Getter或Setter方法: 1.在对象初始化时定义 2.在对象定义后通过Object的__defineGetter__.__defineSetter__方法来追加定义 在使用对象初始化过程来定义Getter和Setter方法时唯一要做的事情就是在getter方法前面加上"get&qu

  • JavaScript getter setter金字塔​​​​​​​

    目录 引言 函数 getters setters getter-getters setter-setter iterables promises Observables GetterSetterFunctionValue async iterables operators 引言 英文链接: staltz.com/javascript-… 函数是JavaScript的基石. 它是一种灵活的.抽象的,可作为其他抽象的基础, 如:Promise.Iterables.Observables等. 之前我在

  • Vue 理解之白话 getter/setter详解

    当你把一个普通的 JavaScript 对象传给 Vue 实例的 data 选项,Vue 将遍历此对象所有的属性,并使用Object.defineProperty把这些属性全部转为 getter/setter.Object.defineProperty 是 ES5 中一个无法 shim 的特性,这也就是为什么 Vue 不支持 IE8 以及更低版本浏览器 以上摘自深入响应式原理 那么,把这些属性全部转为 getter/setter 具体是怎样一个过程呢?本文不深入具体,简单大致了解其过程,旨在整体

  • 详解lombok @Getter @Setter 使用注意事项

    前言:半夜睡不着觉,把星星哼成歌,只好到csdn寻找另一个梦境..... lombok是一个帮助简化代码的工具,通过注解的形式例如@Setter @Getter,可以替代代码中的getter和setter方法,虽然eclipse自带的setter.getter代码生成也不需要我们手动的去敲写,但是使用@Setter @Getter这样的注解,能够使我们的代码看上去更加的简洁.优雅. lombok下有许多常用注解,这里不过多赘述,主要讲@Getter和@Setter的一个注意点. package

  • python中@property的作用和getter setter的解释

    @property作用: python的@property是python的一种装饰器,是用来修饰方法的. 我们可以使用@property装饰器来创建只读属性,@property装饰器会将方法转换为相同名称的只读属性,可以与所定义的属性配合使用,这样可以防止属性被修改. 1.修饰方法,让方法可以像属性一样访问. class DataSet(object): @property def method_with_property(self): ##含有@property return 15 def m

  • 实现了一个PHP5的getter/setter基类的代码

    PHP3.PHP4都拥有类,但它们的类定义的实在很不像样,效率还挺难为情的,但资料上说PHP5重新构造了面向对象的支持,尽管并不是完全面向对象,但也算能拿出来见人了. 昨天晚上闲着无聊便弄起这玩意,感觉PHP5增加的类成员权限关键字挺好,但问题又来了,似乎还没一种方便的方式可以定义字段的getter以及setter,传统的方式是这样定义的: class a {     private $field;     public function get_field() { return $this->

  • Kotlin基础教程之伴生对象,getter,setter,内部,局部,匿名类,可变参数

    先来看一个名为Message的类 在这个类中有一段包含在companion object中的代码,需要说一下的是,Kotlin的class并不支持static变量,所以需要使用companion object来声明static变量,其实这个platformStatic变量也不是真正的static变量,而是一个伴生对象, 这个伴生对象位于Message类中定义的一个叫做Companion的内部类中,如图: 可以看到在Kotlin中编译器自动生成类是很常见的事情,那么这个伴生对象作何理解呢? 我的理

  • Python @property及getter setter原理详解

    @property作用: python的@property是python的一种装饰器,是用来修饰方法的. 我们可以使用@property装饰器来创建只读属性,@property装饰器会将方法转换为相同名称的只读属性,可以与所定义的属性配合使用,这样可以防止属性被修改. 1.修饰方法,让方法可以像属性一样访问. class DataSet(object): @property def method_with_property(self): ##含有@property return 15 def m

  • Yii框架getter与setter方法功能与用法分析

    本文实例讲述了Yii框架getter与setter方法功能与用法.分享给大家供大家参考,具体如下: 1.属性 在 PHP 中,类的成员变量也被称为属性.它们是类定义的一部分, 用来表现一个实例的状态(也就是区分类的不同实例). 在具体实践中,常常会想用一个稍微特殊些的方法实现属性的读写. 例如,如果有需求每次都要对 label 属性执行 trim 操作, 就可以用以下代码实现: $object->label = trim($label); 上述代码的缺点是只要修改 label 属性就必须再次调用

随机推荐