unity_优化app开发测试

UnityTestTool实用解释

2014-07-15  本文已影响8414人  DonaldW

UnityTestTool实用解释

概述

以下的场景是否似曾相识:

UnityTestTool简介

UnityTestTool是Unity提供的官方测试工具,它的核心功能包括:

测试用例

测试用例能做什么

测试用例之所以能解决一开始提出的那些问题,是因为:

  1. 测试用例可以验证功能实现是否正常:前提是测试用例本身设计正确。
  2. 测试用例和功能实现是高度匹配的:模块A的功能实现和模块A的测试用例,是由同一个(批)人,在同一个时间段内编写的。另外,更改功能的接口时,需要同时更改测试用例。
  3. 测试用例本身就是实用文档:测试用例直接使用代码的形式描述了该怎么使用模块、和使用模块的注意点(因为测试用例使用抛出异常的方法来描述不正确使用模块的情况)。
  4. 测试驱动:测试用例甚至可以反过来驱动功能实现。测试用例代码比功能实现代码简单得多,且直接面对功能模块的接口。所以先写测试用例再写功能实现能帮助开发者整理思路、设计更加稳妥的接口。
  5. 使用测试用例进行测试是简单的、可自动化的:工具如果设计恰当,只需简单点击、甚至使用脚本自动化,就完成了测试。另,可以在迭代的某些关键时刻进行自动化测试,比如:每次组员提交代码的时刻、每整点、每次构建版本时。
  6. 测试用例是永久性的:不同于口口相传的瞬时性。(当然文档、代码注释也是永久性的)。

测试用例不能做什么

测试用例只是开发阶段的测试,其不能代替后面测试阶段的真正人肉集成测试。

测试用例的心理纠结

“写测试用例太麻烦、太浪费时间啦!用不用测试用例呢?”。
其实就是一个“重要事情”和“紧急事情”的权衡。都是有生活经验的人了,如何权衡,应该都懂。
总之,建议,在没养成写测试用例的习惯之前、在没体验过测试用例到底花费多少时间之前,都不要武断给下结论“不写测试用例”。

UnityTestTool使用方法

准备工具

下载UnityTestTool,并导入到你的unity项目工程中

编写单元测试

using NUnit.Framework;
using System;
using MoreFun.Collections;

namespace MoreFun.Editor.Test.Collections
{
    [TestFixture]
    class BasicTreeTest
    {
        [Test]
        public void AddAndRemoveChild()
        {
            TreeData data = new TreeData();
            BasicTree tree1 = new BasicTree(data);

            TreeData data2 = new TreeData();
            BasicTree tree2 = new BasicTree(data2);

            tree1.AddChild(tree2);
            if (tree1.GetChildrenCount() != 1)
            {
                throw new Exception("测试失败");
            }

            //省略剩下代码...
        }
    }
}

编写集成测试

2014-0714-1607-25-12.png

脚本自动化

使用脚本自动化的关键无非下面几点:

注:RunIntegrationTests()本只能是从命令行取得需要测试的场景列表。建议修改RunIntegrationTests()的参数,允许以函数参数的方式传入需要测试的场景列表。

具体脚本可参见最下面的附录。

更多

更多详细用法,可以阅读导入到工程的UnityTestTool/Docs/下的pdf。

附录

自动执行单元测试用例、集成测试用例的bat脚本:

echo "Start Unity Test Case"
%unity% -batchmode -projectPath %projectPath% -executeMethod CommandBuild.RunUnitTests -resultFilePath=%buildPath%/BuildTemp -quit -logFile %buildPath%/BuildTemp/RunUnitTests.log
%unity% -batchmode -projectPath %projectPath% -executeMethod CommandBuild.RunIntegrationTests -targetPlatform=%testTargetPlatform% -resultsFileDirectory=%buildPath%/BuildTemp -quit -logFile %buildPath%/BuildTemp/RunIntegrationTests.log
echo "End Unity Test Case, please see log: BuildTemp/RunUnitTests.log, BuildTemp/RunIntegrationTests.log"


echo "Start Build Unity to App"
%unity% -batchmode -projectPath %projectPath% -executeMethod CommandBuild.PreBuild %debugParam% -quit -logFile %buildPath%/BuildTemp/PreBuild.log
%unity% -batchmode -projectPath %projectPath% -executeMethod CommandBuild.Build %debugParam% -android -buildPath=%buildPath% -quit -logFile ./BuildTemp/Build.log
echo "End Build, please see log BuildTemp/PreBuild.log and BuildTemp/Build.log"

自动执行单元测试用例、集成测试用例的C#代码:

using UnityEngine;
using UnityEditor;

public class CommandBuild
{
    private static string[] ms_scenes =
    {
        "Assets/Scenes/KillerStarter.unity"
    };

    private static System.Collections.Generic.List<string> ms_lstTestScenes = new System.Collections.Generic.List<string>()
    {
        "Assets/KillerInteTest/TestBattle/TestBattle.unity",
        "Assets/KillerInteTest/TestUIBase/TestUIBase.unity"
    };


    /// <summary>
    /// 执行UnityTestTool的单元测试
    /// </summary>
    public static void RunUnitTests()
    {
        UnityTest.Batch.RunUnitTests();
    }
    /// <summary>
    /// 执行UnityTestTool的集成测试
    /// </summary>
    public static void RunIntegrationTests()
    {
        UnityTest.Batch.RunIntegrationTests(ms_lstTestScenes);
    }
    /// <summary>
    /// 检测单元测试、集成测试输出的所有xml
    /// </summary>
    /// <returns></returns>
    private static bool CheckTestResult()
    {
        UpdateBuildTempFolderPath();
        string[] lstXml = System.IO.Directory.GetFiles(ms_buildTempFolder, "*.xml");
        if(0 == lstXml.Length)
        {
            Debug.Log("在" + ms_buildTempFolder + "目录未找到任何单元测试结果xml文件!");
            return false;
        }
        else
        {
            foreach(string oneXmlFile in lstXml)
            {
                string oneXmlContent = System.IO.File.ReadAllText(oneXmlFile).ToLower();
                if (oneXmlContent.Contains("success=\"false\"") ||
                    oneXmlContent.Contains("result=\"error\"")
                    )
                {
                    Debug.Log("找到单元测试失败结果!在" + oneXmlFile + "。请查阅单元测试结果xml文件!");
                    return false;
                }
            }
        }

        Debug.Log("未检测到单元测试失败结果!检测文件列表:\n" + lstXml.JoinToString("\n"));
        return true;
    }

    public static void Build()
    {
        Debug.Log("Build");

        if (false == CheckTestResult())
        {
            throw new System.Exception("检测到测试用例结果失败!终止构建!");
        }

      // 省略以下构建代码
    }
}


上一篇 下一篇

猜你喜欢

热点阅读