【生生不息原创】自己动手写Web自动化测试框架5 - 判断浏览器是否加载完成
上面的几次课程中,我们介绍了如何打开浏览器,如何获取每个web控件的信息,并且控制并验证他们。
从上面的文章中,我相信大家已经可以写出简单的测试程序了。但是还有一个很重要的问题没有解决:如何判断浏览器是否加载完成?
前面的文章我们没有对浏览器的加载进行判断,而只是简简单单的等一段时间,这不是一个很好的解决方法,一方面浪费了时间,另一方面,我们也无法知道应该等多久,导致我们的测试程序不够稳定。
接下来我们假设被测网页没有Ajax和框架,以这种情况来分析如何判断网页加载完毕。
现在比较常用判断是否加载的方法有三种:
-
不停判断IE的状态,如果没有准备好就等待。
-
实现IE的DocumentComplete事件,标志完成。
-
不停去查找页面有没有我们想要控件,没有就等待。
第一种方法:不停判断IE的状态,我们要判断IE的哪些状态呢?
一方面,我们需要判断IE的Busy状态,看IE是不是在忙着解析东西,另一方面判断IE的ReadyState状态,看html文档是不是被完全加载进来。
while (ie.Busy || ie.ReadyState != tagREADYSTATE.READYSTATE_COMPLETE) |
用如上的代码就可以等待IE到完成。
| |
这里只是简简单单的Demo,所以用了很简单的预计进行判断,我们假设我们的网页没有Ajax,也不会出现Load的死锁,真正的实际工作要比这个复杂一些,比如要定一个Time out,如果除了Timeout的范围,就强行终止,以防止测试过程中的死锁。 而如何判断Ajax是否被加载完,不是我们这个系列的讨论范围,请关注以后的其他系列文章。 |
这种方法是我比较推荐的一种方法,虽然《.net软件测试自动化之道》推荐的是第二种方法,不过我经过实际的测试,推荐第一种方法。这个方法可以比较好的处理Navigate、Submit等情况,也是WatiN使用的方法(WatiN的用法要复杂很多,考虑到了Frame等其他情况)。
第二种方法:通过绑定DocumentComplete,用AutoResetEvent来等待。
InternetExplorer给我们提供了DocumentComplete事件,会在IE被Load之后被调用,我们可以使用这个来等待。等待方法就是使用System.Threading.AutoResetEvent对象来。
所以我们需要做的是:
下面是源代码: //1.声明AutoResetEvent对象实例 private static AutoResetEvent DocComplete = new AutoResetEvent(false); //...省略得到IE对象 //2.绑定DocumentComplete事件 ie.DocumentComplete += new DWebBrowserEvents2_DocumentCompleteEventHandler(ie_DocumentComplete); Console.WriteLine("Navigating ..."); object o = null; ie.Navigate("http://www.baidu.com/", ref o, ref o, ref o, ref o); //调用WaitOne等待 DocComplete.WaitOne(); HTMLDocument doc = (HTMLDocument)ie.Document; HTMLInputElement keyword = (HTMLInputElement)doc.getElementById("kw"); keyword.value = "colblog.net"; HTMLButtonElement submit = (HTMLButtonElement)doc.getElementById("sb"); submit.click(); //调用WaitOne等待 DocComplete.WaitOne(); ie.Quit(); } //实现DocumentComplete事件,调用Set方法。 static void ie_DocumentComplete(object pDisp, ref object URL) { DocComplete.Set(); } 省略了一些前几篇文章的东西,完整代码请下载源代码。 第三种方法是不停的去读我们要的控件是否出来。这个方法一般需要配合前面两个使用,而且也可以部分解决Ajax的问题。因为具体实现代码比较多,而又不是我们要讲的重点,就不贴出源代码,只是讲一下实现的思路,让大家了解一下。 这个方法的优点是可以部分解决Ajax,但是缺点也就是实现比较复杂,而且如果报错,无法分清是因为页面没有这个控件,还是因为速度较慢,控件还没有被读入。 以上介绍了在没有Ajax和框架情况下,如何判断浏览器是否已经加载完成。从下一篇文章开始,我们就要进入正题,开始我们的自动化测试框架了。
static void Main(string[] args)
{
这种方法有自己的好处,就是使用了IE自己的事件,判断代码很简单,不过有也不好的地方,第一就是帮度DocumentComplete事件以后,打开的IE会变得响应很慢,尤其是当设置断点调试的时候,IE会变得尤其的慢。第二个缺点就是如果加上框架,还有页面的跳转,就会是这个方法很难捉摸。比如,如果上面的例子里面,URL写成"baidu.cn",就会出错。
具体实践方法是:设置一个Timeout,在这个Timeout时间之前,不停的去看我们要验证的控件是不是被Load进来。如果对页面进行建模,就会去看我们的这个页面的所有的已经定义了的控件是不是被Load进来,如果出了Timeout,就会报错。
- circleoflife's blog
- 要发表评论,请先登录 或 注册
标签云
感谢您关注我的个人博客!
为了您的更好的阅读和留言,建议您注册用户,这样您可以获得更大的权限,以及更方便的留言。
注册只需要一分钟的时间。但是您需要提供您的邮箱。注册后,密码会自动发到您的邮箱,请在第一次登录时修改。
我承诺不会将您的Email泄露给他人,或者发送垃圾邮件。
首先我自己在写的时候出现这样的错误,下了这里的源代码,还是出现错误
Launching IE ...
Attaching to IE ...
未处理的异常: System.Runtime.InteropServices.COMException (0x80004005): 对 COM
组件的调用返回了错误 HRESULT E_FAIL。
在 SHDocVw.IWebBrowser2.get_HWND()
在 TestFrameworkDemo.Program.Main(String[] args) 位置 C:\Documents and Settin
gs\bzcy\桌面\TestFrameworkDemo\TestFrameworkDemo\Program.cs:行号 33
请按任意键继续. . .
是在
InternetExplorer e = (InternetExplorer)allBrowser.Item(i);
if (e.HWND == (int)p.MainWindowHandle) -------->对COM组件的调用返回了错误。。。
{
ie = e;
break;
}
帮忙看下额~VS2005环境下。
我在调试的时候有可能会出现这样的问题,一般我的做法有:
1. 如果你有打开状态的Maxthon,请关闭掉。
2. 重启一下explorer进程,然后试一下。
3. 还是不行的话重启一下计算机。
这个情况是间歇性出现,我也没有很明白到底是为什么,多试几次,没准会好起来。
AutoResetEvent和ManualRestEvent 有啥区别啊 我看有的框架用ManualResetEvent
请教你一个问题!
针对VC++,有没有相关接口的支持啊?抛开DotNet,利用VC++是否也有这么方便的调用?
如果不可行,单独利用VC++实现网页自动化测试应该是怎样一个方向?
向你请教,我是DotNet盲,可能问题很业余,包涵!
时常关注!
我提到的两个类库都是基于COM的,所以调用没有问题,至于是不是也这么方便,我觉得应该没什么问题吧。
和你相反,我是一个VC盲,呵呵,只知道很少的关于VC的知识。所以不能给出很确切的答案,抱歉。
不短了^_^
字数多稿费多嘛!