Skip to content

Android 问题笔记【持续更新】

/dev/kvm 设备权限被拒绝的问题

在使用 AndroidStudio 时出现 /dev/kvm 设备权限被拒绝的问题,解决方案如下:

bash
sudo chown wolfx /dev/kvm
sudo chown wolfx /dev/kvm

wolfx 是我当前的登录名

Android Studio 报错显示 mips64el-linux-android-strip 找不到

如果 ndk 版本在 r17 版, 很有可能出现这个问题. 我的解决方案是:

Android 默认签名与签名的 SHA-1

在使用第三方服务时,经常会用到 KeyStore SHA-1 这里记录下获取方式。

默认 KeyStore 存储路径

Mac/Linux 系统中,debug.keystore 文件默认储存在 ~/.android/ 路径下。

Windows 系统中,debug.keystore 文件将默认存储在 C:\Users\{USERNAME}\.android\ 路径下。

获取默认 KeyStore SHA-1

输入下面命令查看指纹信息:

bash
keytool -list -v -keystore ~/.android/debug.keystore -alias androiddebugkey -storepass android -keypass android
keytool -list -v -keystore ~/.android/debug.keystore -alias androiddebugkey -storepass android -keypass android

新版 android 兼容处理

文件路径问题

通过 Uri.fromFile(file); 获取的 file:// 开头的路径,在新 android 上无权限使用。

解决方案是用 FileProvider 获得 content:// 开头的路径。

首先在 manifest 中配置:

xml
<application>
  <provider
      android:name="android.support.v4.content.FileProvider"
      android:authorities="com.junl.fileprovider"
      android:exported="false"
      android:grantUriPermissions="true">
      <meta-data
          android:name="android.support.FILE_PROVIDER_PATHS"
          android:resource="@xml/file_paths" />
  </provider>
</application>
<application>
  <provider
      android:name="android.support.v4.content.FileProvider"
      android:authorities="com.junl.fileprovider"
      android:exported="false"
      android:grantUriPermissions="true">
      <meta-data
          android:name="android.support.FILE_PROVIDER_PATHS"
          android:resource="@xml/file_paths" />
  </provider>
</application>

res/xml/file_paths.xml 中添加

xml
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="."/>
</paths>
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="."/>
</paths>

编写 Java 代码如下:

java
package com.junl.cordova;

import android.content.Context;
import android.net.Uri;
import android.os.Build;
import android.support.v4.content.FileProvider;

import java.io.File;

public class Utils {
    public static Uri getUriFromFile(Context ctx, File file) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            // 兼容 7.0
            return FileProvider.getUriForFile(ctx, "com.junl.fileprovider", file);
        } else {
            return Uri.fromFile(file);
        }
    }
}
package com.junl.cordova;

import android.content.Context;
import android.net.Uri;
import android.os.Build;
import android.support.v4.content.FileProvider;

import java.io.File;

public class Utils {
    public static Uri getUriFromFile(Context ctx, File file) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            // 兼容 7.0
            return FileProvider.getUriForFile(ctx, "com.junl.fileprovider", file);
        } else {
            return Uri.fromFile(file);
        }
    }
}

权限问题

新 android 的权限不仅仅要在 manifest 中申明,也需要在代码中授权:

写了个工具类如下:

java
package com.junl.cordova;

import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Build;

import org.apache.cordova.CordovaActivity;

public class PermissionUtil {
    private static final int PERMISSIONS_REQUEST = 1;
    private static final int PERMISSION_GRANTED = PackageManager.PERMISSION_GRANTED;
    private static final String[] Permission = new String[]{
            Manifest.permission.ACCESS_COARSE_LOCATION, // 用于进行网络定位
            Manifest.permission.ACCESS_FINE_LOCATION, // 用于访问GPS定位
            Manifest.permission.ACCESS_LOCATION_EXTRA_COMMANDS, // 用于访问A-GPS定位
            Manifest.permission.ACCESS_NETWORK_STATE, // 用于获取运营商信息,用于支持提供运营商信息相关的接口
            Manifest.permission.ACCESS_WIFI_STATE, // 用于访问wifi网络信息,wifi信息会用于进行网络定位
            Manifest.permission.CHANGE_WIFI_STATE, // 用于获取wifi的获取权限,wifi信息会用来进行网络定位
            Manifest.permission.INTERNET, // 用于访问网络,网络定位需要上网
            Manifest.permission.READ_PHONE_STATE, // 用于读取手机当前的状态
            Manifest.permission.WRITE_EXTERNAL_STORAGE, // 用于写入缓存数据到扩展存储卡
            Manifest.permission.READ_EXTERNAL_STORAGE,
            Manifest.permission.BLUETOOTH,
            Manifest.permission.BLUETOOTH_ADMIN,
            Manifest.permission.CAMERA,
    };

    /* 判断程序是否有所需权限 android22 以上需要自申请权限 */
    public static boolean hasPermission(CordovaActivity cordovaActivity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            for (int i = 0; i < Permission.length; i++) {
                if (cordovaActivity.checkSelfPermission(Permission[i]) != PERMISSION_GRANTED) {
                    return false;
                }
            }
            return true;
        } else {
            return true;
        }
    }

    /* 请求程序所需权限 */
    public static void requestPermission(CordovaActivity cordovaActivity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            cordovaActivity.requestPermissions(Permission, PERMISSIONS_REQUEST);
        }
    }
}
package com.junl.cordova;

import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Build;

import org.apache.cordova.CordovaActivity;

public class PermissionUtil {
    private static final int PERMISSIONS_REQUEST = 1;
    private static final int PERMISSION_GRANTED = PackageManager.PERMISSION_GRANTED;
    private static final String[] Permission = new String[]{
            Manifest.permission.ACCESS_COARSE_LOCATION, // 用于进行网络定位
            Manifest.permission.ACCESS_FINE_LOCATION, // 用于访问GPS定位
            Manifest.permission.ACCESS_LOCATION_EXTRA_COMMANDS, // 用于访问A-GPS定位
            Manifest.permission.ACCESS_NETWORK_STATE, // 用于获取运营商信息,用于支持提供运营商信息相关的接口
            Manifest.permission.ACCESS_WIFI_STATE, // 用于访问wifi网络信息,wifi信息会用于进行网络定位
            Manifest.permission.CHANGE_WIFI_STATE, // 用于获取wifi的获取权限,wifi信息会用来进行网络定位
            Manifest.permission.INTERNET, // 用于访问网络,网络定位需要上网
            Manifest.permission.READ_PHONE_STATE, // 用于读取手机当前的状态
            Manifest.permission.WRITE_EXTERNAL_STORAGE, // 用于写入缓存数据到扩展存储卡
            Manifest.permission.READ_EXTERNAL_STORAGE,
            Manifest.permission.BLUETOOTH,
            Manifest.permission.BLUETOOTH_ADMIN,
            Manifest.permission.CAMERA,
    };

    /* 判断程序是否有所需权限 android22 以上需要自申请权限 */
    public static boolean hasPermission(CordovaActivity cordovaActivity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            for (int i = 0; i < Permission.length; i++) {
                if (cordovaActivity.checkSelfPermission(Permission[i]) != PERMISSION_GRANTED) {
                    return false;
                }
            }
            return true;
        } else {
            return true;
        }
    }

    /* 请求程序所需权限 */
    public static void requestPermission(CordovaActivity cordovaActivity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            cordovaActivity.requestPermissions(Permission, PERMISSIONS_REQUEST);
        }
    }
}

在 Activity 中调用:

java
package com.example.hello;

import android.os.Bundle;

import com.junl.cordova.PermissionUtil;

import org.apache.cordova.CordovaActivity;

public class MainActivity extends CordovaActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        // 允许 Cordova 后台运行
        Bundle extras = getIntent().getExtras();
        if (extras != null && extras.getBoolean("cdvStartInBackground", false)) {
            moveTaskToBack(true);
        }

        /* 申请程序所需权限 */
        if (!PermissionUtil.hasPermission(this)) {
            PermissionUtil.requestPermission(this);
        }

        loadUrl(launchUrl);
        // loadUrl("https://www.baidu.com");
    }

}
package com.example.hello;

import android.os.Bundle;

import com.junl.cordova.PermissionUtil;

import org.apache.cordova.CordovaActivity;

public class MainActivity extends CordovaActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        // 允许 Cordova 后台运行
        Bundle extras = getIntent().getExtras();
        if (extras != null && extras.getBoolean("cdvStartInBackground", false)) {
            moveTaskToBack(true);
        }

        /* 申请程序所需权限 */
        if (!PermissionUtil.hasPermission(this)) {
            PermissionUtil.requestPermission(this);
        }

        loadUrl(launchUrl);
        // loadUrl("https://www.baidu.com");
    }

}

拍照裁剪保存问题

需要添加 addFlags 兼容高版本

java
// 剪裁
Intent crop = new Intent("com.android.camera.action.CROP");
crop.setDataAndType(fileUri, "image/*");
crop.putExtra("scale", true);
// 设置宽高比例
crop.putExtra("aspectX", 1);
crop.putExtra("aspectY", 1);
// 设置裁剪图片宽高
crop.putExtra("outputX", 340);
crop.putExtra("outputY", 340);
crop.putExtra(MediaStore.EXTRA_OUTPUT, fileUri);
// 高版本系统需要 addFlags
crop.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
crop.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
// 剪裁
Intent crop = new Intent("com.android.camera.action.CROP");
crop.setDataAndType(fileUri, "image/*");
crop.putExtra("scale", true);
// 设置宽高比例
crop.putExtra("aspectX", 1);
crop.putExtra("aspectY", 1);
// 设置裁剪图片宽高
crop.putExtra("outputX", 340);
crop.putExtra("outputY", 340);
crop.putExtra(MediaStore.EXTRA_OUTPUT, fileUri);
// 高版本系统需要 addFlags
crop.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
crop.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

最后编辑时间:

Version 4.0 (framework-1.0.0-rc.20)