AngularJs学习笔记--unit-testing

news/2024/12/13 4:47:02/

  javascript是一门动态类型语言,这给她带来了很强的表现能力,但同时也使编译器几乎不能给开发者提供任何帮助。因为这个原因,我们感受到编写任何javascript代码都必须有一套强大完整的测试。angular拥有许多功能,让我们更加容易地测试我们的应用。我们应该没有借口不去写测试(这个嘛……)。

一、 It is all about NOT mixing concerns(全部都关于避免代码关系变得复杂……)

  单元测试,正如名称那样,是关于测试单个“单元”的代码。单元测试努力解答这些问题:我对逻辑的考虑是否已经正确?排序方法得出的结果是否正确?为了解答这些问题,将这些问题独立出来显得尤其重要。这是因为当我们在测试排序方法的时候,我们不想关心其他相关的片段,例如DOM元素或者发起XHR请求获取数据等。明显地,通常比较难做到在典型的项目中单独调用一个函数。导致这个问题的原因是,开发者通常把关系弄得很复杂,最终让一个代码片段看起来可以做所有事情。它通过XHR获取数据,对数据进行排序,然后操纵DOM。与angular一起,我们可以更加容易地写出较好的代码,所以angular为我们提供XHR(我们可以模拟它)的依赖注入,angular还创建允许我们对model进行排序而不需要操作DOM的抽象。所以,到最后,我们可以简单地写一个排序方法,然后通过测试用例创建数据集合,供排序方法测试时使用,然后判断结果model是否符合预期。测试无须等待XHR、者创建对应的DOM和判断函数是否正确操作DOM。angular的核心思想包含代码的可测试性,但同时也要求我们去做正确的事情。angular致力于简化做正确事情的方法,但angular不是魔法,这意味着我们如果不遵循以下的几点,我们最终可能会得出一个不可测试的应用。

1. Dependency Inject

  有许多办法可以获得依赖的资源:1)我们可以使用new操作符;2)我们使用一个众所周知的方式,被称为” 全局单例”;3)我们可以向registry service请求(但我们如何取得一个registry?可以查看后面的章节);4)我们可以期待它会被传递过来。

  上面列出的方法中,只有最后一个是可测试的,让我们看看为什么:

1) Using the new operator

  使用new操作符时基本上没有错误,但问题是通过new调用构造函数将会永久地将调用方与type绑定起来。举个例子,我们尝试实例化一个XHR对象,以让我们可以从服务器获得一些数据。

function MyClass() {this.doWork = function() {var xhr = new XRH();xhr.open(method,url,true);xhr.onreadystatechange = function() {…};xhr.send();}
}

  问题来了,在测试时,我们通常需要实例化一个可以返回测试数据或者网络错误的虚拟的XHR。通过调用new XHR(),我们永久地绑定了真实的XHR,并且没有一个很好的方法去替代它。当然,有一个糟糕的补救办法,有很多理由可以证明那是一个糟糕的想法:

var oldXHR = XHR;
XHR = new MockXHR() {};
myClass.doWork();
//判断MockXHR是否通过正常的参数进行调用
XHR = oldXHR;//如果忘了这一步,很容易会发生悲催的事情。

2) Global look-up

  解决问题的另外一个方法是在一个众所周知的地方获取依赖的资源。

function MyClass() {this.doWork = function() {global.xhr({…});};
}

  没有创建新依赖对象的实例的情况下,问题基本上与new一致,除了那个悲催的补丁以外,没有一个很好的方法可以再测试时拦截global.xhr的调用。测试的最基本的问题是global变量需要改为调用虚拟的方法而被修改。想进一步了解它的坏处,可以参观这里:http://misko.hevery.com/code-reviewers-guide/flaw-brittle-global-state-singletons/

  上面的代码比较难去测试,所以我们必须修改global state:

var oldXHR = global.xhr;
global.xhr = function mockXHR(){…};
var myClass = new MyClass();
//判断MockXHR是否通过正常的参数进行调用
global.xhr = oldXHR;//如果忘了这一步,很容易会发生悲催的事情。

3) Service Registry

  拥有一个包含所有service的registry的话,似乎可以解决问题,然后,在测试代码中替换所需要的service。

function MyClass() {var serviceRegistry = ???;this.doWork = function() {var xhr = serviceRegistry.get(“xhr”);…};
}

  但是,serviceRegistry来自哪里?if it is: * new-ed up, the the test has no chance to reset the services for testing * global look-up, then the service returned is global as well (but resetting is easier, since there is only one global variable to be reset)(这里后面的文字跟乱码一样……没看懂)

  根据这个方法,将上面的Class修改为如下的方式:

var oldServiceLocator = global.serviceLocator;
global.serviceLocator.set('xhr', function mockXHR() {});
var myClass = new MyClass();
myClass.doWork();
//判断MockXHR是否通过正常的参数进行调用
global.serviceLocator = oldServiceLocator; //如果忘了这一步,很容易会发生悲催的事情。

4) Passing in Dependencies

  最后,依赖资源可以被传入。

function MyClass(xhr) {this.doWork = function() {xhr({…});};
}

  这个是首选的方式,因为代码无须理会xhr是从哪来的,也不关心谁创建了传进来的xhr。因此,类的创建者与类的使用者可以分开编码,这将创建的责任从逻辑中分离出来,这就是依赖注入的概述。

  这个class很容易测试,在测试中我们可以这样写:

function xhrMock(args) {…}
var myClass = new MyClass(xhrMock);
myClass.doWrok();
//做一些判断……
通过这个测试代码,我们可以意识到没有任何全局变量被破坏。

  angular附带的dependency-injection(http://www.cnblogs.com/lcllao/archive/2012/09/23/2699401.html),通过这种方式编写的代码,更加容易编写测试代码,如果我们想编写可测试性强的代码,我们最好使用它。

2. Controllers

  逻辑使每一个应用都是唯一的,这就是我们想去测试的。如果我们的逻辑里面混杂着DOM的操作,这将会跟下面的例子一样难测试:

function PasswordController() {// 获取DOM对象的引用var msg = $('.ex1 span');var input = $('.ex1 input');var strength;this.grade = function() {msg.removeClass(strength);var pwd = input.val();password.text(pwd);if (pwd.length > 8) {strength = 'strong';} else if (pwd.length > 3) {strength = 'medium';} else {strength = 'weak';}msg.addClass(strength).text(strength);}
}

  上面的代码在测试时会遇到问题,因为它需要我们的执行测试时候,需要有正确的DOM。测试代码会如下:

var input = $('<input type="text"/>');
var span = $('<span>');
$('body').html('<div class="ex1">').find('div').append(input).append(span);
var pc = new PasswordController();
input.val('abc');
pc.grade();
expect(span.text()).toEqual('weak');
$('body').html('');

  在angular中,controller严格地将DOM操作逻辑分离出来,将大大降低编写测试用例的难度,看看下面的例子:

function PasswordCntrl($scope) {$scope.password = '';$scope.grade = function() {var size = $scope.password.length;if (size > 8) {$scope.strength = 'strong';} else if (size > 3) {$scope.strength = 'medium';} else {$scope.strength = 'weak';}};
}

  测试代码直截了当:

var pc = new PasswordController($scope);
pc.password('abc');
pc.grade();
expect($scope.strength).toEqual('weak');

  值得注意的是,测试代码不仅仅更加间断,而且更加容易追踪。我们一直说测试用例是在讲故事,而不是判断其他不相关的东西。

3. Filters

  filter(http://docs.angularjs.org/api/ng.$filter)是用于将数据转换为对用户友好的格式。它们很重要,因为它们将转换格式的责任从应用逻辑中分离出来,进一步简化了应用逻辑。

myModule.filter('length', function() {return function(text){return (''+(text||'')).length;}
});var length = $filter('length');
expect(length(null)).toEqual(0);
expect(length('abc')).toEqual(3);

4. Directives

5. Mocks

6. Global State Isolation

7. Preferred way of Testing

8. JavascriptTestDriver

9. Jasmine

10.   Sample project

 


http://www.ppmy.cn/news/203548.html

相关文章

vb6 Webview2微软Edge Chromium内核执行JS取网页数据测速

微软Edge Chromium内核执行JS获取网页数据测试 ExcuteScript eval(document.body.innerHTML) from : https://www.163.com 采集的网页HTM字符串占用字节空间1.2MB ExcuteScript回调事件中取得JS执行结果&#xff0c;用时 54 毫秒 其中JSON转字符13.5209毫秒 jSON数据长度: 增…

联想拯救者笔记本电脑Fn键失效,Fn功能相反,开关FnLock

使用设备 联想拯救者R7000 2021 Win11家庭版 问题 误操作后发现 F1&#xff0c;F2等功能被替换成了FnF1&#xff0c;FnF2 解决方法 同时按下FnFnLock(ESC)就可以切换

计算机关机怎么按,按什么键电脑关机

大家好&#xff0c;我是时间财富网智能客服时间君&#xff0c;上述问题将由我为大家进行解答。 电脑关机快捷键及操作方法如下&#xff1a; 方法一、按下ctrlaltdel三个键&#xff0c;就会弹出任务管理器&#xff0c;然后按下alt u键选择关机的选项&#xff0c;同时再按住ctrl…

联想电脑为什么合盖和按电源键不睡眠

这一个星期我发现笔记本&#xff08;原来是win10系统因为无法睡眠后升到win11系统还是无法睡眠&#xff09;&#xff0c;也就是主动点击电源按钮&#xff0c;笔记本只是熄灭屏幕&#xff0c;不会锁屏。在经过好多次的测试&#xff0c;发现我的笔记本在插电的时候不能睡眠&#…

计算机死机后 通过任务管理器关闭程序,电脑死机后,按哪个键结束程序

工具/原料 电脑 方法/步骤 1.为什么会出现电脑突然死机&#xff0c;应用程序无响应此类情况&#xff0c;这要分为硬件与软件两方面的原因了&#xff0c;硬件一般不会出问题&#xff0c;那么就是软件的原因了。程序把数据放在内存中存放数据的缓冲区内&#xff0c;但是需要系统的…

电脑死机按什么键恢复?快速恢复,试试这3个方法

​电脑死机是一个常见的问题&#xff0c;无论什么电脑都会死机&#xff0c;那么电脑死机按什么键恢复呢&#xff1f;常见的方式是按下键盘“Ctrl”“Alt”“Del”组合键&#xff0c;选择重启来解决这个问题。此外&#xff0c;小编还为您提供了其他电脑死机的方法&#xff0c;有…

按什么键启用计算机管理,电脑结束任务按什么键

如今&#xff0c;电脑在我们日常生活中几乎无处不在&#xff0c;然而有时候可能我们在使用电脑的过程中会出现这样或者那样的小问题&#xff0c;比如有时候电脑打开的软件太多&#xff0c;导致电脑有点卡机&#xff0c;我们想要关闭一些程序却无法快速关闭&#xff0c;最常见的…

计算机强制退出程序键,电脑强制关闭程序按哪三个键

电脑同时运行的程序过多&#xff0c;电脑就会非常的卡顿&#xff0c;那么电脑强制关闭程序可以通过什么按键来实现呢&#xff1f;一起来看看吧。 产品型号&#xff1a;Dell 灵越5000、macbook air 系统版本&#xff1a;Windows 10、MAC I OS 10.9 windows电脑 同时按住ctrlaltd…