作者:zhanhailiang 日期:2012-10-24
基本概念介绍
策略模式支持你在运行时选择算法。代码的客户端可以使用同一个接口来工作。但是它却根据客户正在试图执行任务的上下文,从多个算法中选择用于处理特定任务的算法。
使用策略模式的其中一个例子是解决表单验证问题。可以创建一个具有validate()方法的验证器(validator)对象。无论表单的具体类型是什么,该方法都将会被调用,并且总是返回相同的结果,一个未经验证的数据列表以及任意的错误消息。
但是根据具体的表单形式及待验证的数据,验证器的客户端可以选择不同类型的检查方法,验证器将选择最佳的策略以处理任务,并将具体的数据验证委托给适当的算法。
数据验证示例
假设有以下数据块,它可能来自于网页上的一个表单,而你需要验证它是否有效:
var data = { first_name : "Super", last_name : "Man", age : "unknown", username : "o_0" };
在这个具体的例子中,为使验证器知道什么 是最好的策略,首先需要配置该验证器,并且设置认为是有效的且可接受的规则。
validator.config = { first_name : 'isNonEmpty', age : 'isNumber', username : 'isAlphaNum' }; validator.validate(data); if(validator.hasErrors()) { console.log(validator.messages.join("\n")); }
接下来实现validator。首先实现用于检查的可用算法。
validator.types.isNonEmpty = { validate : function(value) { return value !== ''; }, instructions : "the value cannot be empty." }; validator.types.isNumber = { validate : function(value) { return !isNaN(value); }, instructions : "the value can only be a valid number, e.g. 1, 3.14 or 2010." }; validator.types.isAlphaNum = { validate : function(value) { return !/[^a-z0-9]/i.test(value); }, instructions : "the value can only contain characters and numbers, no special symbols." };
最后,核心的validator对象如下所示:
var validator = { // 所有可用的检查 types : {}, // 在当前验证会话中的错误信息 messages : [], // 当前验证配置名称:验证类型 config : {}, // 接口方法 `data`为key-value对 validate : function(data) { var i, msg, type, checker, result_ok; // 重置所有信息 this.messages = []; for(i in data) { if(data.hasOwnProperty(i)) { type = this.config[i]; checker = this.types[type]; if(!type) { continue; } if(!checker) { throw { name : "ValidationError", message : "No handler to validate type " + type }; } result_ok = checker.validate(data[i]); if(!result_ok) { msg = "Invalid value for *" + i + "*, " + checker.instructions; this.messages.push(msg); } } } return this.hasErrors(); }, // 帮助函数 hasErrors : function() { return this.messages.length !== 0; } };
完整代码:
var validator = { // 所有可用的检查 types : {}, // 在当前验证会话中的错误信息 messages : [], // 当前验证配置名称:验证类型 config : {}, // 接口方法 `data`为key-value对 validate : function(data) { var i, msg, type, checker, result_ok; // 重置所有信息 this.messages = []; for(i in data) { if(data.hasOwnProperty(i)) { type = this.config[i]; checker = this.types[type]; if(!type) { continue; } if(!checker) { throw { name : "ValidationError", message : "No handler to validate type " + type }; } result_ok = checker.validate(data[i]); if(!result_ok) { msg = "Invalid value for *" + i + "*, " + checker.instructions; this.messages.push(msg); } } } return this.hasErrors(); }, // 帮助函数 hasErrors : function() { return this.messages.length !== 0; } }; validator.types.isNonEmpty = { validate : function(value) { return value !== ''; }, instructions : "the value cannot be empty." }; validator.types.isNumber = { validate : function(value) { return !isNaN(value); }, instructions : "the value can only be a valid number, e.g. 1, 3.14 or 2010." }; validator.types.isAlphaNum = { validate : function(value) { return !/[^a-z0-9]/i.test(value); }, instructions : "the value can only contain characters and numbers, no special symbols." }; var data = { first_name : "Super", last_name : "Man", age : "unknown", username : "o_0" }; validator.config = { first_name : 'isNonEmpty', age : 'isNumber', username : 'isAlphaNum' }; validator.validate(data); if(validator.hasErrors()) { console.log(validator.messages.join("\n")); }
以上validator对象是通用的,增强validator对象的方法是添加更多的类型检查。若在多个页面中使用它,很快就会有一个优良的特定检查集合,故以后针对每个新的用例,所需做的就是配置该验证器并运行validate()方法。
- 1楼BestForYou_昨天 15:20
- 嗯 不错 学习了