uiautomator2.0+脫離PC運行(apk啟動uiautomator2.0+)的實現(xiàn)方案

效果:

打開MyTest.apk,點擊run uiautomator,就能直接運行你的腳本。

方案概述:

  • 新建一個Android app工程MyTest,在Activity中添加Button,用于啟動腳本

  • 給這個app添加系統(tǒng)簽名

  • 在MyTest中新建一個module,命名為MyTestCase,用于編寫腳本

  • 使用am instrument命令實現(xiàn)腳本的運行

1. 新建一個Android應(yīng)用,命名為MyTest

新建一個Android應(yīng)用

選擇Android的版本,并且選擇empty Activity,一路next到Finish

2. 修改activity_main.xml文件如下:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="cxq.com.mytest.MainActivity">

    <Button
        android:onClick="runMyUiautomator"
        android:id="@+id/runBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="run" />
</RelativeLayout>

3. 修改MainActivity文件如下:

package cxq.com.mytest;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button;

public class MainActivity extends AppCompatActivity {
    private static final String TAG = "MainActivity";
    Button runBtn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        runBtn= (Button) findViewById(R.id.runBtn);
    }

    /**
     * 點擊按鈕對應(yīng)的方法
     * @param v
     */
    public void runMyUiautomator(View v){
        Log.i(TAG, "runMyUiautomator: ");
        new UiautomatorThread().start();
    }

    /**
     * 運行uiautomator是個費時的操作,不應(yīng)該放在主線程,因此另起一個線程運行
     */
    class UiautomatorThread extends Thread {
        @Override
        public void run() {
            super.run();
            String command=generateCommand("cxq.com.testcase", "TestDemo_1", "demo");
           CMDUtils.CMD_Result rs= CMDUtils.runCMD(command,true,true);
            Log.e(TAG, "run: " + rs.error + "-------" + rs.success);
        }

        /**
         * 生成命令
         * @param pkgName 包名
         * @param clsName 類名
         * @param mtdName 方法名
         * @return
         */
        public  String generateCommand(String pkgName, String clsName, String mtdName) {
            String command = "am instrument  --user 0 -w -r   -e debug false -e class "
                    + pkgName + "." + clsName + "#" + mtdName + " "
                    + pkgName + ".test/android.support.test.runner.AndroidJUnitRunner";
            Log.e("test1: ", command);
            return command;
        }
    }
}

其中CMDUtils.java內(nèi)容如下:

package cxq.com.mytest;

import android.util.Log;

import java.io.BufferedReader;
import java.io.InputStreamReader;

/**
 * 執(zhí)行命令
 */
public class CMDUtils {

    private static final String TAG = "CMDUtils";

    public static class CMD_Result {
        public int resultCode;
        public String error;
        public String success;

        public CMD_Result(int resultCode, String error, String success) {
            this.resultCode = resultCode;
            this.error = error;
            this.success = success;
        }

    }

    /**
     * 執(zhí)行命令
     *
     * @param command         命令
     * @param isShowCommand   是否顯示執(zhí)行的命令
     * @param isNeedResultMsg 是否反饋執(zhí)行的結(jié)果
     * @retrun CMD_Result
     */
    public static CMD_Result runCMD(String command, boolean isShowCommand,
                                    boolean isNeedResultMsg) {
        if (isShowCommand)
            Log.i(TAG, "runCMD:" + command);
        CMD_Result cmdRsult = null;
        int result;
        try {
            Process process = Runtime.getRuntime().exec(command);
            result = process.waitFor();
            if (isNeedResultMsg) {
                StringBuilder successMsg = new StringBuilder();
                StringBuilder errorMsg = new StringBuilder();
                BufferedReader successResult = new BufferedReader(
                        new InputStreamReader(process.getInputStream()));
                BufferedReader errorResult = new BufferedReader(
                        new InputStreamReader(process.getErrorStream()));
                String s;
                while ((s = successResult.readLine()) != null) {
                    successMsg.append(s);
                }
                while ((s = errorResult.readLine()) != null) {
                    errorMsg.append(s);
                }
                cmdRsult = new CMD_Result(result, errorMsg.toString(),
                        successMsg.toString());
            }
        } catch (Exception e) {
            Log.e(TAG, "run CMD:" + command + " failed");
            e.printStackTrace();
        }
        return cmdRsult;
    }

}

4. 新建一個Module,用于寫你的測試腳本

  • 在Android Studio的左側(cè)欄,右擊“New -- Module -- Phone&Table Module”,并填寫如下信息
新建一個Module

next之后,選擇add no activity -- finish

5. 修改mytestcase模塊的gradle文件

  • 修改默認(rèn)的runner,在defaultConfig中添加runner
 defaultConfig {
        applicationId "cxq.com.mytestcast"
        minSdkVersion 23
        targetSdkVersion 23
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
  • 在dependencies中添加依賴
dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:23.3.0'
    compile 'junit:junit:4.12'
    compile 'com.android.support.test:runner:0.4.1'
    compile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.2'
}

6. 新建你的測試用例TestOne

新建你的測試用例

代碼如下:

package cxq.com.mytestcast;

import android.support.test.InstrumentationRegistry;
import android.support.test.uiautomator.UiDevice;
import android.support.test.uiautomator.UiObject;
import android.support.test.uiautomator.UiObjectNotFoundException;
import android.support.test.uiautomator.UiSelector;

import org.junit.Test;
import org.junit.runner.RunWith;

@RunWith(AndroidJUnit4.class)
public class TestOne {
    private UiDevice mDevice;

    @Test
    public void demo() throws UiObjectNotFoundException {
        mDevice = UiDevice.getInstance(InstrumentationRegistry.getInstrumentation());
        mDevice.pressHome();
        mDevice.pressHome();
        UiObject x=mDevice.findObject(new UiSelector().text("聯(lián)系人"));
        x.click();

    }
}

Case說明:

@RunWith注解,代表使用什么runner
@Test注解,表示當(dāng)前的方法為測試方法,同理還有其他的@Before等注解,具體case的寫法本文不贅述

運行一下你的case,測試是否有效,運行方法:點擊帶有Test注解的方法左側(cè)綠色三角形

AS運行用例

此時,run窗口會有如下信息:

run窗口信息

如果你只是急于達(dá)到效果,而不注重原理,可以跳過原理分析的閱讀
原理分析
可以看到,窗口中其實運行了3條命令

  • adb push D:\DEVl\MyTest\mytestcast\build\outputs\apk\mytestcast-debug.apk /data/local/tmp/cxq.com.mytestcast
    在我們點擊運行之后,AS會自動將用例打包成apk文件,路徑為
    $工程目錄\build\outputs\apk\工程名-debug.apk
    這個apk文件中就包含著我們的測試用例

  • adb shell pm install -r "/data/local/tmp/cxq.com.mytestcast
    安裝這個apk到手機(jī)

  • adb shell am instrument -w -r -e debug false -e class cxq.com.mytestcast.TestOne#demo cxq.com.mytestcast.test/android.support.test.runner.AndroidJUnitRunner
    使用AndroidJunitRunner啟動你的用例

通過分析這個過程,就知道AS是怎么把用例跑起來的,仿照這個原理就可以自己實現(xiàn)通過apk調(diào)用uiautomator用例,只要讓app中的button的響應(yīng)事件去執(zhí)行am instrument命令即可,但是由于執(zhí)行這個命令也是需要權(quán)限的,因此需要給app添加系統(tǒng)簽名

7. 給APP添加系統(tǒng)簽名

大致過程如下(具體的細(xì)節(jié)可以參照我之前寫過的文章Android Studio自動生成帶系統(tǒng)簽名的apk):
a. 修改App中的manifest.xml文件,添加android:sharedUserId="android.uid.system"

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    android:sharedUserId="android.uid.system"
    package="cxq.com.mytest">

b. 生成js文件
c. 使用keytool-importkeypair對jks文件引入系統(tǒng)簽名
d. 配置gradle(app)

8. 通過app啟動uiautomator

運行MyTest下的app,界面如下:


運行圖

點擊RUN,則回到桌面,并且打開桌面上的聯(lián)系人,至此,通過apk啟動uiautomator教程完畢。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,030評論 25 709
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,554評論 19 139
  • 較之寶哥哥、林妹妹催人淚下的愛情悲劇,王熙鳳和賈璉夫婦的婚姻悲劇更讓人唏噓不已,耐人尋味。王熙鳳,賈璉,從屬于四大...
    紅閣樓的夢閱讀 5,616評論 20 46
  • 最近鄰居的嫂子說她2歲多的兒子已經(jīng)連續(xù)幾個晚上睡著睡著就突然驚叫大哭,哄了一會睡著了沒多久又驚叫著醒過來。提醒她說...
    擁抱幸福a閱讀 1,946評論 0 0
  • 心燈 文/舟亮 比翼 舞動孤鴻影成雙 顫抖著 捧出那輪金黃 端詳你的模樣 呢喃 羞赧漫天云裳 耳畔的諾言 消逝著青...
    舟亮閱讀 352評論 0 1

友情鏈接更多精彩內(nèi)容