Android中应用调用系统权限

时间:2022-04-24
本文章向大家介绍Android中应用调用系统权限,主要内容包括Android6.0蓝牙开发中获取附近低功耗蓝牙设备结果权限问题分析、Android中应用调用系统权限、chapter one 系统应用添加系统权限、chapter two 三方应用添加系统权限、chapter three DEMO实现、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。

现在设备的安全性越来越受到重视,随之而来的便是开发中的各种不便,比如有普通权限,运行时权限,系统权限之分。对于运行时权限的添加可以参考

对于Android中各个权限的含义可以参考android/frameworks/base/core/res/AndroidManifest.xml中的释义

Android6.0蓝牙开发中获取附近低功耗蓝牙设备结果权限问题分析

而对于系统权限,却没有很好的解决方案,暂时只有一些特定的解决方案。

转载请注明出处,本文出自

海天之蓝

的博客

Android中应用调用系统权限

chapter one 系统应用添加系统权限

对于可以编译到源码里的apk添加系统权限很简单,就两步

一,在androidmanifest.xml文件中添加uid,即让应用跑在system进程中:android:sharduserid= “”

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.fang.zrf.networkdemo"
    android:sharedUserId="android.uid.system">

二,在该应用的mk文件中添加签名,即让该应用使用系统签名

LOCAL_CERTIFICATE := platform

其中platform对应的签名文件的位置为android/build/target/product/security  platform.pk8和platform.x509.pem

然后使用编译命令编译apk即可。

但如果没有mk的应用该怎么办呢?

其实也可以总结出来,让一个应用使用系统权限不外乎两步

第一就是uid改为system第二就是使用系统的签名,明白了这些之后我们就可以对应用使用系统权限了

chapter two 三方应用添加系统权限

第一步同上,添加Android:shareduserid属性

第二步,将打包好的apk使用系统签名重新签名打包。签名打包使用到一个Android源码中自带的一个工具叫做signapk,所在目录为Android/out/host/linux-x86/framework

使用cmd命令进行重新签名打包java -jar signapk.jar platform.x509.pem platform.pk8 old.apk new.apk

其中old.apk是打包好的apk,而new.apk是重新签名的apk。然后可以使用adb install -r new.apk将应用安装到手机上,此时即可获取系统权限。但有一个条件,既然用的是系统的签名,那首先就要保证你应用的系统签名和手机的系统签名一致,这个可能各大厂商会有所改动,所以不太能兼容。

chapter three DEMO实现

好了,说了这么多,来个demo看一看吧。demo所实现的功能有三个,

一是开启手机移动网络

二是关闭手机移动网络

三是获取手机移动网络

其实总结起来就是对手机移动网络的set和get两个功能而已。源码上对于set和get的方法属于hide的,不供三方应用使用,所以我们可以利用反射来调用到。

首先,先来看看需要什么权限

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
    <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE"/>
    <uses-permission android:name="android.permission.MODIFY_PHONE_STATE"/>

其中MODIFY_PHONE_STATE为系统权限,所以我们只能是按chapter two所示的做法来做。先把代码完成了生成了apk再重新签名。

紧接着就是设置手机移动网络的 状态

public static void setMobileDataState(Context context, boolean enable){
        Class[] argsClass = new Class[1];
        argsClass[0] = boolean.class;//setmobiledataenable方法中要传入一个boolean类型的参数
        ConnectivityManager connMr = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
        try {
            Method method = connMr.getClass().getMethod("setMobileDataEnabled",argsClass);

            method.invoke(connMr,enable);
        } catch (Exception e) {
            e.printStackTrace();
            Toast.makeText(context,"set failure + e = " + e.getCause(),Toast.LENGTH_LONG).show();
        }
    }

然后是获取手机移动网络的状态

public static boolean getMobileDataState(Context context){

        ConnectivityManager connMr = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);

        Class[] argsclass = null;//get时不需要传入参数,所以直接传入一个null的数组即可
        try {
            Method method = connMr.getClass().getDeclaredMethod("getMobileDataEnabled",argsclass);
            return (boolean)method.invoke(connMr);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return false;
    }

接下来就是界面设计的方法的调用了,

先来看看界面实现效果

public class MainActivity extends Activity implements View.OnClickListener{


    private TextView mOpenNet;
    private TextView mCloseNet;
    private TextView mGetNetState;
    private TextView mNetState;

    private boolean isOpenState;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        initView();
        initData();
    }


    /**
     * 初始化view
     */
    private void initView(){
        mOpenNet = (TextView) findViewById(R.id.open_network);
        mCloseNet = (TextView) findViewById(R.id.close_network);
        mGetNetState = (TextView) findViewById(R.id.get_state);
        mNetState = (TextView) findViewById(R.id.network_state);
        setClickListener(mOpenNet,mCloseNet,mGetNetState);
    }


    /**
     * 初始化数据
     */
    private void initData(){
        updateView();

    }

    /**
     * 设置view的点击事件
     * @param views
     */
    private void setClickListener(View...views){

        for (View view:views) {
            if (view != null){
                view.setOnClickListener(this);
            }
        }

    }
    @Override
    public void onClick(View v) {
        switch (v.getId()){
            case R.id.open_network:
                NetManager.setMobileDataState(getApplicationContext(),true);
                updateView();
                break;
            case R.id.close_network:
                NetManager.setMobileDataState(getApplicationContext(),false);
                updateView();
                break;
            case R.id.get_state:
                updateView();
                break;
            default:
                break;
        }
    }

    @Override
    protected void onResume() {
        super.onResume();
        updateView();
    }

    private void updateView(){
        isOpenState = NetManager.getMobileDataState(getApplicationContext());
        String str = isOpenState ? "true" : "false";
        if (str.equals(mNetState.getText().toString())){
            return;
        }
        mNetState.setText(str);
        setTvEnable(!isOpenState,isOpenState);
    }

    private void setTvEnable(boolean setEnable,boolean closeEnable){
        mOpenNet.setEnabled(setEnable);
        mCloseNet.setEnabled(closeEnable);
    }
}

代码完成后,在outputs目录下找到apk文件使用cmd命令进行重新签名。

到此,结束