技术开发 频道

Web开发者必知:Javascript的策略模式

  通过使用类,我们与anexecutemethod对象定义了一个策略。客户端可以使用任何策略实现该接口。

  同样注意我又是怎样创建GreetingStrategy的。有趣的部分是对methodexecute的重载。它以其他函数的形式定义。现在类的后继子类可以改变特定的行为,如thesayHiorsayByemethod,并不改变常规的算法。这种模式叫做模板方法,非常适合策略模式。

  让我们看个究竟。

// Since the GreetingStrategy#execute method uses methods to define its algorithm,
// the Template Method pattern, we can subclass it and simply override one of those
// methods to alter the behavior without changing the algorithm.

var PoliteGreetingStrategy
= function() {};
PoliteGreetingStrategy.prototype
= Object.create(GreetingStrategy.prototype);
PoliteGreetingStrategy.prototype.sayHi
= function() {
  return
"Welcome sir, ";
};

var FriendlyGreetingStrategy
= function() {};
FriendlyGreetingStrategy.prototype
= Object.create(GreetingStrategy.prototype);
FriendlyGreetingStrategy.prototype.sayHi
= function() {
  return
"Hey, ";
};

var BoredGreetingStrategy
= function() {};
BoredGreetingStrategy.prototype
= Object.create(GreetingStrategy.prototype);
BoredGreetingStrategy.prototype.sayHi
= function() {
  return
"sup, ";
};

var politeGreeter  
= new Greeter(new PoliteGreetingStrategy());
var friendlyGreeter
= new Greeter(new FriendlyGreetingStrategy());
var boredGreeter    
= new Greeter(new BoredGreetingStrategy());

politeGreeter.greet();  
//=> 'Welcome sir, Goodbye.'
friendlyGreeter.greet(); //=> 'Hey, Goodbye.'
boredGreeter.greet();    //=> 'sup, Goodbye.'

  GreetingStrategy 通过指定theexecutemethod的步骤,创建了一个类的算法。在上面的代码片段中,我们通过创建专门的算法从而利用了这一点。

  没有使用子类,我们的Greeter 依然展示出一种多态行为。没有必要在Greeter 的不同类型上进行切换来触发正确的算法。这一切都绑定到每一个Greeter 对象上。

var greeters = [
  
new Greeter(new BoredGreetingStrategy()),
  
new Greeter(new PoliteGreetingStrategy()),
  
new Greeter(new FriendlyGreetingStrategy()),
];

greeters.forEach(
function(greeter) {
  
  
// Since each greeter knows its strategy, there's no need
  // to do any type checking. We just greet, and the object
  
// knows how to handle it.
  greeter.greet();
});

  多环境下的策略模式

  我最喜欢的有关策略模式的例子之一,实在 Passport.js库中。Passport.js提供了一种在Node中处理身份验证的简单方式。大范围内的供应商都支持(Facebook, Twitter, Google等等),每一个都作为一种策略实现。

  该库作为一个npm包是可行的,其所有的策略也一样。库的用户可以决定为他们特有的用例安装哪一个npm包。下面是展示其如何实现的代码片段:

// Taken from http://passportjs.org

var passport
= require('passport')
    
    
// Each authentication mechanism is provided as an npm package.
    
// These packages expose a Strategy object.
  , LocalStrategy
= require('passport-local').Strategy
  , FacebookStrategy = require('passport-facebook').Strategy;

// Passport can be instanciated using any Strategy.
passport.use(
new LocalStrategy(
  
function(username, password, done) {
    User.findOne({ username: username },
function (err, user) {
      
if (err) { return done(err); }
      
if (!user) {
        return done(
null, false, { message: 'Incorrect username.' });
      }
      
if (!user.validPassword(password)) {
        return done(
null, false, { message: 'Incorrect password.' });
      }
      return done(
null, user);
    });
  }
));

// In this case, we instanciate a Facebook Strategy
passport.use(
new FacebookStrategy({
    clientID: FACEBOOK_APP_ID,
    clientSecret: FACEBOOK_APP_SECRET,
    callbackURL:
"http://www.example.com/auth/facebook/callback"
  },
  
function(accessToken, refreshToken, profile, done) {
    User.findOrCreate(...,
function(err, user) {
      
if (err) { return done(err); }
      done(
null, user);
    });
  }
));

  Passport.js库只配备了一两个简单的身份验证机制。除此之外,它没有超过一个符合上下文对象的一个策略类的接口。这种机制让他的使用者,很容易的实现他们自己的身份验证机制,而对项目不产生不利的影响。

  反思

  策略模式为你的代码提供了一种增加模块化和可测性的方式。这并不意味着(策略模式)总是有效。Mixins 同样可以被用来进行功能性注入,如在运行时的一个对象的算法。扁平的老式 duck-typing多态有时候也可以足够简单。

  然而,使用策略模式允许你在一开始没有引入大型体系的情况下,随着负载型的增长,扩大你的代码的规模。正如我们在Passport.js例子中看到的一样,对于维护人员在将来增加另外的策略,将变得更加方便。

0
相关文章