- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
被测代码:
module lib {
export class Topic {
private _callbacks: JQueryCallback;
public id: string;
public publish: any;
public subscribe: any;
public unsubscribe: any;
public test: any;
constructor(id: string) {
this.id = id;
this._callbacks = jQuery.Callbacks();
this.publish = this._callbacks.fire;
this.subscribe = this._callbacks.add;
this.unsubscribe = this._callbacks.remove;
}
}
export class Bus {
private static _topics: Object = {};
static topic(id: string): Topic {
var topic = id && this._topics[id];
if (!topic) {
topic = new Topic(id);
if (id) {
this._topics[id] = topic;
}
}
return topic;
}
}
}
规范测试对象:
module lib {
class Person {
private _dfd: JQueryDeferred<Topic>;
private _topic: Topic;
constructor(public firstName: string) {
this._dfd = jQuery.Deferred();
this._topic = Bus.topic("user:logon");
this._dfd.done(this._topic.publish);
}
logon() {
this._dfd.resolve(this);
}
}
class ApiService {
constructor() {
Bus.topic("user:logon").subscribe(this.callLogonApi);
}
callLogonApi(person: Person) {
console.log("Person.firstname: " + person.firstName);
}
}
describe("Bus", () => {
var person: Person;
var apiService: ApiService;
beforeEach(() => {
person = new Person("Michael");
apiService = new ApiService();
spyOn(apiService, "callLogonApi");
//or this fails as well
//spyOn(apiService, "callLogonApi").and.callThrough();
person.logon();
});
it("should create subscription and catch the published event", () => {
expect(apiService.callLogonApi).toHaveBeenCalled();
//this fails too
//expect(apiService.callLogonApi).toHaveBeenCalledWith(person);
});
});
}
调用了 callLogonApi 函数并按预期写入控制台,但输出为:
Expected spy callLogonApi to have been called.
Error: Expected spy callLogonApi to have been called.
* 现在使用 ApiService 的构造函数更改为:
constructor() {
Bus.topic("user:logon").subscribe((data)=> { this.callLogonApi(data); });
}
* 而 spyOn 需要
spyOn(apiService, "callLogonApi").and.callThrough();
感谢 Ryan 的出色回答!!
最佳答案
这是正在发生的事情的一个较小版本。
首先,这是一个更简单的版本 spyOn
方法:
function spyOn(obj: any, methodName: string) {
var prev = obj[methodName];
obj[methodName] = function() {
console.log(methodName + ' got called');
prev();
}
}
现在让我们用一个简单的类来尝试一下:
/** OK **/
class Thing1 {
sayHello() {
console.log('Hello, world');
}
}
var x = new Thing1();
spyOn(x, 'sayHello');
x.sayHello(); // 'sayHello got called'
这按预期工作。关于延迟版本,这是您的代码正在执行的操作:
/** Not OK **/
class Thing2 {
private helloMethod;
constructor() {
this.helloMethod = this.sayHello;
}
deferredHello() {
window.setTimeout(this.helloMethod, 10);
}
sayHello() {
console.log('Hello, world');
}
}
var y = new Thing2();
spyOn(y, 'sayHello');
y.deferredHello(); // Why no spy?
最后是固定版本。我将解释为什么它很快就会被修复:
/** OK now **/
class Thing3 {
private helloMethod;
constructor() {
this.helloMethod = () => { this.sayHello(); }
}
deferredHello() {
window.setTimeout(this.helloMethod, 10);
}
sayHello() {
console.log('Hello, world');
}
}
var z = new Thing3();
spyOn(z, 'sayHello');
z.deferredHello(); // Spy works!
这是怎么回事?
请注意 spyOn
函数接受一个对象,包装方法,然后在对象本身上设置一个属性来替换被侦测的函数实例。这非常重要,因为它会改变方法名称的属性查找最终发生的位置。
在正常情况下 ( Thing1
),我们会覆盖 spyOn
上的属性(使用 x
)然后在 x
上调用相同的方法.一切正常,因为我们正在调用与 spyOn
完全相同的函数。包裹。
在延迟的情况下(Thing2
),y.sayHello
改变整个代码的含义。当我们第一次在构造函数中获取它时,我们得到了 sayHello
方法来自类的原型(prototype)。当我们spyOn
y.sayHello
,被包装的函数是一个新的对象,但是我们在执行之前得到的引用仍然指向sayHello
的实现。在原型(prototype)中。
在固定情况下(Thing3
),我们使用一个函数来更懒惰地获取sayHello
的值。 ,所以当z.sayHello
变化(因为我们发现了它),deferredHello
调用“看到”现在位于实例对象而不是类原型(prototype)上的新方法对象。
关于typescript - jasmine spyOn 没有正确报告 toHaveBeenCalled 和 toHaveBeenCalledWith,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22642598/
我使用 File > Settings > JavaScript > Libraries 在 Webstorm 8.0.4 中设置 Jasmine 集成。我添加了 karma-jasmine带有我的
假设我有 spyOn($cookieStore,'get').and.returnValue('abc'); 对于我的用例来说,这太笼统了。任何时候我们打电话 $cookieStore.get('so
我正在阅读有关 Jasmine 测试的教程,它建议您应该使用标准 Jasmine 目录树结构(这似乎与 RSpec 目录树结构有关)。我用谷歌搜索了一个小时,找不到任何有关此标准目录树的信息。 考虑到
问题 如何使用Jasmine插件通过Jasmine运行Karma时更新所使用的karma-jasmine的版本? 仅当Jasmine插件集成了较新版本的karma-jasmine时,才会更新Jasmi
我想在每个规范执行后获得测试的通过或失败状态: var passed = jasmine.getEnv().currentSpec.results().passed(); if (!pas
我有一个与 Karma 一起运行的 Jasmine 测试: describe('When a logged in user chooses Rent and Payment PIN is enable
在我的新手理解中,Jasmine 提供了以下语法: describe('FooBar', function(){ it('should blah-blah', function(){ ex
我认为我通过编写这样的 Jasmine 模拟取得了很好的进展。但我无法修复这个错误。 spyOn 方法实际上是如何工作的?事实上这种方法似乎被广泛使用。 2.0 有什么变化 describe('Tes
有大量文档展示了如何将匹配器添加到 Jasmine 规范(例如 here )。 有人找到了向整个环境添加匹配器的方法吗?我想创建一组有用的匹配器,供任何和所有测试调用,而无需在我的规范中复制粘贴。 目
我正在尝试查找所有现有 Jasmine 期望匹配器(例如“toContain”等)的列表... 哪里可以找到这个?我搜索了一段时间,但找不到像 api 这样的东西。 Jasmine 网站也没有任何类型
我熟悉 python 单元测试测试,如果断言失败,该测试将被标记为“失败”,并继续进行其他测试。另一方面,即使其中一个失败, Jasmine 也会继续完成所有期望。如何让 Jasmine 在第一个期望
工具:Protractor 3.3.0、Jasmine 2.4.1、Selenium Standalone Server。 我有一个测试套件,里面有大量的 spec.js 文件,每个文件都包含我的应用
Jasmine 中是否有与 console.log 类似的方法?如果我想注销我 mock 的 $controller 或服务的值,有没有办法做到这一点? 最佳答案 您应该只使用 console.log
我已经通过运行 npm install jasmine -g 全局安装了 jasmine。 运行 jasmine -v 给我 jasmine v2.5.0 jasmine-core v2.5.0 然后
我有gulp-jasmine-phantom正在运行,但我得到 ReferenceError: Tictactoe is not defined 。我感觉我犯了一些根本性的错误。 我的文件结构: gu
我已经创建了一个 JavaScript 项目,需要使用 Jasmine 进行单元测试。我遵循了以下步骤: 1. npm install jasmine --save-dev - 从项目根目录 2.现在
我正在为我的应用程序使用 Yeoman+Angular Generator,我一直在努力与 Jasmine 相处!这就是我卡住的地方。我希望能够在 Jasmine 测试中使用 jQuery 选择器。我
it('should for something', function check(done) { browser.sleep(2000); $('.csTx').isPresent().th
Javascript 编码有点新,所以请耐心等待。 我阅读了以下链接 jasmine with jscoverage automated testing 但是,由于我没有 Ruby 项目,因此它似乎不
在Jasmine headless (headless)Webkit中运行测试时,我遇到了一个简单的TypeError: 'undefined' is not an object失败。但是没有提示在哪
我是一名优秀的程序员,十分优秀!