Android 在Android代碼中執行命令行

来源:http://www.cnblogs.com/lipeineng/archive/2016/11/18/6078859.html
-Advertisement-
Play Games

1.路徑最好不要是自己拼寫的路徑/mnt/shell/emulated/0/wifidog.conf 最好是通過方法獲取的路徑,不然可能導致命令無效 (掛載點的原因) public static final String SDCARD_ROOT=Environment.getExternalStor ...


1.路徑最好不要是自己拼寫的路徑/mnt/shell/emulated/0/wifidog.conf

最好是通過方法獲取的路徑,不然可能導致命令無效  (掛載點的原因)

public static final String SDCARD_ROOT=Environment.getExternalStorageDirectory().getAbsolutePath(); 

public static final String AAA_PATH=SDCARD_ROOT+"/wifidog.conf";

 

Android 命令行執行工具類

package com.example.videotest.utils;

import android.os.Environment;
import android.util.Log;

import java.io.BufferedReader;
import java.io.DataOutputStream;
import java.io.InputStreamReader;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

import static java.lang.Runtime.getRuntime;

/**
 * 執行命令的類
 * Created by Kappa
 */
public class ExeCommand {
    //shell進程
    private Process process;
    //對應進程的3個流
    private BufferedReader successResult;
    private BufferedReader errorResult;
    private DataOutputStream os;
    //是否同步,true:run會一直阻塞至完成或超時。false:run會立刻返回
    private boolean bSynchronous;
    //表示shell進程是否還在運行
    private boolean bRunning = false;
    //同步鎖
    ReadWriteLock lock = new ReentrantReadWriteLock();

    //保存執行結果
    private StringBuffer result = new StringBuffer();

    /**
     * 構造函數
     *
     * @param synchronous true:同步,false:非同步
     */
    public ExeCommand(boolean synchronous) {
        bSynchronous = synchronous;
    }

    /**
     * 預設構造函數,預設是同步執行
     */
    public ExeCommand() {
        bSynchronous = true;
    }

    /**
     * 還沒開始執行,和已經執行完成 這兩種情況都返回false
     *
     * @return 是否正在執行
     */
    public boolean isRunning() {
        return bRunning;
    }

    /**
     * @return 返回執行結果
     */
    public String getResult() {
        Lock readLock = lock.readLock();
        readLock.lock();
        try {
            Log.i("auto", "getResult");
            return new String(result);
        } finally {
            readLock.unlock();
        }
    }

    /**
     * 執行命令
     *
     * @param command eg: cat /sdcard/test.txt 
     * 路徑最好不要是自己拼寫的路徑,最好是通過方法獲取的路徑 
     * example:Environment.getExternalStorageDirectory()
     * @param maxTime 最大等待時間 (ms)
     * @return this
     */
    public ExeCommand run(String command, final int maxTime) {
        Log.i("auto", "run command:" + command + ",maxtime:" + maxTime);
        if (command == null || command.length() == 0) {
            return this;
        }

        try {
            process = getRuntime().exec("sh");//看情況可能是su
        } catch (Exception e) {
            return this;
        }
        bRunning = true;
        successResult = new BufferedReader(new InputStreamReader(process.getInputStream()));
        errorResult = new BufferedReader(new InputStreamReader(process.getErrorStream()));
        os = new DataOutputStream(process.getOutputStream());

        try {
            //向sh寫入要執行的命令
            os.write(command.getBytes());
            os.writeBytes("\n");
            os.flush();

            os.writeBytes("exit\n");
            os.flush();

            os.close();
            //如果等待時間設置為非正,就不開啟超時關閉功能
            if (maxTime > 0) {
                //超時就關閉進程
                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        try {
                            Thread.sleep(maxTime);
                        } catch (Exception e) {
                        }
                        try {
                            int ret = process.exitValue();
                            Log.i("auto", "exitValue Stream over"+ret);
                        } catch (IllegalThreadStateException e) {
                            Log.i("auto", "take maxTime,forced to destroy process");
                            process.destroy();
                        } 
                    }
                }).start();
            }

            //開一個線程來處理input流
            final Thread t1 = new Thread(new Runnable() {
                @Override
                public void run() {
                    String line;
                    Lock writeLock = lock.writeLock();
                    try {
                        while ((line = successResult.readLine()) != null) {
                            line += "\n";
                            writeLock.lock();
                            result.append(line);
                            writeLock.unlock();
                        }
                    } catch (Exception e) {
                        Log.i("auto", "read InputStream exception:" + e.toString());
                    } finally {
                        try {
                            successResult.close();
                            Log.i("auto", "read InputStream over");
                        } catch (Exception e) {
                            Log.i("auto", "close InputStream exception:" + e.toString());
                        }
                    }
                }
            });
            t1.start();

            //開一個線程來處理error流
            final Thread t2 = new Thread(new Runnable() {
                @Override
                public void run() {
                    String line;
                    Lock writeLock = lock.writeLock();
                    try {
                        while ((line = errorResult.readLine()) != null) {
                            line += "\n";
                            writeLock.lock();
                            result.append(line);
                            writeLock.unlock();
                        }
                    } catch (Exception e) {
                        Log.i("auto", "read ErrorStream exception:" + e.toString());
                    } finally {
                        try {
                            errorResult.close();
                            Log.i("auto", "read ErrorStream over");
                        } catch (Exception e) {
                            Log.i("auto", "read ErrorStream exception:" + e.toString());
                        }
                    }
                }
            });
            t2.start();

            Thread t3 = new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        //等待執行完畢
                        t1.join();
                        t2.join();
                        process.waitFor();
                    } catch (Exception e) {

                    } finally {
                        bRunning = false;
                        Log.i("auto", "run command process end");
                    }
                }
            });
            t3.start();

            if (bSynchronous) {
                Log.i("auto", "run is go to end");
                t3.join();
                Log.i("auto", "run is end");
            }
        } catch (Exception e) {
            Log.i("auto", "run command process exception:" + e.toString());
        }
        return this;
    }

}

講解一下關鍵點,首先是啟動一個sh進程,當然如果你用的是root的設備,可以使用su。

這個進程包含 input、output、error 三個流,這三個流要處理好,否則可能不能正常結束進程,

另外也存在執行的命令已經結束,但是依然還有input流的情況,也需要處理。

其他請參考代碼

 

使用方式為2種。首先是阻塞方式,這種調用方式會一直阻塞至命令執行完成,返回命令行的輸出結果

public static final String SDCARD_ROOT=Environment.getExternalStorageDirectory().getAbsolutePath();                     
public static final String AAA_PATH=SDCARD_ROOT+"/wifidog.conf";


//讀取目標文件(絕對路徑)指定內容“#TrustedMACList ”的那一行
String cmd3="sed -n '/#TrustedMACList /,//p' "+AAA_PATH;
String str3 = new ExeCommand().run(cmd3, 10000).getResult();
Log.i("auto", str3+"button3");
Toast.makeText(MainActivity.this, str3,Toast.LENGTH_SHORT).show();

LOG   程式執行的順序

I/auto    ( 5542): run command:sed -n '/#TrustedMACList /,//p' /storage/emulated/0/wifidog.conf,maxtime:10000
I/auto    ( 5542): run is go to end
I/auto    ( 5542): read ErrorStream over
I/auto    ( 5542): read InputStream over
I/auto    ( 5542): run command process end
I/auto    ( 5542): run is end
I/auto    ( 5542): getResult
I/auto    ( 5542): #TrustedMACList 00:00:C0:1D:F0:0D,00:00:C0:1D:F0:0D,00:00:C0:1D:F0:0D,00:00:C0:1D:F0:0D,00:00:C0:1D:F0:0D,00:00:DE:AD:BE:AF,00:00:C0:1D:F0:0D
I/auto    ( 5542): 
I/auto    ( 5542): button3

 

 

 

還有一種是非同步方式,這種調用方式會直接返回,之後可以使用 getResult() 獲取結果,使用 isRunning() 來判斷是否完成,比如

 ExeCommand cmd = new ExeCommand(false).run("your cmd", 60000);
    while(cmd.isRunning())
    {
        try {
            sleep(1000);
        } catch (Exception e) {

        }
        String buf = cmd.getResult();
        //do something
    }

 

 

 

 

 

 

 

//修改目標文件指定內容“#TrustedMACList ”
String cmd="sed -i 's/#TrustedMACList /#TrustedMACList 00:00:C0:1D:F0:0D,/g' "+AAA_PATH; String str = new ExeCommand().run(cmd, 10000).getResult(); Log.i("auto", str+"button4"); Toast.makeText(MainActivity.this, str,Toast.LENGTH_SHORT).show();

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本例用來展示 QTimer 的使用,如何定時的更新一個視窗部件。 1 QLCDNumber 類 QLCDNumber 是一種可將數字顯示為類似 LCD 形式的視窗部件,它同 QLabel 一樣,都繼承自 QFrame,而 QFrame 繼承自 QWidget 頭文件 digitalclock.h D ...
  • 文章地址 文章地址 拓展閱讀: RabbitMQ + PHP (一)入門與安裝 RabbitMQ + PHP (二)AMQP拓展安裝 RabbitMQ + PHP (三)案例演示 RabbitMQ + PHP (一)入門與安裝 RabbitMQ + PHP (二)AMQP拓展安裝 RabbitMQ ...
  • 原文地址 以下是消息隊列以下的大綱,本文主要介紹消息隊列概述,消息隊列應用場景和消息中間件示例(電商,日誌系統)。 本次分享大綱 消息隊列概述 消息隊列應用場景 消息中間件示例 JMS消息服務(見第二篇:大型網站架構系列:分散式消息隊列(二)) 常用消息隊列(見第二篇:大型網站架構系列:分散式消息隊 ...
  • hello,各位小伙伴們,在很多小伙伴們剛剛開始學習android的時候,常常會有一些project裡面需要有一些基本動畫的插入,那麼具體是要怎麼實現呢?我們接下一起分析一下在android中的幾種基本動畫。 (這裡我們預設是用個人覺得比eclipse更智能一些的Android Studio來實現。 ...
  • 生命周期對程式員很重要,特別當我們瞭解,就可以寫出更流暢的程式,更好的來避規性能瓶頸,讓我們的APP擁有更好的用戶體驗。接下來我們來分享一下“返回棧”。 安卓的活動中重疊的,當我們打開一個新的活動時,會覆蓋上一個活動。然後點“back”時會銷毀最上面的活動,下麵的一個活動就會顯示出來。... ...
  • AndRodi Studio中的按鈕時件註冊一定要寫在onCraete中 ...
  • 相信很多同學都為調試蘋果的通知煩惱過,特別是通過通知啟動app這個功能,簡直讓人欲哭無淚!!! 然而我們都遇到的問題,蘋果怎麼可能沒有想到,原來早就有了官方的解決辦法,只是我們不知道而已。。。 這次又是從stackOverFlow上找到了答案,必須記錄一下!!! iOS10以後,通知框架被完全重構了 ...
  • 當一個程式第一次啟動的時候,Android會啟動一個LINUX進程和一個主線程。預設的情況下,所有該程式的組件都將在該進程和線程中運行。 同時,Android會為每個應用程式分配一個單獨的LINUX用戶。Android會儘量保留一個正在運行進程,只在記憶體資源出現不足時,Android會嘗試停止一些進 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...