深入理解JavaScript系列(49):Function模式(上篇)

2024-09-01 15:32

本文主要是介绍深入理解JavaScript系列(49):Function模式(上篇),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

介绍

本篇主要是介绍Function方面使用的一些技巧(上篇),利用Function特性可以编写出很多非常有意思的代码,本篇主要包括:回调模式、配置对象、返回函数、分布程序、柯里化(Currying)。

回调函数

在JavaScript中,当一个函数A作为另外一个函数B的其中一个参数时,则函数A称为回调函数,即A可以在函数B的周期内执行(开始、中间、结束时均可)。

举例来说,有一个函数用于生成node

var complexComputation = function () { /* 内部处理,并返回一个node*/};

有一个findNodes函数声明用于查找所有的节点,然后通过callback回调进行执行代码。

var findNodes = function (callback) {
var nodes = [];var node = complexComputation();// 如果回调函数可用,则执行它
if (typeof callback === "function") {
callback(node);
}nodes.push(node);
return nodes;
};

关于callback的定义,我们可以事先定义好来用:

// 定义callback
var hide = function (node) {
node.style.display = "none";
};// 查找node,然后隐藏所有的node
var hiddenNodes = findNodes(hide);

也可以直接在调用的时候使用匿名定义,如下:

// 使用匿名函数定义callback
var blockNodes = findNodes(function (node) {
node.style.display = 'block';
});

我们平时用的最多的,估计就数jQuery的ajax方法的调用了,通过在done/faild上定义callback,以便在ajax调用成功或者失败的时候做进一步处理,代码如下(本代码基于jquery1.8版):

var menuId = $("ul.nav").first().attr("id");
var request = $.ajax({url: "script.php",type: "POST",data: {id : menuId},dataType: "html"
});//调用成功时的回调处理
request.done(function(msg) {$("#log").html( msg );
});//调用失败时的回调处理
request.fail(function(jqXHR, textStatus) {alert( "Request failed: " + textStatus );
});

配置对象

如果一个函数(或方法)的参数只有一个参数,并且参数为对象字面量,我们则称这种模式为配置对象模式。例如,如下代码:

var conf = {username:"shichuan",first:"Chuan",last:"Shi"
};
addPerson(conf);

则在addPerson内部,就可以随意使用conf的值了,一般用于初始化工作,例如jquery里的ajaxSetup也就是这种方式来实现的:

// 事先设置好初始值
$.ajaxSetup({url: "/xmlhttp/",global: false,type: "POST"});// 然后再调用$.ajax({ data: myData });

另外,很多jquery的插件也有这种形式的传参,只不过也可以不传,不传的时候则就使用默认值了。

返回函数

返回函数,则是指在一个函数的返回值为另外一个函数,或者根据特定的条件灵活创建的新函数,示例代码如下:

var setup = function () {console.log(1);return function () {console.log(2);};
};// 调用setup 函数
var my = setup(); // 输出 1
my(); // 输出 2
// 或者直接调用也可
setup()();

或者你可以利用闭包的特性,在setup函数里记录一个私有的计数器数字,通过每次调用来增加计数器,代码如下:

var setup = function () {var count = 0;return function () {return ++count;};
};// 用法
var next = setup();
next(); // 返回 1
next(); // 返回 2
next(); // 返回 3

偏应用

这里的偏应用,其实是将参数的传入工作分开进行,在有的时候一系列的操作可能会有某一个或几个参数始终完全一样,那么我们就可以先定义一个偏函数,然后再去执行这个函数(执行时传入剩余的不同参数)。

举个例子,代码如下:

var partialAny = (function (aps) {// 该函数是你们自执行函数表达式的结果,并且赋值给了partialAny变量
    function func(fn) {var argsOrig = aps.call(arguments, 1);return function () {var args = [],argsPartial = aps.call(arguments),i = 0;// 变量所有的原始参数集,// 如果参数是partialAny._ 占位符,则使用下一个函数参数对应的值// 否则使用原始参数里的值for (; i < argsOrig.length; i++) {args[i] = argsOrig[i] === func._? argsPartial.shift(): argsOrig[i];}// 如果有任何多余的参数,则添加到尾部return fn.apply(this, args.concat(argsPartial));};}// 用于占位符设置func._ = {};return func;
})(Array.prototype.slice);

使用方式如下:

// 定义处理函数
function hex(r, g, b) {return '#' + r + g + b;
}//定义偏函数, 将hex的第一个参数r作为不变的参数值ff
var redMax = partialAny(hex, 'ff', partialAny._, partialAny._);// 新函数redMax的调用方式如下,只需要传入2个参数了:
console.log(redMax('11', '22')); // "#ff1122"

如果觉得partialAny._太长,可以用__代替哦。

var __ = partialAny._;var greenMax = partialAny(hex, __, 'ff');
console.log(greenMax('33', '44'));var blueMax = partialAny(hex, __, __, 'ff');
console.log(blueMax('55', '66'));var magentaMax = partialAny(hex, 'ff', __, 'ff');
console.log(magentaMax('77')); 

这样使用,就简洁多了吧。

Currying

Currying是函数式编程的一个特性,将多个参数的处理转化成单个参数的处理,类似链式调用。

举一个简单的add函数的例子:

function add(x, y) {var oldx = x, oldy = y;if (typeof oldy === "undefined") { // partialreturn function (newy) {return oldx + newy;}}return x + y;
}

这样调用方式就可以有多种了,比如:

// 测试
typeof add(5); // "function"
add(3)(4); // 7// 也可以这样调用
var add2000 = add(2000);
add2000(10); // 2010

接下来,我们来定义一个比较通用的currying函数:

// 第一个参数为要应用的function,第二个参数是需要传入的最少参数个数
function curry(func, minArgs) {if (minArgs == undefined) {minArgs = 1;}function funcWithArgsFrozen(frozenargs) {return function () {// 优化处理,如果调用时没有参数,返回该函数本身var args = Array.prototype.slice.call(arguments);var newArgs = frozenargs.concat(args);if (newArgs.length >= minArgs) {return func.apply(this, newArgs);} else {return funcWithArgsFrozen(newArgs);}};}return funcWithArgsFrozen([]);
}

这样,我们就可以随意定义我们的业务行为了,比如定义加法:

var plus = curry(function () {var result = 0;for (var i = 0; i < arguments.length; ++i) {result += arguments[i];}return result;
}, 2);

使用方式,真实多种多样哇。

plus(3, 2) // 正常调用
plus(3) // 偏应用,返回一个函数(返回值为3+参数值)
plus(3)(2) // 完整应用(返回5)
plus()(3)()()(2) // 返回 5
plus(3, 2, 4, 5) // 可以接收多个参数
plus(3)(2, 3, 5) // 同理

如下是减法的例子

var minus = curry(function (x) {var result = x;for (var i = 1; i < arguments.length; ++i) {result -= arguments[i];}return result;
}, 2);

或者如果你想交换参数的顺序,你可以这样定义

var flip = curry(function (func) {return curry(function (a, b) {return func(b, a);}, 2);
});

更多资料,可以参考如下地址:

http://www.cnblogs.com/rubylouvre/archive/2009/11/09/1598761.html

http://www.cnblogs.com/sanshi/archive/2009/02/17/javascript_currying.html

总结

JavaScript里的Function有很多特殊的功效,可以利用闭包以及arguments参数特性实现很多不同的技巧,下一篇我们将继续介绍利用Function进行初始化的技巧。

参考地址:http://shichuan.github.com/javascript-patterns/#function-patterns

这篇关于深入理解JavaScript系列(49):Function模式(上篇)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1127334

相关文章

Spring Boot 整合 SSE的高级实践(Server-Sent Events)

《SpringBoot整合SSE的高级实践(Server-SentEvents)》SSE(Server-SentEvents)是一种基于HTTP协议的单向通信机制,允许服务器向浏览器持续发送实... 目录1、简述2、Spring Boot 中的SSE实现2.1 添加依赖2.2 实现后端接口2.3 配置超时时

Spring Boot读取配置文件的五种方式小结

《SpringBoot读取配置文件的五种方式小结》SpringBoot提供了灵活多样的方式来读取配置文件,这篇文章为大家介绍了5种常见的读取方式,文中的示例代码简洁易懂,大家可以根据自己的需要进... 目录1. 配置文件位置与加载顺序2. 读取配置文件的方式汇总方式一:使用 @Value 注解读取配置方式二

一文详解Java异常处理你都了解哪些知识

《一文详解Java异常处理你都了解哪些知识》:本文主要介绍Java异常处理的相关资料,包括异常的分类、捕获和处理异常的语法、常见的异常类型以及自定义异常的实现,文中通过代码介绍的非常详细,需要的朋... 目录前言一、什么是异常二、异常的分类2.1 受检异常2.2 非受检异常三、异常处理的语法3.1 try-

Java中的@SneakyThrows注解用法详解

《Java中的@SneakyThrows注解用法详解》:本文主要介绍Java中的@SneakyThrows注解用法的相关资料,Lombok的@SneakyThrows注解简化了Java方法中的异常... 目录前言一、@SneakyThrows 简介1.1 什么是 Lombok?二、@SneakyThrows

Java中字符串转时间与时间转字符串的操作详解

《Java中字符串转时间与时间转字符串的操作详解》Java的java.time包提供了强大的日期和时间处理功能,通过DateTimeFormatter可以轻松地在日期时间对象和字符串之间进行转换,下面... 目录一、字符串转时间(一)使用预定义格式(二)自定义格式二、时间转字符串(一)使用预定义格式(二)自

Spring 请求之传递 JSON 数据的操作方法

《Spring请求之传递JSON数据的操作方法》JSON就是一种数据格式,有自己的格式和语法,使用文本表示一个对象或数组的信息,因此JSON本质是字符串,主要负责在不同的语言中数据传递和交换,这... 目录jsON 概念JSON 语法JSON 的语法JSON 的两种结构JSON 字符串和 Java 对象互转

JAVA保证HashMap线程安全的几种方式

《JAVA保证HashMap线程安全的几种方式》HashMap是线程不安全的,这意味着如果多个线程并发地访问和修改同一个HashMap实例,可能会导致数据不一致和其他线程安全问题,本文主要介绍了JAV... 目录1. 使用 Collections.synchronizedMap2. 使用 Concurren

Java Response返回值的最佳处理方案

《JavaResponse返回值的最佳处理方案》在开发Web应用程序时,我们经常需要通过HTTP请求从服务器获取响应数据,这些数据可以是JSON、XML、甚至是文件,本篇文章将详细解析Java中处理... 目录摘要概述核心问题:关键技术点:源码解析示例 1:使用HttpURLConnection获取Resp

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组