2024年Android最全Android Unit Test 入门,BAT大厂面试基础题集合

文章介绍了Android高级进阶视频、架构师项目实战、以及Robolectric和Espresso等测试工具的使用。强调了系统学习的重要性,提供了一个包括测试、架构、UI优化等多方面的知识体系,鼓励IT从业者和新人加入技术交流圈子以共同成长。
摘要由CSDN通过智能技术生成

最后

在此为大家准备了四节优质的Android高级进阶视频:

架构师项目实战——全球首批Android开发者对Android架构的见解

附相关架构及资料

image.png

往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

}

@Test

fun testStudentBorrowBook() {

val library = mockk()

val student = new Student(libaray)

//需要通过图书馆借书时直接返回一本书

every {libaray.borrowBook() } returns Book(“book name”)

every {library.swipeCard(student.bookCard) } just Runs

student.borrowBook()

verify {library.swipeCard(student.bookCard) }

}

Robolectric

如果我们想测试一些Android特性的东西,如启动application,获取UI信息等,但是又不想连接设备去测试,这时候就需要Robolectric了。

testImplementation ‘org.robolectric:robolectric:4.3.1’

android {

testOptions {

unitTests.includeAndroidResources = true

}

}

下面是一个简单的测试示例

@RunWith(RobolectricTestRunner::class)

@Config(sdk = [Build.VERSION_CODES.P], application = TestApplication::class)

class ExampleUnitTest {

@Test

fun testActivity() {

val activity = Robolectric.buildActivity(MainActivity::class.java).create().start().resume().get()

val textView = activity.findViewById(R.id.main_hello_text_view)

assertEquals(textView.text, “Hello World!”)

}

}

启动某个activity会先启动Application,正常项目里Application会有很多初始化动作,通常我们为了方便可以在config指定一个TestApplication,config里还可以指定其他一些配置,比如manifest。

buildActivity

启动Activity的方法。Robolectric.buildActivity(MainActivity::class.java)该方法会返回ActivityController对象,通过该对象可以方便的控制Activity的生命周期。

shadowOf()

每个shadow对象都可以认为是对应控件的测试方法库。

方法里面可以传入很多常用的控件类型返回相应的shadow对象,比如Application,Activity,ListView,AlertDialog,Theme等。

比如Activity和ListView提供了以下的方法

@RunWith(RobolectricTestRunner::class)

@Config(sdk = [Build.VERSION_CODES.P], application = TestApplication::class)

class ExampleUnitTest {

@Test

fun testActivity() {

val activity = Robolectric.buildActivity(MainActivity::class.java).create().start().resume().get()

val textView = activity.findViewById(R.id.main_hello_text_view)

assertEquals(textView.text, “Hello World!”)

val activityShadow = shadowOf(activity)

//可以动态授权

activityShadow.grantPermissions(Manifest.permission.READ_SMS)

//假设点击后会跳转到SecondActivity

textView.performClick()

val nextActivityName = activityShadow.nextStartedActivity.component?.className

//验证点击后跳转的activity是否正确

assertEquals(nextActivityName, SecondActivity::class.java.simpleName)

}

}

@Test

fun testListView() {

val activity = Robolectric.buildActivity(MainActivity::class.java).create().start().resume().get()

val listView = activity.findViewById(R.id.main_hello_text_view)

val listViewShadow = shadowOf(listView)

//需要先调populateItems才能正确绘制listView

listViewShadow.populateItems()

//点击第一个item

listViewShadow.clickFirstItemContainingText(“First”)

}

具体使用的时候可以查看shadow对象源码了解其提供了哪些方法。

fragment test

Robolectric提供了buildFragment(class)的方法

但是如果我们使用的是androidx的fragment是没办法使用该方法直接创建 fragment的,这时如果我们想简单的启动一个fragment需要另外引入一个支持库

debugImplementation ‘androidx.fragment:fragment-testing:1.2.3’

testImplementation “androidx.test:core:1.2.0”

该库提供了直接启动fragment和activity的方法

//fragment会放入一个emptyActivity之中,并且直接到onResume状态。

val fragment = launchFragment(themeResId = R.style.AppTheme, factory = FragmentFactory())

//启动一个activity,并且直接到onResume状态

val activityScenario = ActivityScenario.launch(MainActivity::class.java)

activityScenario.onActivity { activity ->

}

这种启动方式的缺点是状态无法控制,直接到resume,如果onCreate等生命周期中有需要mock的对象则难以控制。如果想较为灵活的控制就可以通过Android的API去加载fragment

@Test

fun testFragment() {

val activityController = Robolectric.buildActivity(MainActivity::class.java).create()

val fragment = TestFragment()

activityController.get().supportFragmentManager.beginTransaction().add(R.id.fragment_container, fragment).commit()

activityController.start().resume()

}

Espresso

设备化测试库,可以方便的进行UI操作

onView

可以通过id或者文本内容找到相应的view

onView(withId(R.id.bottom)).perform(click())

onView(withText(“next”)).perform(click())

perform

可以模拟各种用户操作,点击、长按、输入文本、滑动等

//输入文本

onView(withId(R.id.bottom)).perform(typeText(“Steve”))

《设计思想解读开源框架》

第一章、 热修复设计

  • 第一节、 AOT/JIT & dexopt 与 dex2oat

  • 第二节、 热修复设计之 CLASS_ISPREVERIFIED 问题

  • 第三节、热修复设计之热修复原理

  • 第四节、Tinker 的集成与使用(自动补丁包生成)

    第二章、 插件化框架设计

  • 第一节、 Class 文件与 Dex 文件的结构解读

  • 第二节、 Android 资源加载机制详解

  • 第三节、 四大组件调用原理

  • 第四节、 so 文件加载机制

  • 第五节、 Android 系统服务实现原理

    第三章、 组件化框架设计

  • 第一节、阿里巴巴开源路由框——ARouter 原理分析

  • 第二节、APT 编译时期自动生成代码&动态类加载

  • 第三节、 Java SPI 机制

  • 第四节、 AOP&IOC

  • 第五节、 手写组件化架构

    第四章、图片加载框架

  • 第一节、图片加载框架选型

  • 第二节、Glide 原理分析

  • 第三节、手写图片加载框架实战

    第五章、网络访问框架设计

  • 第一节、网络通信必备基础

  • 第二节、OkHttp 源码解读

  • 第三节、Retrofit 源码解析

    第六章、 RXJava 响应式编程框架设计

  • 第一节、链式调用

  • 第二节、 扩展的观察者模式

  • 第三节、事件变换设计

  • 第四节、Scheduler 线程控制

    第七章、 IOC 架构设计

  • 第一节、 依赖注入与控制反转

  • 第二节、ButterKnife 原理上篇、中篇、下篇

  • 第三节、Dagger 架构设计核心解密

    第八章、 Android 架构组件 Jetpack

  • 第一节、 LiveData 原理

  • 第二节、 Navigation 如何解决 tabLayout 问题

  • 第三节、 ViewModel 如何感知 View 生命周期及内核原理

  • 第四节、 Room 架构方式方法

  • 第五节、 dataBinding 为什么能够支持 MVVM

  • 第六节、 WorkManager 内核揭秘

  • 第七节、 Lifecycles 生命周期


    本文包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

cles 生命周期**

[外链图片转存中…(img-i4eYYfiU-1714823402169)]
本文包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…
[外链图片转存中…(img-tilVX51M-1714823402170)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 7
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值