16

我正在尝试对具有异步方法但没有运气的服务进行单元测试。

我试图通过使用 angularjs 中的 $q 支持来实现承诺。

任何帮助,将不胜感激。

http://jsfiddle.net/9pBze/37/

angular.module('myapp', ['myservice']);

angular.module('myservice', []).factory('myservice', function($q) {
  var ls = {};

  ls.DoIt = function() {
    var deferred = $q.defer();

    setTimeout(function(){
        deferred.resolve(5);
    },3000);

    return deferred.promise;
  }

  return ls;

});

describe('services', function () {

    beforeEach(module('myservice'));

    it("should equal 2",  inject(function(myservice) {
        myservice.DoIt().then(function(returned) {
            expect(returned).toEqual(2);
        });        
    }));
});
4

2 回答 2

30

首先,setTimeout由于难以模拟,因此测试起来特别棘手。幸运的是,AngularJS 有一个包装器$timeout

  ls.DoIt = function() {
    var deferred = $q.defer();

    $timeout(function(){
        deferred.resolve(5);
    },3000);

    return deferred.promise;
  }

提供的模拟$timeout允许我们轻松模拟经过的时间(使用$timeout.flush()),这意味着我们的测试可以快速运行,而无需真正等待异步事件完成(请注意,生产代码仍在使用异步 API!)。

更改后的测试如下所示:

it("should equal 5",  inject(function(myservice, $timeout) {

    var valueToVerify;
    myservice.DoIt().then(function(returned) {
      valueToVerify = returned;  
    });  
    $timeout.flush();        
    expect(valueToVerify).toEqual(5);
}));

最后是工作 jsFiddle:http: //jsfiddle.net/v9L9G/1/

于 2013-03-18T13:23:52.167 回答
4

它与 Angular 本身无关,但与Jasmine 异步测试有关。

如果您需要setTimeout使用 Angular $timeout。如果您希望对 setTimeout/$timeout 执行进行精细控制,请使用模拟 Clock

于 2013-03-18T13:31:17.193 回答