Flutter测试

一. 单元测试

单元测试是针对一个函数或者类进行测试

1.1. 添加测试依赖

将 test 或者 flutter_test加入依赖文件,默认创建的Flutter程序已经有了依赖:

  • Test 包提供了编写测试所需要的核心功能
dev_dependencies:
  flutter_test:
    sdk: flutter

1.2. 创建需要测试的类

单元测试通常是测试一个函数或者类,这个函数或者类被称之为是一个单元。

在这里,我们按照官方示例,创建一个简单的Counter类来演示:

class Counter {
  int value = 0;

  void increment() => value++;
  void decrement() => value--;
}

1.3. 创建测试文件

我们在test目录下(注意:不是lib目录下),创建一个测试文件:counter_test.dart

  • 通常测试代码都会放在该目录下,并且测试文件不会打包到最终的应用程序中;
  • 测试文件通常以 _test.dart 命名,这是 test runner 寻找测试文件的惯例;
import 'package:flutter_test/flutter_test.dart';
import 'package:test_demo/counter.dart';

// 测试的入口函数
void main() {
  // test函数,第一个参数是测试描述,第二个函数是测试执行的函数
  test("Counter Class test", () {
    // 1.创建Counter并且执行操作
    final counter = Counter();
    counter.increment();
    // 2.通过expect来监测结果和期望的值是否一样,如果一样就通过,否则不通过
    expect(counter.value, 1);
  });
}

测试文件写好之后,点击左边的开始按钮,即可测试:

显示✅,则说明测试通过。

1.4. 整合多个测试

如果对同一个类或函数有多个测试,我们希望它们关联在一起进行测试,可以使用group。

import 'dart:math';
import 'package:flutter_test/flutter_test.dart';
import 'package:test_demo/counter.dart';

void main() {
  group("Counter Test", () {
    test("Counter Default Value", () {
      expect(Counter().value, 0);
    });

    test("Counter Increment test", () {
      final counter = Counter();
      counter.increment();
      expect(counter.value, 1);
    });

    test("Counter Decrement test", () {
      final counter = Counter();
      counter.decrement();
      expect(counter.value, -1);
    });
  });
}

1.5. 执行测试结果

用 IntelliJ 或 VSCode 执行测试

IntelliJ 和 VSCode 的 Flutter 插件支持执行测试。用这种方式执行测试是最好的,因为它可以提供最快的反馈闭环,而且还支持断点调试。

  • IntelliJ
  1. 打开文件 counter_test.dart
  2. 选择菜单 Run
  3. 点击选项 Run 'tests in counter_test.dart'
  4. 或者,也可以使用系统快捷键!
  • VSCode
  1. 打开文件 counter_test.dart
  2. 选择菜单 Debug
  3. 点击选项 Start Debugging
  4. 或者,也可以使用系统快捷键!

在终端执行测试

我们也可以打开终端,在工程根目录输入以下命令来执行测试:

flutter test test/counter_test.dart

二. Widget测试

Widget测试主要是针对某一个封装的 Widget进行单独测试

1.1. 添加测试依赖

Widget测试需要先给 pubspec.yaml 文件的 dev_dependencies 段添加 flutter_test 依赖。

  • 在单元测试中我们已经说过,默认创建的Flutter项目已经添加了
dev_dependencies:
  flutter_test:
    sdk: flutter

1.2. 创建测试Widget

import 'package:flutter/material.dart';

class HYKeywords extends StatelessWidget {
  final List<String> _keywords;
  HYKeywords(this._keywords);

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: ListView(
        children: keywords.map((keyword) {
          return ListTile(
            leading: Icon(Icons.people),
            title: Text(keyword),
          );
        }).toList(),
      ),
    );
  }
}

1.3. 编写测试代码

创建对应的测试文件,编写对应的测试代码:

  • testWidgets:flutter_test中用于测试Widget的函数;
  • tester.pumpWidget:pumpWidget 方法会建立并渲染我们提供的 widget;
  • find:find() 方法来创建我们的 Finders;
    findsNothing:验证没有可被查找的 widgets。
    findsWidgets:验证一个或多个 widgets 被找到。
    findsNWidgets:验证特定数量的 widgets 被找到。
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:test_demo/keywords.dart';

void main() {
  testWidgets("KeywordWidget Test", (WidgetTester tester) async {
    // 注入widget
    await tester.pumpWidget(MaterialApp(title: "demo", home: HYKeywords(["abc", "cba", "nba"]),));

    // 在HYKeywords中查找有没有Text,并且文字是abc
    final abcText = find.text("abc");
    final cbaText = find.text("cba");
    // 在HYKeywords中查找有没有指定的icon
    final icons = find.byIcon(Icons.people);

    // 验证找到一个
    expect(abcText, findsOneWidget);
    expect(cbaText, findsOneWidget);
    // 验证找到三个
    expect(icons, findsNWidgets(3));
  });
}

官方文档中还有更多关于Widget的测试:

三. 集成测试

单元测试和Widget测试都是在测试独立的类或函数或Widget,它们并不能测试单独的模块形成的整体或者获取真实设备或模拟器上应用运行的状态;

这些测试任务可以交给集成测试来完成;

集成测试需要有两个大的步骤:

  1. 发布一个可测试应用程序到真实设备或者模拟器上;
  2. 利用独立的测试套件去驱动应用程序,检查仪器是否完好可用;

3.1. 创建可测试应用程序

我们需要创建一个可以运行在模拟器或者真实设备的应用程序。

这里我直接使用了官方的示例程序,但是不同的是我这里给两个Widget添加了两个Key。

  • 显示数字的Text Widget:ValueKey("counter")
  • 点击按钮的FloatingActionButton Widget:key: ValueKey("increment")
import 'package:flutter/material.dart';

void main() => runApp(MyApp());

class MyApp extends StatelessWidget {
  // This widget is the root of your application.
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: MyHomePage(title: 'Flutter Demo Home Page'),
    );
  }
}

class MyHomePage extends StatefulWidget {
  MyHomePage({Key key, this.title}) : super(key: key);

  final String title;

  @override
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  int _counter = 0;

  void _incrementCounter() {
    setState(() {
      _counter++;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            Text(
              'You have pushed the button this many times:',
            ),
            Text(
              '$_counter',
              key: ValueKey("counter"),
              style: Theme.of(context).textTheme.display1,
            ),
          ],
        ),
      ),
      floatingActionButton: FloatingActionButton(
        key: ValueKey("increment"),
        onPressed: _incrementCounter,
        tooltip: 'Increment',
        child: Icon(Icons.add),
      ), // This trailing comma makes auto-formatting nicer for build methods.
    );
  }
}

3.2. 添加flutter_driver依赖

我们需要用到 flutter_driver 包来编写集成测试,所以我们需要把 flutter_driver 依赖添加到应用 pubspec.yaml 文件的 dev_dependencies 位置:

dev_dependencies:
  flutter_driver:
    sdk: flutter
  flutter_test:
    sdk: flutter
  test: any

3.3. 创建测试文件

和单元测试以及Widget测试不同的是,集成测试的程序和待测试的应用并不在同一个进程内,所以我们通常会创建两个文件:

  • 文件一:用于启动待测试的应用程序
  • 文件二:编写测试的代码

我们可以将这两个文件放到一个文件中:test_driver。
注意:文件夹必须叫test_driver,并且和lib同一层级。

lib/
  main.dart
test_driver/
  app.dart
  app_test.dart

3.4. 编写安装应用代码

安装应用程序代码在app.dart中,分层两步完成:

  • 让 flutter driver 的扩展可用
  • 运行应用程序

test_driver/app.dart 文件中增加以下代码:

import 'package:flutter_driver/driver_extension.dart';
import 'package:test_demo/main.dart' as app;

void main() {
  // 开启DriverExtension
  enableFlutterDriverExtension();

  // 手动调用main函数, 启动应用程序
  app.main();
}

3.5. 编写集成测试代码

现在我们有了待测应用,我们可以为它编写测试文件了。

  • 在 setUpAll() 函数中运行测试案例前,先与待测应用建立连接
  • 测试重要场景
  • 完成测试后,在 teardownAll() 函数中与待测应用断开连接

test_driver/app_test.dart 文件中增加以下代码:

import 'package:flutter_driver/flutter_driver.dart';
import 'package:test/test.dart';

void main() {
  group("Counter App Test", () {
    FlutterDriver driver;

    // 初始化操作,所有测试test开始前,执行的操作
    setUpAll(() async {
      driver = await FlutterDriver.connect();
    });

    // 测试结束操作
    tearDownAll(() {
      if (driver != null) {
        driver.close();
      }
    });

    // 编写测试代码
    final counterTextFinder = find.byValueKey('counter');
    final buttonFinder = find.byValueKey('increment');

    test("starts at 0", () async {
      expect(await driver.getText(counterTextFinder), "0");
    });

    test("on tap click", () async {
      // 点击一下按钮
      await driver.tap(buttonFinder);
      // 期望显示1
      expect(await driver.getText(counterTextFinder), "1");
    });
  });
}

3.6. 运行集成测试

首先,启动安卓模拟器或者 iOS 模拟器,或者直接把 iOS 或 Android 真机连接到你的电脑上。

接着,在项目的根文件夹下运行下面的命令:

flutter drive --target=test_driver/app.dart

这个指令的作用:

  1. 创建 --target 目标应用并且把它安装在模拟器或真机中
  2. 启动应用程序
  3. 运行位于 test_driver/ 文件夹下的 app_test.dart 测试套件

运行结果:我们会发现正常运行,并且结果app中的FloatingActionButton自动被点击了一次,然后界面显示1。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,013评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,205评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,370评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,168评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,153评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,954评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,271评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,916评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,382评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,877评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,989评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,624评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,209评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,199评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,418评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,401评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,700评论 2 345

推荐阅读更多精彩内容