内容要点:

   和Lisp、Haskell不同,JS并非函数式编程语言,但在JS中可以像操控对象一样操控函数,

  也就是说可以在JS中应用函数式编程技术。ES5中的数组方法(诸如map()和reduce())就可以非常适用于函数式编程风格。

一.使用函数处理数组

   假设有一个数组,数组元素都是数字,我们想要计算这些元素的平均值和标准差。

   若用非函数式编程风格的话,代码是这样:

        var data = [1,1,3,5,5]; //这里是待处理的数组

       //平均数是所有元素的累加和值除以元素个数

        var total = 0;

        for(var i=0;i<data.length;i++) total +=data[i];

        var mean = total/data.length; //平均数是3

      //计算标准差,首先计算每个数据减去平均数之后偏差的平方然后求和

        total = 0;

        for(var i=0;i<data.length;i++){

             var deviation = data[i] - mean;

            total +=deviation*deviation;

         }

        var stddev = Math.sqrt(total/(data.length-1)); //标准差的值是2

      可以使用数组方法map()和reduce()来实现同样的计算,

          //首先定义两个简单的函数

           var sum = function(x,y){ return x+y; };

           var square = function(x){ return x*x; };

         //然后将这些函数和数组方法配合使用计算平均数和标准差

           var data = [1,1,3,5,5];

           var mean = data.reduce(sum)/data.length;

           var deviations = data.map(function(x){ return x-mean; });

           var stddev = Math.sqrt(deviations.map(square).reduce(sum)/(data.length-1));

           console.log(mean); //3
           console.log(deviations); //[-2,-2,0,2,2]
           console.log(deviations.map(square));//[4, 4, 0, 4, 4]
           console.log(deviations.map(square).reduce(sum)); //16
           console.log(stddev); //2

         ES3并不包含这些数组方法,如果不存在内置方法的话我们可以自定义map()和reduce()函数:

            //对于每个数组元素调用函数f(),并返回一个结果数组。如果Array.prototype.map定义了的话,就使用这个方法。

              var map = Array.prototype.map

                             ?  function(a,f){ return a.map(f); }

                             :   function(a,f){

                                     var result = [];

                                      for(var i =0,len=a.length;i<len;i++){

                                          if(i in a) results[i] = f.call(null,a[i],i,a);

                                 }

                                   return result;

                              };

               //使用函数f()和可选的初始值将数组a减至一个值,如果Array.prototype.reduce存在的话,就使用这个方法

                 var reduce = Array.prototype.reduce

                                    ? function(a,f,initial){   //如果reduce()方法存在的话

                                    if(arguments.length>2)

                                           return a.reduce(f,initial); //如果传入了一个初始值

                                           else return a.reduce(f); //否则没有初始值 

                                         }

                                      :function(a,f,initial){   //这个算法来自ES5规范

                                          var i =0,len = a.length,accumulator;

                                          //以特定的初始值开始,否则第一个值取自a

                                           if(arguments.length>2) accumulator = initial;

                                           else{

                                                  if(len ==0) throw TypeError();

                                                   while(i<len){

                                                          if(i in a){

                                                                accumulator = a[i++];

                                                                 break;

                                                             }

                                                            else i++;

                                                    }

                                                   if(i==len)  throw TypeError();

                                             }

                                        //对于数组中剩下的元素依次调用f()

                                        while(i<len){

                                             if(i in a)  accumulator = f.call(undefined,accumulator,a[i],i,a);

                                             i++;                        

                                          }

                                          return accumulator;

                                 };

      使用定义的map()和reduce()函数,计算平均值和标准差的代码:

            var data = [1,1,3,5,5];

            var sum = function(x,y){ return x+y; };

            var square = function(x,y){ return x*y; };       

            var mean = reduce(data,sum)/data.length;

            var deviations = map(data,function(x){ return x-mean; });      

            var  stddev = Math.sqrt(reduce(map(deviations,square),sum)/(data.length-1));

《JS权威指南学习总结--8.8 函数式编程和8.8.1使用函数处理数组》的更多相关文章

  1. 简单物联网:外网访问内网路由器下树莓派Flask服务器

    最近做一个小东西,大概过程就是想在教室,宿舍控制实验室的一些设备. 已经在树莓上搭了一个轻量的flask服务器,在实验室的路由器下,任何设备都是可以访问的:但是有一些限制条件,比如我想在宿舍控制我种花 ...

  2. 利用ssh反向代理以及autossh实现从外网连接内网服务器

    前言 最近遇到这样一个问题,我在实验室架设了一台服务器,给师弟或者小伙伴练习Linux用,然后平时在实验室这边直接连接是没有问题的,都是内网嘛.但是回到宿舍问题出来了,使用校园网的童鞋还是能连接上,使 ...

  3. 外网访问内网Docker容器

    外网访问内网Docker容器 本地安装了Docker容器,只能在局域网内访问,怎样从外网也能访问本地Docker容器? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Docker容器 ...

  4. 外网访问内网SpringBoot

    外网访问内网SpringBoot 本地安装了SpringBoot,只能在局域网内访问,怎样从外网也能访问本地SpringBoot? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装Java 1 ...

  5. 外网访问内网Elasticsearch WEB

    外网访问内网Elasticsearch WEB 本地安装了Elasticsearch,只能在局域网内访问其WEB,怎样从外网也能访问本地Elasticsearch? 本文将介绍具体的实现步骤. 1. ...

  6. 怎样从外网访问内网Rails

    外网访问内网Rails 本地安装了Rails,只能在局域网内访问,怎样从外网也能访问本地Rails? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Rails 默认安装的Rails端口 ...

  7. 怎样从外网访问内网Memcached数据库

    外网访问内网Memcached数据库 本地安装了Memcached数据库,只能在局域网内访问,怎样从外网也能访问本地Memcached数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装 ...

  8. 怎样从外网访问内网CouchDB数据库

    外网访问内网CouchDB数据库 本地安装了CouchDB数据库,只能在局域网内访问,怎样从外网也能访问本地CouchDB数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Cou ...

  9. 怎样从外网访问内网DB2数据库

    外网访问内网DB2数据库 本地安装了DB2数据库,只能在局域网内访问,怎样从外网也能访问本地DB2数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动DB2数据库 默认安装的DB2 ...

  10. 怎样从外网访问内网OpenLDAP数据库

    外网访问内网OpenLDAP数据库 本地安装了OpenLDAP数据库,只能在局域网内访问,怎样从外网也能访问本地OpenLDAP数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动 ...

随机推荐

  1. MemoryStream和FileStream

    一,FileStream对象的数据来自文件,而MemoryStream对象的数据来自内存缓冲区.这两个类都继承自Stream类. 二,抽象基类System.IO.Stream代表流,它提供Read和W ...

  2. 安装hadoop-eclipse-plugin插件报错解决办法

    安装myecplise的hadoop-eclipse-plugin-2.2.0插件的时候,  把插件放到这个文件夹里 打开myeclipse windows-->preferences 点击 H ...

  3. c#注释

    c#的注释分为:这里不能不说一下什么是注释. 注释本身不会执行,只是说明性文字,只供程序员阅读. 注释又分为:单行注释,多行注释,文档注释. 单行注释://开始 多行注释:/*开始, */结束. 文档 ...

  4. log4j基本使用方法

    通过配置文件可知,我们需要配置3个方面的内容: 1.根目录(级别和目的地) 2.目的地(控制台和文件等) 3.输出样式 Log4j由三个重要的组件构成: 1.日志信息的优先级 日志信息的优先级从高到低 ...

  5. ios在项目中打开word文档、ppt等总结

    最近在项目开发中遇到下载附件文档预览需求,在这里总结一下我的实现方法,本文最后会附带我写的demo下载地址 这里我总结了三种实现方法(1)用webView预览(2)通过UIDocumentIntera ...

  6. VFL语言

    VFL语言 VFL即Visual Format Language,可视化格式语言 NSDictionaryOfVariableBindings(testViewA, testViewB):此为一个宏, ...

  7. android设备的vpn功能

    VPN是什么? VPN:Virtual Private Network,虚拟专用网络:是通过私有的隧道技术在公共数据网络上仿真一条点到点的专线技术,其实质上就是利用加密技术在公网上封装出一个数据通讯隧 ...

  8. 锅巴视频工作室 ----------------android端蓝牙测试demo--app

    android端蓝牙测试demo--app 这个是为一个客户做蓝牙项目时的一个测试demo,用来测试蓝牙单片机的收发情况,代码中没有做一些兼容性测试,请理解 锅巴视频工作室,专注于android视频相 ...

  9. web端和手机端测试有什么不同

    面试中经常被问到web端测试和手机端测试有什么相同点和区别呢?现在总结一下这个问题,如有不对敬请指正 web端和手机端测试有什么区别 1.相同点 不管是web测试还是手机App测试,都离不开测试的相关 ...

  10. SSLv3协议、TLSv1.2协议配置不对导致javax.ws.rs.ProcessingException: java.net.SocketException: Connection reset

    SSl:Secure Sockets Layer 安全套接层 TLS:Transport Layer Security传输层安全 是为网络通信提供安全及数据完整性的一种安全协议.TLS与SSL在传输层 ...