`

update

 
阅读更多

update模块用于快速作数据处理。

 

'use strict';

var _prodInvariant = require('./reactProdInvariant'),
    _assign = require('object-assign');

var invariant = require('fbjs/lib/invariant');
var hasOwnProperty = {}.hasOwnProperty;

function shallowCopy(x) {
  if (Array.isArray(x)) {
    return x.concat();
  } else if (x && typeof x === 'object') {
    return _assign(new x.constructor(), x);// 使用x对象的构造函数创建对象
  } else {
    return x;
  }
}

var COMMAND_PUSH = '$push';
var COMMAND_UNSHIFT = '$unshift';
var COMMAND_SPLICE = '$splice';
var COMMAND_SET = '$set';
var COMMAND_MERGE = '$merge';
var COMMAND_APPLY = '$apply';

var ALL_COMMANDS_LIST = [COMMAND_PUSH, COMMAND_UNSHIFT, COMMAND_SPLICE, COMMAND_SET, COMMAND_MERGE, COMMAND_APPLY];

var ALL_COMMANDS_SET = {};

ALL_COMMANDS_LIST.forEach(function (command) {
  ALL_COMMANDS_SET[command] = true;
});

function invariantArrayCase(value, spec, command) {
  // value需要为数组形式
  !Array.isArray(value) ? 
    process.env.NODE_ENV !== 'production' ? 
      invariant(false, 'update(): expected target of %s to be an array; got %s.', command, value) 
      : _prodInvariant('1', command, value) 
    : void 0;

  // spec[command]需要为数组形式
  var specValue = spec[command];
  !Array.isArray(specValue) ? 
    process.env.NODE_ENV !== 'production' ? 
      invariant(false, 'update(): expected spec of %s to be an array; got %s.' 
        + ' Did you forget to wrap your parameter in an array?', command, specValue) 
      : _prodInvariant('2', command, specValue) 
    : void 0;
}

// 配置形式快速作数据处理后返回,支持'$push'、'$unshift'、'$splice'、'$set'、'$merge'、'$apply'
// const newData = update(myData, {
//   x: {y: {z: {$set: 7}}},
//   a: {b: {$push: [9]}}
// });
function update(value, spec) {
  // spec不能是非对象
  !(typeof spec === 'object') ? 
    process.env.NODE_ENV !== 'production' ? 
      invariant(false, 
        'update(): You provided a key path to update() that did not contain one of %s. ' 
        + 'Did you forget to include {%s: ...}?', ALL_COMMANDS_LIST.join(', '), COMMAND_SET) 
      : _prodInvariant('3', ALL_COMMANDS_LIST.join(', '), COMMAND_SET) : void 0;

  // spec含$set属性时,只能接受单个属性
  if (hasOwnProperty.call(spec, COMMAND_SET)) {
    !(Object.keys(spec).length === 1) ? 
      process.env.NODE_ENV !== 'production' ? 
        invariant(false, 'Cannot have more than one key in an object with %s', COMMAND_SET) 
        : _prodInvariant('4', COMMAND_SET) : void 0;

    return spec[COMMAND_SET];
  }

  var nextValue = shallowCopy(value);

  // spec含$merge属性时,将sepc["$merge"]对象合并到value对象中
  if (hasOwnProperty.call(spec, COMMAND_MERGE)) {
    var mergeObj = spec[COMMAND_MERGE];

    // sepc["$merge"]需要为对象形式
    !(mergeObj && typeof mergeObj === 'object') ? 
      process.env.NODE_ENV !== 'production' ? 
        invariant(false, 'update(): %s expects a spec of type \'object\'; got %s', 
          COMMAND_MERGE, mergeObj) 
        : _prodInvariant('5', COMMAND_MERGE, mergeObj) 
      : void 0;

    // value需要为对象形式
    !(nextValue && typeof nextValue === 'object') ? 
      process.env.NODE_ENV !== 'production' ? 
        invariant(false, 'update(): %s expects a target of type \'object\'; got %s', 
          COMMAND_MERGE, nextValue) 
        : _prodInvariant('6', COMMAND_MERGE, nextValue) 
      : void 0;

    _assign(nextValue, spec[COMMAND_MERGE]);
  }

  // spec含$push属性时,将sepc["$push"]数组添加到value数组的尾部
  if (hasOwnProperty.call(spec, COMMAND_PUSH)) {
    invariantArrayCase(value, spec, COMMAND_PUSH);
    spec[COMMAND_PUSH].forEach(function (item) {
      nextValue.push(item);
    });
  }

  // spec含$unshift属性时,将sepc["$unshift"]数组添加到value数组的顶部
  if (hasOwnProperty.call(spec, COMMAND_UNSHIFT)) {
    invariantArrayCase(value, spec, COMMAND_UNSHIFT);
    spec[COMMAND_UNSHIFT].forEach(function (item) {
      nextValue.unshift(item);
    });
  }

  // spec含$splice属性时,对value作splice处理后返回
  if (hasOwnProperty.call(spec, COMMAND_SPLICE)) {
    // value需要为数组
    !Array.isArray(value) ? 
      process.env.NODE_ENV !== 'production' ? 
        invariant(false, 'Expected %s target to be an array; got %s', COMMAND_SPLICE, value) 
        : _prodInvariant('7', COMMAND_SPLICE, value) 
      : void 0;

    // spec["$splice"]须是数组
    !Array.isArray(spec[COMMAND_SPLICE]) ? 
      process.env.NODE_ENV !== 'production' ? 
        invariant(false, 'update(): expected spec of %s to be an array of arrays;' 
          + ' got %s. Did you forget to wrap your parameters in an array?', 
          COMMAND_SPLICE, spec[COMMAND_SPLICE]) 
        : _prodInvariant('8', COMMAND_SPLICE, spec[COMMAND_SPLICE]) 
      : void 0;


    spec[COMMAND_SPLICE].forEach(function (args) {
      // spec["$splice"]数组元素须是数组
      !Array.isArray(args) ? 
        process.env.NODE_ENV !== 'production' ? 
          invariant(false, 'update(): expected spec of %s to be an array of arrays;' 
            + ' got %s. Did you forget to wrap your parameters in an array?', 
            COMMAND_SPLICE, spec[COMMAND_SPLICE]) 
          : _prodInvariant('8', COMMAND_SPLICE, spec[COMMAND_SPLICE]) 
        : void 0;

      // 使用apply方法将args拆解为单参数传给splice方法,args须是起始项序号、删除元素个数、待添加的元素
      nextValue.splice.apply(nextValue, args);
    });
  }

  // spec含$apply属性时,调用spec["$apply"]函数处理value后返回
  if (hasOwnProperty.call(spec, COMMAND_APPLY)) {
    // spec["$apply"]须是函数
    !(typeof spec[COMMAND_APPLY] === 'function') ? 
      process.env.NODE_ENV !== 'production' ? 
        invariant(false, 'update(): expected spec of %s to be a function; got %s.', 
          COMMAND_APPLY, spec[COMMAND_APPLY]) 
        : _prodInvariant('9', COMMAND_APPLY, spec[COMMAND_APPLY]) 
      : void 0;

    nextValue = spec[COMMAND_APPLY](nextValue);
  }

  // 递归调用
  for (var k in spec) {
    if (!(ALL_COMMANDS_SET.hasOwnProperty(k) && ALL_COMMANDS_SET[k])) {
      nextValue[k] = update(value[k], spec[k]);
    }
  }

  return nextValue;
}

module.exports = update;

 

0
0
分享到:
评论

相关推荐

    FixUpdate与Update的区别1

    FixUpdate与Update的区别 在 Unity 开发中, FixUpdate 和 Update 是两个常用的函数,但它们之间存在着重要的区别。今天,我们将深入探讨 FixUpdate 和 Update 的不同之处,并了解何时使用它们。 首先,让我们从 ...

    数据库oracle for update of和for update的区别

    ### 数据库Oracle锁:FOR UPDATE OF与FOR UPDATE的区别 在Oracle数据库中,为了确保数据的一致性和准确性,尤其是在多用户环境中进行并发操作时,锁机制是必不可少的一部分。本文将详细介绍`FOR UPDATE`与`FOR ...

    Windows更新WindowsUpdate重置工具Reset-WindowsUpdate.rar

    在Windows操作系统中,Windows Update是一项重要的服务,它负责为系统提供最新的安全补丁、功能更新以及驱动程序。然而,有时Windows Update可能会遇到问题,导致更新无法正常进行或更新过程失败。在这种情况下,...

    for_update_和_for_update_nowait_的区别

    标题和描述均聚焦于Oracle数据库中`FOR UPDATE`与`FOR UPDATE NOWAIT`两种锁定机制的区别,这在并发控制和事务处理中具有重要的意义。在深入解析这两种指令之前,我们先来简要回顾一下锁定机制的基本概念。 在...

    Qt+update函数+paintEvent事件

    在Qt框架中,`update()`函数和`paintEvent()`事件是进行界面绘制和更新的核心机制。本文将深入探讨这两个概念,以及它们如何协同工作来实现动态用户界面。 首先,我们来理解`update()`函数。在Qt中,所有的窗口部件...

    windows10 wsl_update_x64.zip wsl_update_x64.exe

    标题中的“windows10 wsl_update_x64.zip”指的是Windows 10操作系统的一个更新工具,主要用于升级或安装Windows Subsystem for Linux (WSL)的更新。WSL是微软为Windows 10引入的一项功能,它允许用户在不启动完整...

    update.app格式解包工具

    在IT行业中,更新应用程序(update.app)是一种常见的软件更新打包格式,主要被用在iOS设备的系统更新中,但此格式也可能被其他平台或开发者采用。`update.app` 文件本质上是一个包含新版本软件的归档文件,用于无痛...

    WindowsUpdate修复工具.zip

    在使用Windows操作系统时,Windows Update是一项至关重要的服务,它负责为系统提供安全更新、功能增强以及驱动程序的更新。然而,有时Windows Update可能会遇到问题,导致无法正常运行。"WindowsUpdate修复工具.zip...

    Mysql Update批量更新的几种方式

    UPDATE mytable SET myfield='value' WHERE other_field='other_value'; 但是,如果你想更新多行数据,并且每行记录的各字段值都是各不一样,你会怎么办呢?刚开始你可能会想到使用循环执行多条UPDATE语句的方式,就...

    Update和Select结合使用

    为了提高效率和减少资源消耗,将`UPDATE`语句与`SELECT`语句相结合是一种非常实用的方法。这种方式可以实现更高效的数据处理,特别是在需要批量更新记录时。下面我们将深入探讨如何将`UPDATE`与`SELECT`结合使用,并...

    Windows Update Blocker一键启用或禁用Windows10更新工具

    Windows Update Blocker是一款实用工具,专门设计用于帮助用户在Windows 10操作系统上轻松地启用或禁用自动更新功能。Windows Update是微软提供的一种服务,它定期为用户提供系统补丁、安全更新以及新功能,以确保...

    vs2013 update5在线升级包

    **Visual Studio 2013 Update 5 知识点详解** Visual Studio 2013 Update 5 是微软发布的一个重要更新版本,用于增强和修复其开发环境Visual Studio 2013的功能和性能。这个在线升级包是专门为解决在使用VS2013时...

    VS2013升级到Update5的升级包

    《Visual Studio 2013升级至Update 5:开发者体验的全面提升》 Visual Studio 2013 Update 5是Microsoft对这款强大开发环境的一次重要更新,它旨在为开发者提供更为高效、便捷的工作环境,提升开发效率,并引入了一...

    wsl_update_x64 .zip

    标题 "wsl_update_x64 .zip" 暗示我们正在处理一个与Windows子系统 for Linux(WSL)更新相关的文件。这个压缩包包含了一个名为 "wsl_update_x64.msi" 的Windows安装程序文件,这通常用于在Windows操作系统上安装或...

    VOLTE呼叫建立过程中的UPDATE含义及过程分析.pdf

    在VoLTE呼叫建立过程中,UPDATE是一个重要的信令过程,它用于资源预留和媒体更新。在VoLTE呼叫建立的不同场景下,UPDATE过程承担着不同的任务和意义。 首先,当主叫方或被叫方有特定的业务如高清语音提示或彩铃时,...

    update语句的优化-oracle .pdf

    在本篇文档中,作者详细总结了在实际项目中针对Oracle数据库的update语句优化的四种方案。以下是对这四种方案的知识点进行详细的阐述: 1. 标准update语法优化: 当面对单表更新或较简单的SQL语句时,可以直接使用...

    ORACLE多表关联的update语句

    在Oracle数据库中,多表关联的UPDATE语句用于更新一个表中的数据,这些数据依赖于另一个或多个表的记录。这种操作在数据整合、错误修正或者数据同步时非常常见。以下我们将详细探讨如何使用不同方式执行多表关联的...

    vs2013_update5下载链接(请使用迅雷下载)

    Visual Studio Premium 2013 with Update 5 (x86) – DVD (Chinese-Simplified) 文件名:cn_visual_studio_premium_2013_with_update_5_x86_dvd_6815741.iso SHA1:9A2B64D73AAC916EA606B42B164A9C6E88D062D6 文件...

    oracle的update的五种方式

    标准 Update 语句是 Oracle 中最基本的更新语句,语法为:`UPDATE 表名称 SET 列名称 = 新值 WHERE 列名称 = 某值`。例如:`UPDATE t_join_situation SET join_state='1' WHERE year='2011'`。这种方式适用于更新...

    oracle执行update语句时卡住问题分析及解决办法

    这种只有update无法执行其他语句可以执行的其实是因为记录锁导致的,在oracle中,执行了update或者insert语句后,都会要求commit,如果不commit却强制关闭连接,oracle就会将这条提交的记录锁住。由于我的java程序...

Global site tag (gtag.js) - Google Analytics