面对android studio Run 一次项目要等好几分钟的痛点,不得不研究一下android studio 的单元测试。
其实我的目的很简单,在不对视图进行操作的前提下,测试一些activity 的生命周期,或网络拉取数据的一些处理,比如解析 json 数据啊,做网络请求啊等等,也就是对 Model层的测试。这些不需要操作视图,但在没有单元测试环境下,比如我们网络请求一些数据,Log 打印看看是否请求成功,却又要 利用模拟器或真机Run 一次项目,花费好几分钟,这是不能容忍的。
于是乎,强大的 android studio 也考虑到了这一点,给我们提供的简单的单元测试类。
让我们来简单的了解学习一下吧。
首先先来了解一下一些名称,方便下面介绍和使用:
在java中咱们有用过 JUnit 的 单元测试 ,那android 也是基于 java 语言编写的,所以也有个 JUnit的单元测试。在做 android 的单元测试需要导入依赖:
androidTestCompile 'junit:junit:4.12'
testCompile 'junit:junit:4.12'
其中, test目录为在本机执行单元测试代码的目录, androidTest为在Android设备上执行单元测试代码的目录。如下图:
Android 自带的 junit单元测试的一些测试类(androidTest测试 需要运行在模拟机或真机上)
1、InstrumentationTestCase框架:
Instrumentation和Activity有点类似,只不过Activity是需要一个界面的,而Instrumentation并不是这样的,我们可以将它理解为一种没有图形界面的,具有启动能力的,用于监控其他类(用Target Package声明)的工具类。
举个例子,利用InstrumentationTestCase 启动一个activity:
在androidTest下新建一个java类,并且继承自InstrumentationTestCase编写一个public void的方法,但是必须要是方法名以test打头,比如testPublishSubject,并不需要@Test注解
public class TestSubject extends InstrumentationTestCase { private static final String LOG_TAG = "test"; public void testPublishSubject() { launchActivity("demo.zts.com.demo",SecondActivity.class,null); } }
2、ApplicationTestCase——测试整个应用程序的类。它允许你注入一个模拟的Context到应用程序中,在应用程序启动之前初始化测试参数,并在应用程序结束之后销毁之前检查应用程序。
使用Context,你可以浏览资源,文件,数据库等等。基类是AndroidTestCase,一般常见的是它的子类,和特定组件关联。
测试代码如下:
public class MyApp extends Application { @Override public void onCreate() { super.onCreate(); String app_name = getResources().getString(R.string.app_name); Log.i("MyApp",".........MyApp....app_name.........."+app_name); } } public class ApplicationTest extends ApplicationTestCase<MyApp> { public ApplicationTest() { super(MyApp.class); } public void testStart() { String str = null; str = mContext.getResources().getString(R.string.app_name); Log.i("..",".............ApplicationTest ...........app_name............."+str); }
Log 日志:
07-22 23:27:10.276 32259-32259/demo.zts.com.demo I/MyApp: .........MyApp....app_name..........demo
07-22 23:27:10.276 32259-32319/demo.zts.com.demo I/TestRunner: started: testStart(demo.zts.com.demo.ApplicationTest)
07-22 23:27:10.286 32259-32319/demo.zts.com.demo I/..: .............ApplicationTest..........app_name..............demo
3、ActivityUnitTestCase——对单个Activity进行单一测试的类。使用它,你可以注入模拟的Context或Application,或者两者。它用于对Activity进行单元测试。也就是说你可以用于测试单独的activity ,虽然也需要利用模拟机或真机启动,但你启动的只是你需要做测试的activity,于其他activity无关。
测试代码如下:
要测试的 activity
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); System.out.println("...............MainActivity......onCreate............"); Log.i("MainActivity","................onCreate............................"); } @Override protected void onStart() { super.onStart(); System.out.println("...............MainActivity......onStart............"); Log.i("MainActivity","................onStart............................"); } @Override protected void onStop() { super.onStop(); System.out.println("...............MainActivity......onStop............"); Log.i("MainActivity","................onStop............................"); } @Override protected void onDestroy() { super.onDestroy(); System.out.println("...............MainActivity......onDestroy............"); Log.i("MainActivity","................onDestroy............................"); } }
测试类
public class TestActivity extends ActivityInstrumentationTestCase2<MainActivity> { private Context ctx; public TestActivity() { super(MainActivity.class); } @Override protected void setUp() throws Exception { super.setUp(); ctx = getActivity().getApplicationContext(); } public void testStart() { Intent intent = new Intent(ctx, MainActivity.class); intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); ctx.startActivity(intent); Log.i("TestActivity","................startActivity............................"); }
测试 Log 日志:
* 07-22 23:39:44.146 3171-3171/demo.zts.com.demo I/System.out: ...............MainActivity......onCreate............
07-22 23:39:44.146 3171-3171/demo.zts.com.demo I/MainActivity: ................onCreate............................
07-22 23:39:44.151 3171-3171/demo.zts.com.demo D/MZPerfObserver: demo.zts.com.demo onCreate consume 153 ms
07-22 23:39:44.151 3171-3171/demo.zts.com.demo I/System.out: ...............MainActivity......onStart............
07-22 23:39:44.151 3171-3171/demo.zts.com.demo I/MainActivity: ................onStart............................
07-22 23:39:44.326 3171-3171/demo.zts.com.demo D/OpenGLRenderer: Enabling debug mode 0
07-22 23:39:44.361 3171-3171/demo.zts.com.demo I/System.out: ...............MainActivity......onStop............
07-22 23:39:44.361 3171-3171/demo.zts.com.demo I/MainActivity: ................onStop............................
07-22 23:39:44.421 3171-3224/demo.zts.com.demo I/TestActivity: ................startActivity............................
还有很多常见的测试,比如ServiceTestCase,ProviderTestCase2等,大家需要慢慢琢磨。
Android 自带的 junit单元测试的一些测试类(test 测试 ,不需要模拟机,电脑直接运行)
比如我需要测试一段java代码,而这段java代码跟android没关系,也就是不用到android的资源,如context,activity 等,说白了就是简单的 java 测试,当然,嘿嘿,android studio也是可以做java代码测试的。
测试代码如下,测试 4+4 等于几:
public class ExampleUnitTest { @Test public void testAdd() { int i = 0; i = 4+4; System.out.print(".............. "+i); Log.i("TAG","..................."+i); // 比较 i 是否 等于 8 ,相等的话通过测试!!! Assert.assertEquals(8, i); } }
测试成功:
以上测试类的运行是 -点击测试右键 - 选择 RunXXXXX
/*********************华丽分割线***********************/
看了半天好像也没有解决文章最初提到的一个痛点啊,就是我需要测试android的资源,但又不想运行笨重的模拟机或真机,怎么办呢? 妈蛋,被骗了,还钱 -_-、、、 确实,上面提到的测试方法虽然没有解决拜托模拟机测试的痛点,但基于模拟机单元测试的 androidTest 确实方便我们做一些 单独功能的测试,而且能做 UI 测试,因为需要模拟机或真机嘛,所以 UI 或视图测试是没问题的。 还有test 测试,可以做一些不需要android资源的 java代码测试,也是在android开发当中很方便的,不用在启用eclipse 做测试,直接android studio 既可以了。
忽悠,接着 忽悠 -_-////
其实要想脱离 模拟机或真机,又要做使用android资源的测试,如 使用Context,浏览资源,文件,数据库等等。 也是可以的!!! 那 就只有第三方测试框架了 Robolectric
666,你是来做宣传的吗 -_-、、、不过真的很好用,也能很好的解决咱们的痛点。
接下来利用个需求来讲解 Robolectric 测试,免得我忽悠你们。
拿到 android 目录下的 assets 下的json01.txt文件 是一段json数据,让后进行解析,解析后将数据显示。 分析:这个需求就跟android下的资源有关,而咱们利用 Robolectric 做单元测试,并且不需要模拟机或真机的支持。
其中json数据
{ "name": "coolxing", "age": 24, "male": true, "address": { "street": "huiLongGuan", "city": "beijing", "country": "china" } }
首先需要 Robolectric 依赖,在你的 app module 下注入依赖:
testCompile 'org.robolectric:robolectric:3.0'
注意是 testCompile 而不是 androidTestCompile ,不然你有需要启动模拟器了。并且测试类也是 在 test 下的
测试类:
@RunWith(RobolectricGradleTestRunner.class) @Config(constants = BuildConfig.class, sdk = 21) public class MainActivityTest2 { @Test public void testJson(){ String str = null; str = RuntimeEnvironment.application.getResources().getString(R.string.app_name); AssetManager am = null; am = RuntimeEnvironment.application.getAssets(); String strData = null; try { InputStream inputStream = am.open("json01.txt"); byte buf[] = new byte[1024]; inputStream.read(buf); strData = new String(buf); strData =strData.trim(); strData.trim(); } catch (IOException e) { } jsonBean foo = new Gson().fromJson(strData, jsonBean.class); System.out.println("...............json.................."+foo.name); System.out.println("...............json.................."+foo.address); System.out.println("...............json.................."+foo.age); } }
测试结果:
看,咱们利用application 拿到 android 下的资源,但又不像刚才上面的 androidTestCompile 需要模拟机,是不是很6,我电脑配置比较低,本次测试需要40S多,但不真机快多了。
am = RuntimeEnvironment.application.getAssets();
需要注意几点,类头部需要声明 @ 注解:
@RunWith(RobolectricGradleTestRunner.class)
@Config(constants = BuildConfig.class, sdk = 21)
并且测试方法是以 textxxx() 开头的,如上面的 testJson() ,方法也需要@Test注解!!!
Robolectric 还可以测试 activity ,如:
@RunWith(RobolectricGradleTestRunner.class) @Config(constants = BuildConfig.class, sdk = 21) public class MainActivityTest2 { @Test public void testMainActivity() { MainActivity mainActivity = Robolectric.setupActivity(MainActivity.class); mainActivity.findViewById(R.id.main_tv).performClick(); Intent expectedIntent = new Intent(mainActivity, SecondActivity.class); ShadowActivity openActivity = Shadows.shadowOf(mainActivity); Intent actualIntent = openActivity .getNextStartedActivity(); // Assert.assertEquals(expectedIntent, actualIntent); }
其中
MainActivity mainActivity = Robolectric.setupActivity(MainActivity.class);
这句代码就是启动了MainActivity 的生命周期
Robolectric 单元测试类 的 启动 也是跟 上面test 测试类一样,选择 -MainActivityTest2 --右键 -- 选择 Run MainActivityTest2
好了,单元测试就介绍到这里,
其实我也只是初步理解,上面那些基本的也是我做项目的需要我才去学习使用的,还有好多强大的功能大家慢慢探索。