单元测试mockk

单元测试

基于kotlin的mockk: https://mockk.io/

Mockito: 侧重点是纯Java代码的测试:方法调用mock,指定方法行为,截取参数,截取Callback回调

PowerMock : 支持JUnit和TestNG,扩展了EasyMock和Mockito框架,增加了mock private、static、final方法的功能。

UI自动化测试框架: 

Robolectric这是一个专门测试android系统类相关的东西的,比如UI点击事件,Actvity和Service等四大组件生命
周期,Dialog,Application,Bitmap,Resouce等进行行为测试,它之所以能测试他们,其实也是Mock了这些对
象,然后可以动态代理这些对象,不过他用的不是mock,而是ShadowXX,Android系统的每个几乎都有一个相对应
的ShadowXX,我们测试的时候得到对应的ShadowXX就可以像操作傀儡一样操纵原来的对象了,就连Loop都有
ShadowLoop专门用来测试的时候操作和调度线程的。

PowerMockito + Robolectric

Espresso

这里讲一下mockk对私用属性和方法的测试

public class Student {
    private String name = null;
    private String getName() {
        return name;
    }
    private void setName(String name) {
        this.name = name;
    }
    
    public void run(String feed) {
        System.out.println("run: " + feed);
        String result = talk();
        if (!TextUtils.equals(feed, result)) {
            feed = result;
        }
        if (name != null) {
            feed += " , " + name;
        }
        Log.i("@#@", "feed = " + feed);
        System.out.println("run end: " + feed);
    }

    private String talk() {
        System.out.println("talk");
        return "talk or run";
    }
}

测试用例:

class StudentTest {
    @ParameterizedTest
    @CsvSource(
        value = [
            "true, true, talkOrRun", // equalsValue为true,后面两个参数没意义
            "false, false, talkOrRun", // everyTalk为false,后面的参数没意义
            "false, true, talkOrRun" // talkReturn 决定最后的日志结果
        ]
    )
    /**
     * equalsValue  影响 run() 的输出是否和入参一致
     * everyTalk    是否对 talk() 方法做every操作
     * talkReturn   talk() 方法做every操作时的返回值
     */
    fun test_run(equalsValue: Boolean, everyTalk: Boolean, talkReturn: String) {
        // solved io.mockk.MockKException: no answer found for: Context(#1).getApplicationContext()
        // val mContextMock = mockk<Context>(relaxed = true)
        mockkStatic(Log::class)
        every<Int> { Log.i(any(), any()) } returns 0
        mockkStatic(TextUtils::class)
        every { TextUtils.equals(any(), any()) } returns equalsValue

        val mockStudent = spyk(Student(), recordPrivateCalls = true)
        // 设置私有属性的值
        InternalPlatformDsl.dynamicSet(mockStudent, "name", "Nancy")
        // 获取私有属性的值
        Assert.assertEquals(InternalPlatformDsl.dynamicGet(mockStudent, "name"), "Nancy")
        if (everyTalk) {
            // every插桩,对方法设定返回值,talk()方法日志没打印
            every { mockStudent.invokeNoArgs("talk") } returns talkReturn
        }
        // 调用私有方法返回值
        var value = InternalPlatformDsl.dynamicCall(mockStudent, "getName", arrayOf()) { mockk() }
        Assert.assertEquals(value, "Nancy")
        println("value: $value\n")
        // 调用私有方法设置值
        InternalPlatformDsl.dynamicCall(mockStudent, "setName", arrayOf("newName")) { mockk() } // 后面参数是协程相关的参数这里用不上
        // 调用私有方法返回值
        value = InternalPlatformDsl.dynamicCall(mockStudent, "getName", arrayOf()) { mockk() }
        println("new value: $value\n")
        //  真实调用方法
        mockStudent.run("feed")
        // 验证是否执行私有方法
        verify { mockStudent.invokeNoArgs("talk") }
    }
}

对ui文件中对单测用例

public AutoBannerLayout(Context context) {
        super(context);
        onCreateView(context);
    }

    protected void initView(View contentView) {
        adjustBannerHeight(mContentView);

        mDotNumberView = contentView.findViewById(R.id.banner_dot);
        mBannerViewPager = contentView.findViewById(R.id.banner_viewpager);
        mBannerAdapter = new AutoBannerAdapter();

        mBannerViewPager.setOnItemSelectedListener(new AutoBannerViewPager.OnItemSelectedListener() {
            @Override
            public void onItemSelected(int position) {
               
            }
        });

        mBannerAdapter.setImageViewEventCallback(new AutoBannerAdapter.ImageViewEventCallback() {
            @Override
            public void onItemClick(int position) {

            }

            @Override
            public void onItemLoadFailed(int position) {

            }
        });
        mBannerViewPager.setAdapter(mBannerAdapter);
    }
public class AutoBannerAdapter extends PagerAdapter {

    private final ArrayList<ImageView> mViewCaches = new ArrayList<>();
    private List<stMetaBanner> mBannerList = new ArrayList<>();
    private ImageViewEventCallback imageViewEventCallback;
    private static final int MULTIPLE_BANNER_SIZE = 60;

    public void setData(List<stMetaBanner> banners) {
        mBannerList.clear();
        mBannerList.addAll(banners);
        notifyDataSetChanged();
    }

    @Override
    public Object instantiateItem(ViewGroup container, final int position) {
        if (mBannerList != null && mBannerList.size() > 0) {
            int realSize = getItemCount();
            ImageView imageView = null;
            stMetaBanner banner = mBannerList.get(position % realSize);
            if (mViewCaches.isEmpty()) {
                imageView = new ImageView(container.getContext());
                ViewGroup.LayoutParams params = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
                imageView.setLayoutParams(params);
                imageView.setScaleType(ImageView.ScaleType.FIT_CENTER); 
            } else {
                // 当缓存集合有数据时,复用,然后缓存不再持有它的引用
                imageView = mViewCaches.remove(0);
            }
            if (banner == null && imageViewEventCallback != null) {
                imageViewEventCallback.onItemLoadFailed(position % realSize);
                return imageView;
            }

            loadCoverImage(position, realSize, imageView, banner);

            imageView.setTag(R.id.tag_first, banner);
            container.addView(imageView);

            handleClickEvent(position, realSize, imageView);

            return imageView;
        }

        return null;
    }

    /**
     * 处理点击事件
     */
    private void handleClickEvent(final int position, final int realSize, ImageView imageView) {
        if (imageView == null) {
            return;
        }
        imageView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (realSize == 0) {
                    return;
                }
                int realPosition = position % realSize;
                if (imageViewEventCallback != null) {
                    imageViewEventCallback.onItemClick(realPosition);
                }
            }
        });
    }
}

单测用例

  @Test
    fun onCreateView() {
        val contextMock = mockk<Context>(relaxed = true)
        mockkStatic(LayoutInflater::class)
        mockkStatic(DeviceUtils::class)

        // Can't instantiate proxy via default constructor for class android.view.LayoutInflater
        val layoutInflaterMock = mockk<LayoutInflater>()
        every { LayoutInflater.from(any()) } returns layoutInflaterMock
        every { DeviceUtils.getScreenWidth() } returns 1
        val contentViewMock = spyk(View(contextMock))
        val dotNumberViewMock = mockk<DotNumberView>()
        val autoBannerViewPagerMock = mockk<AutoBannerViewPager>()
        every { contentViewMock.findViewById(R.id.banner_dot) as DotNumberView } returns dotNumberViewMock
        every { contentViewMock.findViewById(R.id.banner_viewpager) as AutoBannerViewPager } returns autoBannerViewPagerMock
        every { layoutInflaterMock.inflate(any<Int>(), any<ViewGroup>(), any<Boolean>()) } returns contentViewMock
        every { contentViewMock.layoutParams } returns mockk()
        every { contentViewMock.layoutParams = any() } just runs

        every { autoBannerViewPagerMock.setOnItemSelectedListener(any()) } just runs
        every { autoBannerViewPagerMock.adapter = any() } just runs

        autoBannerViewPagerMock.setOnItemSelectedListener(AutoBannerViewPager.OnItemSelectedListener {
                position -> println("onItemSelected: $position")
        })

        val autoBannerLayoutMock = spyk(AutoBannerLayout(contextMock) )
        val autoBannerAdapterMock = InternalPlatformDsl.dynamicGet(autoBannerLayoutMock, "mBannerAdapter") as AutoBannerAdapter
        Assert.assertNotNull(autoBannerAdapterMock)

        every { dotNumberViewMock.visibility = any() } returns Unit
        every { autoBannerViewPagerMock.visibility = any() } returns Unit
        every { autoBannerViewPagerMock.stopAutoPlay() } returns Unit

        val banner = stMetaBanner(0, "", "", 1, 1, "aaa");
        val list = ArrayList<stMetaBanner>()
        list.add(banner)
        autoBannerLayoutMock.setData(list, 0)
        verify { autoBannerViewPagerMock.stopAutoPlay() }
    }

    @Test
    fun instantiateItem() {
        val autoBannerViewPagerMock = spyk(AutoBannerAdapter())
        val banner = stMetaBanner(0, "bbb", "单元测试", 1, 1, "aaa");
        val list = ArrayList<stMetaBanner>()
        list.add(banner)
        // 设置私有属性的值
        InternalPlatformDsl.dynamicSet(autoBannerViewPagerMock, "mBannerList", list)
        every { autoBannerViewPagerMock.instantiateItem(any(), any()) } returns mockk<ImageView>()

        every { autoBannerViewPagerMock.invoke("loadCoverImage") withArguments listOf(any<Int>(), any<Int>(), any<ImageView>(), any<stMetaBanner>()) } returns Unit
        every { autoBannerViewPagerMock.invoke("handleClickEvent") withArguments listOf(any<Int>(), any<Int>(), any<ImageView>()) } answers {  }

        val viewGroupMock = mockk<ViewGroup>()
        val result = autoBannerViewPagerMock.instantiateItem(viewGroupMock, 0)
        Assert.assertTrue(result is ImageView)
    }

    @Test
    fun handleClickEvent() {
        // objMock should be mock or a spy to call every { ... } block on it.
        // objMock should be mock or a spy to call verify { ... } block on it.
        val autoBannerViewPagerMock = spyk(AutoBannerAdapter())
        val banner = stMetaBanner(0, "bbb", "单元测试", 1, 1, "aaa");
        val list = ArrayList<stMetaBanner>()
        list.add(banner)
        // 设置私有属性的值
        InternalPlatformDsl.dynamicSet(autoBannerViewPagerMock, "mBannerList", list)
        val imageViewEventCallback = object : AutoBannerAdapter.ImageViewEventCallback {
            override fun onItemLoadFailed(position: Int) {
            }
            override fun onItemClick(position: Int) {
                println("onItemClick")
            }
        }
        autoBannerViewPagerMock.setImageViewEventCallback(imageViewEventCallback)
        val imageViewMock = spyk(ImageView(mockk<Context>()))
        InternalPlatformDsl.dynamicCall(autoBannerViewPagerMock, "handleClickEvent", arrayOf(0, 1, imageViewMock)) { mockk() }
        every { imageViewMock.callOnClick() }.run { imageViewEventCallback.onItemClick(0) }
        // io.mockk.MockKException: no answer provided for ImageView(#3).callOnClick())
        every { imageViewMock.callOnClick() } returns true
        // imageViewMock是mock的对象,单独调用有返回值的方法时,需求提前声明返回值
        // 这里的测试没意义,不是执行原文件中的回调
        imageViewMock.callOnClick()
    }

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值