pu369com

重新梳理一下adb操作app(golang版)

主要参考我之前整理的内容https://www.cnblogs.com/pu369/p/10490668.html,梳理简化一下思路,以便于用最简单的代码来应对无聊人士的要求。

需求主要是:打开手机、切换到主页面、找页面关键字、点击、滑动、返回、杀死APP;当然,延时是必须的。

另外我的手机特点是:取消了密码开机,用电源键开机后有广告、但点按home键后会显示主页面

步骤:

1、新建main.go  (有些引用的包可能不需要,具体的一些函数定义在后面)如下:

package main

import (
    "crypto/md5"
    "fmt"
    "image/png"
    "io/ioutil"

    "log"
    "os"
    "regexp"
    "strings"

    "bytes"
    "os/exec"
    "strconv"
    "time"

    "github.com/robfig/cron"
)


func main() {
    dowork()
    c := cron.New()
    c.AddFunc("0 0 12 * * ?", dowork)
    c.Start()
    select {}

}

func dowork(){
...
}

...

说明:main函数主要是用select{}让程序永远执行下去;同时,引用github.com/robfig/cron (可参考:https://www.cnblogs.com/liuzhongchao/p/9521897.html)  使程序每天12点定时执行。(当然,也可参考这里自己实现定时功能:https://www.jianshu.com/p/4955e5d652ae

2、dowork()函数仅用于调用后面的函数

func dowork() {
//点击
AdbShellInputTap()
//延时
TimeSleepDuration()
... 
}

3、延时函数(等待页面加载、或APP要求用户查看多长时间,时间单位为秒)

func TimeSleepDuration(x int) {
    time.Sleep(time.Duration(x) * time.Second)
}

4、打开(关闭)手机电源(为简化就不判断手机是否休眠了,其实是模拟按键,我会找些键值附在本文最后面)

AdbShellInputKeyEvent("26") //power
func AdbShellInputKeyEvent(s string) {
    exec.Command("adb", "shell", "input", "keyevent", s).Run()
}

5、切换到手机主页面

由我的手机没密码,打开电源后有广告,按home会回到主页面,

就用步骤4的AdbShellInputKeyEvent函数模拟home键,在代码中可以多按两次。

AdbShellInputKeyEvent("3") //home

常的还有:  4表示back返回键

6、打开某个APP-直接用下面的AdbShellInputTap(X坐标、Y坐标)函数模拟点击

打开手机  设置-通用-开发者选项-指针位置。将APP放在手机主页面,然后点按APP图标的中心位置查看坐标 X,Y

func AdbShellInputTap(x, y int) {
    x2 := strconv.Itoa(x)
    y2 := strconv.Itoa(y)
    exec.Command("adb", "shell", "input", "tap", x2, y2).Run()
}

7、想点击页面包含某个“关键字”的区域

7.1  首先要将手机屏幕显示页面的源码截取到电脑上,用AdbShellUiautomatorDump()得到window_dump.xml并复制到了当前go程序所在的目录(由于从手机复制到电脑需要时间,所以在这个代码中加了延时2秒,以等待复制完成。实际使用中有可能要根据你的设备速度调整这里的延时时间)

func AdbShellUiautomatorDump() {
    //删除当前目录下的window_dump.xml
    exec.Command("cmd", "/c", "del", "-y", "window_dump.xml").Run()
    //重新dump
    exec.Command("adb", "shell", "uiautomator", "dump", "/sdcard/window_dump.xml").Run()
    exec.Command("adb", "pull", "/sdcard/window_dump.xml", ".").Run()
    exec.Command("adb", "shell", "rm", "/sdcard/window_dump.xml").Run()
  TimeSleepDuration(2)
}

7.2  点击“关键字”,需要指明点击找到的第几个(用正则表达式匹配),用0表示第一个。通常是Tap("关键字",0)

func Tap(s string, ix int) {
    //先执行AdbShellUiautomatorDump函数。
    AdbShellUiautomatorDump()
file, _ := os.Open("window_dump.xml") defer file.Close() doc, _ := ioutil.ReadAll(file) doc1 := string(doc) ss := fmt.Sprintf("%s%s%s", `<node.[^>]+?`, s, `.[^>]+?\[(\d+),(\d+)\]\[(\d+),(\d+)\].+?>`) r := regexp.MustCompile(ss) match := r.FindAllStringSubmatch(doc1, -1) le := len(match) //匹配到1个或多个,ixx表示匹配到的第几个 ixx := ix if le == 0 { log.Println("未匹配到:", s) return } if ix < 0 { ixx = le + ix } if ixx < 0 { ixx = 0 } if ixx >le { ixx = le } x1, _ := strconv.Atoi(fmt.Sprint(match[ixx][1])) y1, _ := strconv.Atoi(fmt.Sprint(match[ixx][2])) x2, _ := strconv.Atoi(fmt.Sprint(match[ixx][3])) y2, _ := strconv.Atoi(fmt.Sprint(match[ixx][4])) xx := (x2-x1)/2 + x1 yy := (y2-y1)/2 + y1 //log.Println(s) AdbShellInputTap(xx, yy) }

8、更复杂的需求,如:

主页面上有N个链接(每个链接的特点是:包括年-月-,类似2019-1-),点第1个,打开一篇文章,看几分钟后,返回;类似的,点第2个链接.......最后,点开第N个链接,查看几分钟后,返回主页面,再向下翻页4次。

实现:开始以为要用递归,后来发现在golang中用for循环即可。为了层次清晰,分成以下几个函数

8.1在步骤2的dowork函数中,写以下循环,表示主页面要向下翻4次(页)

    //学习文章
    for i := 0; i < 5; i++ {
        fmt.Println("article", i)
        getArticles()
    }

8.2 getArticles中,首先findxxbydate(我用xx表示链接的意思:-)用正则查找当前页中,所有包含包括年-月-,即类似2019-1-的链接,存入二维切片[][]string中。

然后用for循环打开每个链接指向的文章,并停留350秒,然后按返回键。当页面所有链接循环操作完毕后,将主页面向上滚动1000px(这个值根据手机页面的实际滚动区域高度来确定。getArticles执行完毕后,进入下一个主循环。

getArticles代码:

func getArticles() {
    //找当前UI中“年-月-”,写入[][]string
    xx := findxxbydate()
    for _, v := range xx {
        //循环打开
        TapRegion4xx(v, 350)
    }
    //向上滚动1000
    AdbShellInputSwipe(500, 1765, 500, 355)
    TimeSleepDuration(6)
}

8.3 上面的findxxbydate用于查找当前手机页面上所有匹配正则表达式的位置,存入二维切片(要根据需要修改ss正则表达式)

func findxxbydate() [][]string {
    //先执行AdbShellUiautomatorDump函数。
    AdbShellUiautomatorDump()
    file, _ := os.Open("window_dump.xml")
    defer file.Close()
    doc, _ := ioutil.ReadAll(file)
    doc1 := string(doc)
    ss := `<node.[^>]+?\d{4}-\d{2}-.[^>]+?\[(\d+),(\d+)\]\[(\d+),(\d+)\].[^>]+?>`
    r := regexp.MustCompile(ss)
    match := r.FindAllStringSubmatch(doc1, -1)
    return match
}

8.4   TapRegion4xx(v, 350)用在步骤8.2的for range循环中,用于点击“年-月-”数组中的每一个链接,并在点开后,停留350秒,后返回

func TapRegion4xx(match []string, t int) {
    x1, _ := strconv.Atoi(fmt.Sprint(match[1]))
    y1, _ := strconv.Atoi(fmt.Sprint(match[2]))
    x2, _ := strconv.Atoi(fmt.Sprint(match[3]))
    y2, _ := strconv.Atoi(fmt.Sprint(match[4]))

    xx := (x2-x1)/2 + x1
    yy := (y2-y1)/2 + y1
    AdbShellInputTap(xx, yy)
    TimeSleepDuration(t)
    AdbShellInputKeyEvent("4") //back
    TimeSleepDuration(2)
}

8.6 模拟滑动手机页面 AdbShellInputSwipe

//模拟滑动
//adb shell input swipe  0 0  600 600
func AdbShellInputSwipe(x1, y1, x2, y2 int) {
    xx1 := strconv.Itoa(x1)
    yy1 := strconv.Itoa(y1)
    xx2 := strconv.Itoa(x2)
    yy2 := strconv.Itoa(y2)
    exec.Command("adb", "shell", "input", "swipe", xx1, yy1, xx2, yy2).Run()
}

注意:之前曾误以为手机页面对应的源码像电脑上的html5一样包括不可见部分。后来才发现每次获取的手机页面源码只包含可见部分。也就是说只要滑动手机屏幕就必须重新用AdbShellUiautomatorDump获取。

另外,关于判断设备是否休眠、查看手机上应用的packageName、查看最上层activity名字等功能,可参考我之前的文章https://www.cnblogs.com/pu369/p/10490668.html

 

posted on 2019-12-20 16:40  pu369com  阅读(2328)  评论(1编辑  收藏  举报

导航