Android自动化性能收集

<h1>Android自动化性能收集</h1>

<p>Android 功能测试自动化框架较多,UIAutomator,Robotium,Appium等。Case执行过程中,可能希望收集手机的性能指标,包括内存、cpu、流量等。使用java+shell+bat简单实现了android手机性能收集。</p>

<h2>简述</h2>

<p>过程很简单:</p>

<ul> <li><p>在启动自动化case前,执行收集信息的命令。<br/> 其实就是一些adb shell命令,如下:</p>

<pre><code> adb shell top -n 1| awk '{print $3" "$10}' &gt;&gt; cpu.dat adb shell ps | awk '{print $5" "$9}' &gt;&gt; mem.dat //android sdk level 大于16 adb shell cat /proc/uid_stat/$uid/tcp_rcv &gt;&gt; $uid"_recv.dat" adb shell cat /proc/uid_stat/$uid/tcp_snd &gt;&gt; $uid"_snd.dat" //android sdk level 小于16 adb shell cat /proc/$pid/net/dev | grep wlan | awk '{print $2" "$10}' &gt;&gt; $pid"_net.dat" </code></pre></li>

<li><p>执行自动化case,可能需要很久。这个过程中,上面的命令在不断地执行</p></li> <li>case执行结束,kill掉第一步中的命令</li> <li>分析收集到的dat文件,使用jscharts绘出走势图</li> </ul>

<h2>脚本</h2>

<h3>cpu</h3>

<p>使用top命令不断查看各进程的cpu占用</p>

<h4>linux</h4>

<p>get-android-cpu.sh</p>

<pre><code> #!/bin/sh #path--target/android-info/mem/dat cd ../../../ mkdir -p target/android-info/cpu/dat mkdir -p target/android-info/cpu/html cp src/main/resources/jscharts.js target/android-info/cpu/html cd target/android-info/cpu/dat while true do adb shell top -n 1| awk '{print $3" "$10}' &gt;&gt; cpu.dat sleep 15 done </code></pre>

<h4>windows</h4>

<p>get-android-cpu.bat</p>

<pre><code> cd ..\..\jenkins\workspace\android-info-end3 mkdir target\android-info\cpu\dat mkdir target\android-info\cpu\html copy src\main\resources\jscharts.js target\android-info\cpu\html cd target\android-info\cpu\dat :run adb shell top -n 1 | awk "{print $3\" \"$10}" &gt;&gt; cpu.dat ping 127.0.0.1 -n 15 &gt; null goto run </code></pre>

<h3>memory</h3>

<p>使用ps命令不断查看各进程的内存占用</p>

<h4>linux</h4>

<p>get-android-mem.sh</p>

<pre><code> #!/bin/sh #path--target/android-info/mem/dat cd ../../../ mkdir -p target/android-info/mem/dat mkdir -p target/android-info/mem/html cp src/main/resources/jscharts.js target/android-info/mem/html cd target/android-info/mem/dat while true do adb shell ps | awk '{print $5" "$9}' &gt;&gt; mem.dat sleep 15 done </code></pre>

<h4>windows</h4>

<pre><code> cd ..\..\jenkins\workspace\android-info-end3 mkdir target\android-info\mem\dat mkdir target\android-info\mem\html copy src\main\resources\jscharts.js target\android-info\mem\html cd target\android-info\mem\dat :run adb shell ps | awk "{print $5\" \"$9}" &gt;&gt; mem.dat ping 127.0.0.1 -n 15 &gt; null goto run </code></pre>

<h3>流量</h3>

<p>android 4.0以上版本可以用<code>/proc/uid_stat/$uid/tcp_rcv</code>和<code>/proc/uid_stat/$uid/tcp_snd</code>来获取某个程序的上下行流量;而4.0以下版本要用<code>cat /proc/$pid/net/dev</code>来查看上下行流量。uid和pid的关系,可以从<code>/data/system/packages.list</code>这个文件中获取。</p>

<h4>linux</h4>

<p>判断android api level:</p>

<pre><code> #!/bin/sh #android 4.0以上和4.0以下方法不同 #get android sdk level apileveltemp=`adb shell getprop | grep ro.build.version.sdk` apilevel=${apileveltemp:25:2} chmod +x *.sh echo "android api level:"$apilevel if [ $apilevel -gt 14 ] then ./get-android-net-gt-4.0.sh elif [ $apilevel -lt 14 ] then ./get-android-net-lt-4.0.sh fi </code></pre>

<p>get-android-net-gt-4.0.sh</p>

<pre><code> #!/bin/sh #android api level great than 14(android 4.0) #path--target/android-info/net/dat cd ../../../ mkdir -p target/android-info/net/dat mkdir -p target/android-info/net/html cp src/main/resources/jscharts.js target/android-info/net/html cd target/android-info/net/dat cd .. echo "adb pull/data/system/packages.list--start" adb pull /data/system/packages.list cd dat while true do echo "get net info from /proc/uid-stat/$uid" for i in `adb shell ls /proc/uid_stat` do #delete the Enter character uid=`echo $i | tr -d ["\r\n"]` adb shell cat /proc/uid_stat/$uid/tcp_rcv &gt;&gt; $uid"_recv.dat" adb shell cat /proc/uid_stat/$uid/tcp_snd &gt;&gt; $uid"_snd.dat" done sleep 15 done </code></pre>

<p>get-android-net-lt-4.0.sh</p>

<pre><code> #!/bin/sh #path--target/android-info/net/dat cd ../../../ mkdir -p target/android-info/net/dat mkdir -p target/android-info/net/html cp src/main/resources/jscharts.js target/android-info/net/html cd target/android-info/net/dat #get /proc/$pid/net/dev while true do #get pid for i in `adb shell ps | awk '{print $2}'` do pid=`echo $i | tr -d ["\r\n"]` echo $pid adb shell cat /proc/$pid/net/dev | grep wlan | awk '{print $2" "$10}' &gt;&gt; $pid"_net.dat" done sleep 15 done </code></pre>

<h4>windows</h4>

<p>bat命令不熟,不知道如何在windows下实现linux下地反转义,这里用java代码实现。</p>

<pre><code> import java.io.*; /** * Created by Xuemeng Wang on 14-9-15. * api Level &gt; 16 */ public class GetNetInfo { public static void main(String[] args) { String uidString = execCmd("adb shell ls /proc/uid_stat"); String[] uidArray = uidString.split("\n"); int length = uidArray.length-1; for(int i=0;i&lt;=length-1;i++) { String contentRcv = execCmd("adb shell cat /proc/uid_stat/"+uidArray[i]+"/tcp_rcv"); String contentSnd = execCmd("adb shell cat /proc/uid_stat/"+uidArray[i]+"/tcp_snd"); System.out.println(System.getProperty("user.dir")); try { File file = new File(uidArray[i]+"_recv.dat"); if(!file.exists()) file.createNewFile(); File file2 = new File(uidArray[i]+"_snd.dat"); if(!file2.exists()) file2.createNewFile(); FileWriter fileWriter1 = new FileWriter(file, true); FileWriter fileWriter2 = new FileWriter(file2, true); fileWriter1.write(contentRcv); fileWriter2.write(contentSnd); fileWriter1.close(); fileWriter2.close(); } catch (IOException e) { e.printStackTrace(); } } } public static String execCmd(String command) { BufferedReader br = null; StringBuffer stringBuffer = new StringBuffer(); try { Process p = Runtime.getRuntime().exec(command); br = new BufferedReader(new InputStreamReader(p.getInputStream())); String line = null; while ((line = br.readLine()) != null) { if("".equals(line.trim())) continue; stringBuffer.append(line+"\n"); } } catch (Exception e) { e.printStackTrace(); } finally { if (br != null) { try { br.close(); } catch (Exception e) { e.printStackTrace(); } } } return stringBuffer.toString(); } } </code></pre>

<p>api level小于16的情况,我没用到,略掉:)。</p>

<h3>杀掉进程</h3>

<p>linux: <code>ps -ef | grep get-android- | grep -v grep | awk '{print $2}' | xargs kill -9</code><br/> windows(后台运行,其实是cmd进程,注意下面会杀掉所有的cmd进程):</p>

<pre><code> taskkill /F /IM cmd.exe taskkill /F /IM adb.exe </code></pre>

<h2>结果收集</h2>

<p>使用java分析dat,给出分析cpu的代码,工程已上传到github,<a href="https://github.com/yeetrack/android-performance">https://github.com/yeetrack/android-performance</a>。</p>

<p>CpuInfo.java</p>

<pre><code> package com.meilishuo.android.performance; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.testng.annotations.Test; import java.io.*; import java.util.ArrayList; import java.util.Iterator; import java.util.List; import java.util.TreeMap; /** * Created by victor on 14-9-9. */ public class CpuInfo { private static final Logger logger = LoggerFactory.getLogger(MemInfo.class); private static final String JSCHARTPATH = "target/android-info/cpu/html/"; private static final String MEMFILEPATH = "target/android-info/cpu/dat/cpu.dat"; TreeMap&lt;String, List&lt;String&gt;&gt; cpuMap = new TreeMap&lt;String, List&lt;String&gt;&gt;(); /** * 解析cpu.dat存储到map中 */ public void parseCpuFile() { File file = new File(MEMFILEPATH); if(null==file) return; try { BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(new FileInputStream(file))); String line = null; while((line=bufferedReader.readLine())!=null) { String[] array = line.trim().split(" "); //空格分隔 if(null==array || array.length!=2 || !array[0].contains("%") || "".equals(array[1]) || "Name".equals(array[1])) continue; if(cpuMap.size()==0 || !cpuMap.containsKey(array[1])) { List&lt;String&gt; memList = new ArrayList&lt;String&gt;(); memList.add(array[0].substring(0, array[0].indexOf("%"))); cpuMap.put(array[1], memList); } else { cpuMap.get(array[1]).add(array[0].substring(0, array[0].indexOf("%"))); } } System.out.println(cpuMap.size()); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } /** * 将map中的数据写到到xml中 */ public void writeXmlFromMap() { if(null==cpuMap || cpuMap.size()==0) return; Iterator&lt;String&gt; it = cpuMap.keySet().iterator(); while(it.hasNext()) { String key = it.next(); List&lt;String&gt; value = cpuMap.get(key); if(null==value || value.size()&lt;=1) continue; //写入xml File file = new File(JSCHARTPATH+key.replace("/", "_").replace(":", "_")+"_cpu.xml"); try { BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file, true))); bufferedWriter.write( "&lt;?xml version=\"1.0\"?&gt;\n" + "&lt;JSChart&gt;\n" + "\t&lt;dataset type=\"line\"&gt;"); int pos = 1; for(String index : value) { bufferedWriter.write("&lt;data unit=\""+pos+"\" value=\""+index+"\"/&gt;\n"); pos++; } bufferedWriter.write( "&lt;/dataset&gt;\n" + "\t&lt;optionset&gt;\n" + "\t\t&lt;option set=\"setLineColor\" value=\"'#8D9386'\"/&gt;\n" + "\t\t&lt;option set=\"setLineWidth\" value=\"4\"/&gt;\n" + "\t\t&lt;option set=\"setTitleColor\" value=\"'#7D7D7D'\"/&gt;\n" + "\t\t&lt;option set=\"setAxisColor\" value=\"'#9F0505'\"/&gt;\n" + "\t\t&lt;option set=\"setGridColor\" value=\"'#a4a4a4'\"/&gt;\n" + "\t\t&lt;option set=\"setAxisValuesColor\" value=\"'#333639'\"/&gt;\n" + "\t\t&lt;option set=\"setAxisNameColor\" value=\"'#333639'\"/&gt;\n" + "\t\t&lt;option set=\"setTextPaddingLeft\" value=\"0\"/&gt;\n" + "\t&lt;/optionset&gt;\n " + "&lt;/JSChart&gt;"); bufferedWriter.flush(); bufferedWriter.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } /** * 生成jschart html文件 */ public void writeJsHtml() { File file = new File(JSCHARTPATH); File[] xmlFiles = file.listFiles(new FileFilter() { @Override public boolean accept(File pathname) { return pathname.getName().endsWith("_cpu.xml"); } }); if(null== xmlFiles || xmlFiles.length==0) return; for(File index : xmlFiles) { File htmlFile = new File(index.getAbsolutePath().replace(".xml", ".html")); try { BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(htmlFile)); </code></pre>

继续阅读->

转载于:https://my.oschina.net/u/147181/blog/314001

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
不多说废话,看题目,本教程适合练手,会python+android基础的人群,文件较大,上传乃是下载链接,下面上目录: 1-1 课程导学 2-1 如何学好Android App性能测试? 2-10 详解【电量】监控值的获取方法 \' D, l" p) d6 d. K9 [7 p 2-11 详解【电量】监控脚本实现和数据分析 O, e4 X& K0 S% h8 v8 V9 ? 2-12 详解【内存】监控值的获取方法 k! e6 e# C" K% z9 k- l 2-13 详解【内存】监控脚本实现和数据分析0 d; e- S% G6 r3 H: g 2-14 详解【FPS&过度渲染】的概念和监控方法 - 分析页面卡慢的方法# G! _2 O9 T* j" K s3 v6 C0 l 2-2 工欲善其事必先利其器-性能测试环境准备 2-3 详解【启动时间】监控值的获取方法0 n( p* l; g C 2-4 详解【启动时间】监控脚本实现% B2 z( C( E& S: n r1 e 2-5 详解【启动时间】数据分析 2-6 【启动时间】时间戳差值监控方法概要介绍 2-7 详解【CPU】监控值的获取方法、脚本实现和数据分析 2-8 详解【流量】监控值的获取方法7 r7 ~/ D5 |+ h9 m9 i6 p) b: Y 2-9 详解【流量】监控脚本实现和数据分析2 [9 {# {$ c9 k/ T, `/ t" \ 3-1 为什么需要使用框架实现自动化测试? 3-10 UnitTest框架之TestCase,TestSuite,TestRunner简介3 A2 {1 F2 @; K 3-11 UnitTest框架之TestSuite,TestRunner自动化测试 3-12 数据驱动框架DDT简介 3-13 数据驱动框架DDT的使用方法 3-14 数据驱动框架DDT实战; 3-2 准备一个被测APP 3-3 工欲善其事必先利其器-自动化测试环境准备 3-4 Android App自动化测试(一) 3-5 Android App自动化测试(二) 3-6 Android App自动化测试(三) 3-7 UnitTest框架之TestFixture简介 3-8 UnitTest框架之TestFixture自动化测试(一) 3-9 UnitTest框架之TestFixture自动化测试(二) 4-1 如何学好Android App API接口测试? 4-10 Fiddler构造HTTP Get请求 4-11 Fiddler构造HTTP Post请求 4-12 Fiddler抓取手机上的网络数据包 4-13 为什么使用PostMan做API接口测试 4-14 工欲善其事必先利其器-PostMan工具准备 4-15 PostMan测试HTTP Get请求 4-16 PostMan测试HTTP Post请求 4-17 数据驱动DDT实现API接口自动化测试简介) 4-18 Python requests测试HTTP中的Get、Post请求 4-19 数据驱动DDT实现API接口自动化测试(一) 4-2 什么是API 4-20 数据驱动DDT实现API接口自动化测试(二); 4-3 抓包神器Fiddler简介 4-4 Fiddler抓包原理解析 4-5 Fiddler修改客户端发出的请求(一) 4-6 Fiddler修改客户端发出的请求(二) 4-7 Fiddler修改服务器端返回的内容 4-8 Fiddler实现会话的过滤、对比及请求的编解码 4-9 Fiddler实现Host的配置 5-1 测试工程师为什么需要掌握持续集成? 5-2 持续集成的概念、流程和意义 5-3 讲解持续集成工具Jenkins3 5-4 Jenkins工具密码的修改 5-5 Jenkins工具的配置说明 5-6 Jenkins工具系统配置和Job配置 5-7 Jenkins工具手动持续集成实战 5-8 Jenkins工具自动化持续集成实战 6-1 Native App自动化测试及Appuim框架介绍 6-2 自动化测试环境、元素识别工具、脚本设计原则-LOVE原则的讲解 6-3 Native App自动化脚本的实现 6-4 Appium自动化测试框架API讲解与案例实践(一) 6-5 Appium自动化测试框架API讲解与案例实践(二) 6-6 Appium自动化测试框架API讲解与案例实践(三) 6-7 Appium自动化测试框架API讲解与案例实践(四) 6-8 Appium自动化测试框架API讲解与案例实践(五) 6-9 Appium自动化测试框架API讲解与案例实践(六) 7-1 Hybrid App自动化测试概要 7-2 Appium基于Sele
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值