• Android开发切换host应用


    Android开发切换host应用



    由于在工作过程中常需要切换手机的host来测试不同服务器上的接口,所以想到需要这么个软件。

    SwitchHost在PC上是一款很好用的修改Host的软件,手机上也需要这么一款App(当然手机需要已经ROOT),于是就尝试做这么个手机应用SwitchHost,主要的功能是要能在不同的Host方案之间进行切换。

    代码在git上:https://github.com/jianfengye/Android_Works/tree/master/SwitchHost

    里面的bin文件夹有apk,可以直接安装。

    需求

    它的UI设计大概是这样的(用的都是系统自带的控件)

    首页弹出浮层修改页设置项增加页

    分析这个功能:

    1 可以增加一个host方案

    2 可以修改一个host方案

    3 可以设置某个host方案为当前方案

    4 可以删除某个host方案

     

    这篇记录下开发这个的过程中遇到的问题:

    1 如何获取ROOT权限

    由于Android的host是在/system/etc/hosts,所以需要使用root来进行文件替换。(当然前提是你的机器是已经可以root了)

    获取ROOT权限使用这么一个句子:

    Process process = Runtime.getRuntime().exec("su");

    但是每次调用这个语句的时候都会出现"该应用已经获取了ROOT权限",所以这个process对象有必要使用单例模式来存储。

     

    还有一点,调用这个语句之后,其实是启动了一个process,并不是说当前执行的Activity已经获取了ROOT权限,所以说后续的root操作就需要使用命令行来执行,而不能使用程序。

    比如这里获得了root的进程之后,我原本想要使用File类来修改/system/etc/hosts发现还是出现权限错误的异常。后来改成了使用/system/xbin/cp命令才能使用。

    在Android中,系统的可执行文件不是在/usr/bin下的,而是在/system/xbin/下的,cp的命令和linux是一样的,加一个-f是强制执行。

     

    下面是一个完整的获取root权限,并执行cp -f source desc的例子:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    Process process = null;
    DataOutputStream os = null;
    try {
     String hostPath = context.getFilesDir() + "/" + hostName;
        String cmd="/system/xbin/cp -f " + hostPath + " " + "/etc/hosts";
        process = Runtime.getRuntime().exec("su"); //切换到root帐号
        os = new DataOutputStream(process.getOutputStream());
        os.writeBytes(cmd + " ");
        os.writeBytes("exit ");
        os.flush();
        process.waitFor();
    } catch (Exception e) {
        return false;
    } finally {
        try {
            if (os != null) {
                os.close();
            }
            process.destroy();
        } catch (Exception e) {
        }
    }

    2 多行显示的EditText框

    原生的EditText框原本在下方是有一个长长的输入线的,要想覆盖这个输入线,就需要设置背景来覆盖住它。

    然后原本的EditText框光标是放在中间的,要是希望光标能放在左上角,就需要设置gravity属性 top|left

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    <EditText
        android:id="@+id/hostContent"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#ffffff"
        android:ems="20"
        android:gravity="top|left"
        android:inputType="textMultiLine"
        android:lines="60"
        android:maxLines="60">

    3 提示框

    网上查了下,提示框是使用Builder类来做的,但是网上好多例子都是使用好几行的代码来做,实际上最简单的提示框使用一行就可以了

    1
    new Builder(this).setTitle("提示").setMessage("已经有相同的方案了").show();

    4 host方案怎么存储?

    我的设计是每个host方案都是一个文件,存储在这个app的默认存储位置。android应用的默认存储位置是/data/data/[包名]/files/。

    android也封装了很简单的api来操作默认存储位置的文件:

    获取文件列表:context.fileList()

    修改增加文件:context.openFileOutput()

    查看文件内容:context.openFileInput()

    删除文件:context.deleteFile()

    5 如何存储当前host是使用host方案

    这里有个需要记录的东西是当前host存储哪个host方案

    原本想的是可以使用一个文件来记录当前host方案的名字,但是这样就破坏了“默认存储位置只存host方案”的规则。所以就想到了将这个配置存储在SharePreference中

    SharePreference是Android用来设置配置项的地方,它不随着应用的关闭而关闭,也是一个持久存储。所以非常符合我的这个需求。

    1
    2
    SharedPreferences prefs = this.getSharedPreferences(SwitchHostActivity.CUR_SHARE_PREFERENCE, Context.MODE_PRIVATE);
    String curHostName = prefs.getString(SwitchHostActivity.CUR_HOSTNAME,SwitchHostActivity.DEFAULT_HOSTNAME);

    6 如何保证页面在主Activity的时候按下返回键就退出程序

    原本我使用每个Acitity的转换都使用Intent来进行切换,那么回退键会非常乱。

    于是我使用的方法是重写onBackPressed(), 并且从其他Activity回到主Activity的时候finish非主的Activty。

    我想应该还有其他的方法,比如onBackPressed()执行的方法是杀死当前进程?之类的,我还没查到。。。如果有人知道的话,麻烦告诉下。

    7 列表的ListView如何控制后面的勾选图片?

    不能直接用ArrayAdapter<String>,需要封装一个对象HostItem,HostItem中有个String和boolean的属性

    然后创建一个HostItemAdapter对象继承ArrayAdapter<HostItem>并重写getView方法

    public View getView(int position, View convertView, ViewGroup parent)

    所有的Item展示的逻辑控制就在这个getView方法内。

    最后再setAdapter绑定这个HostItemAdapter。

    8 如何弹出浮窗

    developer中有写很详细了

    http://developer.android.com/guide/topics/ui/menus.html#FloatingContextMenu

    但是在实际使用的时候我遇到一个问题

    如何在onContextItemSelected中获取到当前使得menu弹出的View?(由于这个是使用Item,所以是不能通过Id来直接获取的)

    实际上是这样获取的:

    1
    2
    3
    AdapterContextMenuInfo info = (AdapterContextMenuInfo) item.getMenuInfo();
    RelativeLayout hostView;
    hostView = (RelativeLayout) info.targetView;

    就是使用AdapterContextMenuInfo的targetView获取的。这个问题我在google中没有查到,是看手册试出来的。

     

    大致跨过了上面几个问题,这个App就可以做出来了。

    Creative Commons License

    本文基于署名-非商业性使用 3.0许可协议发布,欢迎转载,演绎,但是必须保留本文的署名叶剑峰(包含链接http://www.cnblogs.com/yjf512/),且不得用于商业目的。如您有任何疑问或者授权方面的协商,请与我联系

  • 相关阅读:
    获取文件方法名并执行
    mysql使用强制索引查数据
    itertools详解
    python 将数据复制到剪贴板
    [Warning] Changed limits: max_open_files: 1024
    原琴小工具
    WhiteAlbums2小感想
    用python的turtle作图(二)动画吃豆人
    通过例子进阶学习C++(八)进制转换
    我的第一个app之路
  • 原文地址:https://www.cnblogs.com/yangkai-cn/p/4016719.html
Copyright © 2020-2023  润新知