Android 6 与之前的一些变化

本文列出开发过程中遇到的一些Android6.0的变化

1。 Mac地址的获取

android 6.0之前,通过WifiManager-WifiInfo-getMacAddress()就可以获取到设备的mac地址

 public static String getWiFiMac(Context context)
    {
        WifiInfo wifiInfo = getWifiInfo(context);
        if (wifiInfo == null)
            return null;
        String wifiMac = wifiInfo.getMacAddress();     
        return wifiMac;
    }

    public static WifiInfo getWifiInfo(Context context)
    {
        WifiManager wifiManager = getWifiManager(context);
        if (wifiManager != null)
        {
            WifiInfo wifiInfo = wifiManager.getConnectionInfo();
            return wifiInfo;
        }
        else
            return null;
    }

    public static WifiManager getWifiManager(Context context)
    {
        WifiManager wifiManager = (WifiManager) context
                .getSystemService(Context.WIFI_SERVICE);
        return wifiManager;
    }

但是Android6.0通过该方法只能获得一个固定的字符串“02:00:00:00:00:00”
 
  解决方法:

 public static String getMac()
    {
        String str = "";
        String macSerial = "";
        try
        {
            Process pp = Runtime.getRuntime().exec("cat /sys/class/net/wlan0/address ");
            InputStreamReader ir = new InputStreamReader(pp.getInputStream());
            LineNumberReader input = new LineNumberReader(ir);

            for (; null != str;)
            {
                str = input.readLine();
                if (str != null)
                {
                    Log.d(tag, "cat sys/class/net/wlan0/address");
                    macSerial = str.trim();// 去空格  
                    break;
                }
            }
        }
        catch (Exception ex)
        {
            Log.e(tag, "error = " + ex.getMessage().toString());
        }

        if (macSerial == null || "".equals(macSerial))
        {
            try
            {
                Log.d(tag, "sys/class/net/eth0/address");
                return loadFileAsString("/sys/class/net/eth0/address").toUpperCase(
                    Locale.getDefault()).substring(0, 17);
            }
            catch (Exception e)
            {
                Log.e(tag, "error = " + e.getMessage().toString());
            }

        }
        return macSerial;
    }

 public static String loadFileAsString(String fileName) throws Exception
    {
        FileReader reader = new FileReader(fileName);
        String text = loadReaderAsString(reader);
        reader.close();
        return text;
    }

    public static String loadReaderAsString(Reader reader) throws Exception
    {
        StringBuilder builder = new StringBuilder();
        char[] buffer = new char[4096];
        int readLength = reader.read(buffer);
        while (readLength >= 0)
        {
            builder.append(buffer, 0, readLength);
            readLength = reader.read(buffer);
        }
        return builder.toString();
    }

2。 动态权限的申请

Android 6.0之前,开发者只需要在AndroidManifesxt.xml中配置自己想要的权限就可以,当你安装应用程序的时候,会有一个页面提示你当前的应用需要你哪些权限。当前手机的ROM一般的都有管理应用程序权限的设置。
  Android6 Google推出了动态权限的获取,如下危险的权限需要动态的症的用户的同意,比如:身体传感器、日历、摄像头、通讯录、地理位置、麦克风、电话、短信和存储空间。 Android6.0默认的为targetSdkVersion<23的应用程序默认授权了所申请的所有权限,所以如果你以前 APP设置的targetSdkVersion<23,在运行时也不会崩溃,但是这只是一个临时的救急策略。

下面以申请SdCard的写权限为例说明:

    private void requestPermission()
    {
        if (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED)
        {
            if (shouldShowRequestPermissionRationale(Manifest.permission.WRITE_EXTERNAL_STORAGE))
            { //如果你上次拒绝了该权限
                Toast.makeText(this, "必须具有sdcard权限", Toast.LENGTH_LONG).show();
                requestPermissions(
                    new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                    EXTERNAL_STORAGE_REQ_CODE);
            }
            else
            {
                requestPermissions(
                    new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                    EXTERNAL_STORAGE_REQ_CODE);
            }
        }
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, String permissions[],
            int[] grantResults)
    {
        if (grantResults.length <= 0)
            return;

        if (grantResults[0] == PackageManager.PERMISSION_DENIED)
        {//用户拒绝了该权限,需要进行界面逻辑处理。
        }
        else if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
        {
            switch (requestCode)
            {
                case EXTERNAL_STORAGE_REQ_CODE :
                {
                    createFile("hello.txt");
                }
            }
        }

    }

3。 悬浮窗的处理

Android 6.0之(这时候你的targetSdkVersion<23),创建悬浮窗是需要申请权限:android.permission.SYSTEM_ALERT_WINDOW。 让后通过WindowManager就可以创建并显示一个悬浮窗(当前你必须授权该权限,MIUI这种Rom默认的是关闭的,需要在设置中打开)。
  在Android6.0之后,如果你修改了targetSdkVersion为23(这个只能增大,不能变小回退)。 你再次通过WindowManager来创建悬浮窗,你的APP就直接崩溃了。
  解决办法: 如果你将targetSdkVersion设置为23或者更高,在使用SYSTEM_ALERT_WINDOW权限是,需要先调用Setting.canDrawOverlays()来判断是否允许创建悬浮窗。允许直接创建;如果不允许需要发送一个action值为ACTION_MANAGER_OVERLAY_PERMISSION的Intent来让用户同意创建悬浮窗。 具体你代码如下所示:

if(Build.VERSION.SDK_INT >=23) {
    if(Seetings.canDrawOverlays(context) {
        //显示你的的悬浮窗
    }
    else {
        Intent inent = new Intent(Setting.ACTION_MANAGER_OVERLAT_PERMISSION);
        startActivity(intent);
    }
}

“通过将WindowManager.LayoutParams的type设置为TYPE_TOAST”,是否还管用?
  之前有人通过逆向的方法得出结论:将WindowManager.LayoutParams的type设置为TYPE_TOAST,可以不申请SYSTEM_ALERT_WINDOW权限就可以显示悬浮窗,但是这种方法需要处兼容问题,比如:在MIUI下还是需要权限(同理其他ROM可能也会面临同样的问题),并且需要API level>=19(Android4.4)(老版本不响应触摸事件)。 具体的结论如下所示:

在4.0.1以前, 当我们使用TYPE_TOAST, Android会偷偷给我们加上FLAG_NOT_FOCUSABLE和FLAG_NOT_TOUCHABLE, 4.0.1开始, 会额外再去掉FLAG_WATCH_OUTSIDE_TOUCH, 这样真的是什么事件都没了. 而4.4开始, TYPE_TOAST被移除了, 所以从4.4开始, 使用TYPE_TOAST的同时还可以接收触摸事件和按键事件了, 而4.4以前只能显示出来, 不能交互。  
API level 18及以下使用TYPE_TOAST无法接收触摸事件的原因也找到了.
文/Shawon(作者)
原文链接:http://www.jianshu.com/p/634cd056b90c

那么这种方法在Androi6.0中(排除MIUI这种ROM),还可以吗?
  我们拿一个Android 6.0的手机(天机AXON),经过测试(demo):是可以的
  当然,最可靠的方案还是:read the fucking sourecode !!!

4。 Apache HttpClient的移除

早在Android2.3 Android就建议使用HttpURLConnection来进行网络开发。而且Google退出的一些开源都是这么做的,如:Volley等。

5。 WiFi和网络变化

1. 你的app智能修改自己的创建的WifiConfiguration对象的状态,不能修改其他App创建的WifiConfiguration对象。
  2. Android 6.0之前,你可以通过enableNetwork(),设置disableAllOthers = true
,来是的设备软开其他网络,如蜂窝网络,而强制连接指定的Wifi网络。此版本上设备将不会从其他网络断开。

你可能感兴趣的:(Android 6 与之前的一些变化)