当前位置:   article > 正文

讯飞SDK图文使用说明 语音听写、语音合成、声纹密码、人脸识别(适配Android7.0)_人脸识别sdk可以和录音转文字api合成吗

人脸识别sdk可以和录音转文字api合成吗
关于讯飞,他们关于语音做的AI功能SDK特别好,我经过使用有了深刻的体验,这次讲一讲语音听写、语音合成、声纹密码、人脸识别这三种的功能的体验。


1.首先到讯飞开放平台注册账号,然后到右上角点击我的应用创建应用,并给应用添加新功能
http://www.xfyun.cn/


2.然后在我的应用界面点击对应应用的栏目的SDK下载按钮跳转页面,这里我可以看到语音听写、人脸识别、语音合成、声纹识别都是免费的,然后其他的几个有体验标识的功能,每天也也可以使用500。



3.我们点击SDK下载按钮就会下载一个压缩包,解压后我们把sample文件夹下speechDemo文件夹的代码都移植到Android Studio上,因为这些代码原本都是Android Studio上创建的,所以不用改什么。我这里是直接创建一个包名一样的project,然后把speechDemo文件夹的代码直接移植过去。移植完后,还需要将根目录的assets、libs、res文件直接复制过去,选择全部覆盖。


4.适配Android6.0权限申请

在MainActivity的onCreate函数里加入以下代码

  1. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
  2. requestPermissions(new String[]{Manifest.permission.READ_PHONE_STATE, Manifest.permission.ACCESS_FINE_LOCATION
  3. , Manifest.permission.RECORD_AUDIO, Manifest.permission.READ_CONTACTS,
  4. Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.READ_EXTERNAL_STORAGE,
  5. Manifest.permission.ACCESS_COARSE_LOCATION, Manifest.permission.CAMERA}, 1001);
  6. }

5.适配Android7.0

首先在AndroidManifest里添加一个相机权限,因为有的手机调用系统相机需要这个权限

<uses-permission android:name="android.permission.CAMERA"/>

然后修改OnlineFaceDemo这个类,

  1. public class OnlineFaceDemo extends Activity implements View.OnClickListener {
  2. private final String TAG = "OnlineFaceDemo";
  3. private final int REQUEST_PICTURE_CHOOSE = 1;
  4. private final int REQUEST_CAMERA_IMAGE = 2;
  5. private Bitmap mImage = null;
  6. private byte[] mImageData = null;
  7. // authid为6-18个字符长度,用于唯一标识用户
  8. private String mAuthid = null;
  9. private Toast mToast;
  10. // 进度对话框
  11. private ProgressDialog mProDialog;
  12. private EditText online_authid;
  13. // 拍照得到的照片文件
  14. private File mPictureFile;
  15. // FaceRequest对象,集成了人脸识别的各种功能
  16. //private FaceRequest mFaceRequest;
  17. //采用身份识别接口进行在线人脸识别
  18. private IdentityVerifier mIdVerifier;
  19. // 模型操作
  20. private int mModelCmd;
  21. // 删除模型
  22. private final static int MODEL_DEL = 1;
  23. private static final int CODE_GALLERY_REQUEST = 0xa0;
  24. private static final int CODE_CAMERA_REQUEST = 0xa1;
  25. private static final int CODE_RESULT_REQUEST = 0xa2;
  26. private static final int CAMERA_PERMISSIONS_REQUEST_CODE = 0x03;
  27. private static final int STORAGE_PERMISSIONS_REQUEST_CODE = 0x04;
  28. private File fileUri = new File(Environment.getExternalStorageDirectory().getPath() + "/photo.jpg");
  29. private File fileCropUri = new File(Environment.getExternalStorageDirectory().getPath() + "/crop_photo.jpg");
  30. private Uri imageUri;
  31. private Uri cropImageUri;
  32. @SuppressLint("ShowToast")
  33. @Override
  34. protected void onCreate(Bundle savedInstanceState) {
  35. super.onCreate(savedInstanceState);
  36. this.requestWindowFeature(Window.FEATURE_NO_TITLE);
  37. setContentView(R.layout.online_facedemo);
  38. findViewById(R.id.online_pick).setOnClickListener(OnlineFaceDemo.this);
  39. findViewById(R.id.online_reg).setOnClickListener(OnlineFaceDemo.this);
  40. findViewById(R.id.online_verify).setOnClickListener(OnlineFaceDemo.this);
  41. findViewById(R.id.online_camera).setOnClickListener(OnlineFaceDemo.this);
  42. findViewById(R.id.btn_modle_delete).setOnClickListener(OnlineFaceDemo.this);
  43. findViewById(R.id.btn_identity).setOnClickListener(OnlineFaceDemo.this);
  44. online_authid = (EditText) findViewById(R.id.online_authid);
  45. mToast = Toast.makeText(this, "", Toast.LENGTH_SHORT);
  46. mProDialog = new ProgressDialog(this);
  47. mProDialog.setCancelable(true);
  48. mProDialog.setTitle("请稍后");
  49. mProDialog.setOnCancelListener(new DialogInterface.OnCancelListener() {
  50. @Override
  51. public void onCancel(DialogInterface dialog) {
  52. // cancel进度框时,取消正在进行的操作
  53. if (null != mIdVerifier) {
  54. mIdVerifier.cancel();
  55. }
  56. }
  57. });
  58. mIdVerifier = IdentityVerifier.createVerifier(OnlineFaceDemo.this, new InitListener() {
  59. @Override
  60. public void onInit(int errorCode) {
  61. if (ErrorCode.SUCCESS == errorCode) {
  62. showTip("引擎初始化成功");
  63. } else {
  64. showTip("引擎初始化失败,错误码:" + errorCode);
  65. }
  66. }
  67. });
  68. }
  69. private void register(JSONObject obj) throws JSONException {
  70. int ret = obj.getInt("ret");
  71. if (ret != 0) {
  72. showTip("注册失败");
  73. return;
  74. }
  75. if ("success".equals(obj.get("rst"))) {
  76. showTip("注册成功");
  77. } else {
  78. showTip("注册失败");
  79. }
  80. }
  81. private void verify(JSONObject obj) throws JSONException {
  82. int ret = obj.getInt("ret");
  83. if (ret != 0) {
  84. showTip("验证失败");
  85. return;
  86. }
  87. if ("success".equals(obj.get("rst"))) {
  88. if (obj.getBoolean("verf")) {
  89. showTip("通过验证,欢迎回来!");
  90. } else {
  91. showTip("验证不通过");
  92. }
  93. } else {
  94. showTip("验证失败");
  95. }
  96. }
  97. /**
  98. * 人脸注册监听器
  99. */
  100. private IdentityListener mEnrollListener = new IdentityListener() {
  101. @Override
  102. public void onResult(IdentityResult result, boolean islast) {
  103. Log.d(TAG, result.getResultString());
  104. if (null != mProDialog) {
  105. mProDialog.dismiss();
  106. }
  107. try {
  108. JSONObject object = new JSONObject(result.getResultString());
  109. int ret = object.getInt("ret");
  110. if (ErrorCode.SUCCESS == ret) {
  111. showTip("注册成功");
  112. }else {
  113. showTip(new SpeechError(ret).getPlainDescription(true));
  114. }
  115. } catch (JSONException e) {
  116. e.printStackTrace();
  117. }
  118. }
  119. @Override
  120. public void onEvent(int eventType, int arg1, int arg2, Bundle obj) {
  121. }
  122. @Override
  123. public void onError(SpeechError error) {
  124. if (null != mProDialog) {
  125. mProDialog.dismiss();
  126. }
  127. showTip(error.getPlainDescription(true));
  128. }
  129. };
  130. /**
  131. * 人脸验证监听器
  132. */
  133. private IdentityListener mVerifyListener = new IdentityListener() {
  134. @Override
  135. public void onResult(IdentityResult result, boolean islast) {
  136. Log.d(TAG, result.getResultString());
  137. if (null != mProDialog) {
  138. mProDialog.dismiss();
  139. }
  140. try {
  141. JSONObject object = new JSONObject(result.getResultString());
  142. Log.d(TAG,"object is: "+object.toString());
  143. String decision = object.getString("decision");
  144. if ("accepted".equalsIgnoreCase(decision)) {
  145. showTip("通过验证");
  146. } else {
  147. showTip("验证失败");
  148. }
  149. } catch (JSONException e) {
  150. e.printStackTrace();
  151. }
  152. }
  153. @Override
  154. public void onEvent(int eventType, int arg1, int arg2, Bundle obj) {
  155. }
  156. @Override
  157. public void onError(SpeechError error) {
  158. if (null != mProDialog) {
  159. mProDialog.dismiss();
  160. }
  161. showTip(error.getPlainDescription(true));
  162. }
  163. };
  164. /**
  165. * 人脸模型操作监听器
  166. */
  167. private IdentityListener mModelListener = new IdentityListener() {
  168. @Override
  169. public void onResult(IdentityResult result, boolean islast) {
  170. Log.d(TAG, result.getResultString());
  171. JSONObject jsonResult = null;
  172. int ret = ErrorCode.SUCCESS;
  173. try {
  174. jsonResult = new JSONObject(result.getResultString());
  175. ret = jsonResult.getInt("ret");
  176. } catch (JSONException e) {
  177. e.printStackTrace();
  178. }
  179. // 根据操作类型判断结果类型
  180. switch (mModelCmd) {
  181. case MODEL_DEL:
  182. if (ErrorCode.SUCCESS == ret) {
  183. online_authid.setEnabled(true);
  184. showTip("删除成功");
  185. } else {
  186. showTip("删除失败");
  187. }
  188. break;
  189. default:
  190. break;
  191. }
  192. }
  193. @Override
  194. public void onEvent(int eventType, int arg1, int arg2, Bundle obj) {
  195. }
  196. @Override
  197. public void onError(SpeechError error) {
  198. // 弹出错误信息
  199. showTip(error.getPlainDescription(true));
  200. }
  201. };
  202. private void executeModelCommand(String cmd) {
  203. // 设置人脸模型操作参数
  204. // 清空参数
  205. mIdVerifier.setParameter(SpeechConstant.PARAMS, null);
  206. // 设置会话场景
  207. mIdVerifier.setParameter(SpeechConstant.MFV_SCENES, "ifr");
  208. // 用户id
  209. mIdVerifier.setParameter(SpeechConstant.AUTH_ID, mAuthid);
  210. // 设置模型参数,若无可以传空字符传
  211. StringBuffer params = new StringBuffer();
  212. // 执行模型操作
  213. mIdVerifier.execute("ifr", cmd, params.toString(), mModelListener);
  214. }
  215. @Override
  216. public void onClick(View view) {
  217. int ret = ErrorCode.SUCCESS;
  218. mAuthid = online_authid.getText().toString();
  219. if(TextUtils.isEmpty(mAuthid)) {
  220. showTip("请输入用户ID");
  221. return;
  222. }else {
  223. online_authid.setEnabled(false);
  224. }
  225. switch (view.getId()) {
  226. case R.id.online_pick:
  227. PhotoUtils.openPic(this, CODE_GALLERY_REQUEST);
  228. /* Intent intent = new Intent();
  229. intent.setType("image*//*");
  230. intent.setAction(Intent.ACTION_PICK);
  231. startActivityForResult(intent, REQUEST_PICTURE_CHOOSE);*/
  232. break;
  233. case R.id.online_reg:
  234. if (TextUtils.isEmpty(mAuthid)) {
  235. showTip("authid不能为空");
  236. return;
  237. }
  238. if (null != mImageData) {
  239. mProDialog.setMessage("注册中...");
  240. mProDialog.show();
  241. // 设置用户标识,格式为6-18个字符(由字母、数字、下划线组成,不得以数字开头,不能包含空格)。
  242. // 当不设置时,云端将使用用户设备的设备ID来标识终端用户。
  243. // 设置人脸注册参数
  244. // 清空参数
  245. mIdVerifier.setParameter(SpeechConstant.PARAMS, null);
  246. // 设置会话场景
  247. mIdVerifier.setParameter(SpeechConstant.MFV_SCENES, "ifr");
  248. // 设置会话类型
  249. mIdVerifier.setParameter(SpeechConstant.MFV_SST, "enroll");
  250. // 设置用户id
  251. mIdVerifier.setParameter(SpeechConstant.AUTH_ID, mAuthid);
  252. // 设置监听器,开始会话
  253. mIdVerifier.startWorking(mEnrollListener);
  254. // 子业务执行参数,若无可以传空字符传
  255. StringBuffer params = new StringBuffer();
  256. // 向子业务写入数据,人脸数据可以一次写入
  257. mIdVerifier.writeData("ifr", params.toString(), mImageData, 0, mImageData.length);
  258. // 停止写入
  259. mIdVerifier.stopWrite("ifr");
  260. } else {
  261. showTip("请选择图片后再注册");
  262. }
  263. break;
  264. case R.id.online_verify:
  265. mAuthid = ((EditText) findViewById(R.id.online_authid)).getText().toString();
  266. if (TextUtils.isEmpty(mAuthid)) {
  267. showTip("authid不能为空");
  268. return;
  269. }
  270. if (null != mImageData) {
  271. mProDialog.setMessage("验证中...");
  272. mProDialog.show();
  273. // 设置人脸验证参数
  274. // 清空参数
  275. mIdVerifier.setParameter(SpeechConstant.PARAMS, null);
  276. // 设置会话场景
  277. mIdVerifier.setParameter(SpeechConstant.MFV_SCENES, "ifr");
  278. // 设置会话类型
  279. mIdVerifier.setParameter(SpeechConstant.MFV_SST, "verify");
  280. // 设置验证模式,单一验证模式:sin
  281. mIdVerifier.setParameter(SpeechConstant.MFV_VCM, "sin");
  282. // 用户id
  283. mIdVerifier.setParameter(SpeechConstant.AUTH_ID, mAuthid);
  284. // 设置监听器,开始会话
  285. mIdVerifier.startWorking(mVerifyListener);
  286. // 子业务执行参数,若无可以传空字符传
  287. StringBuffer params = new StringBuffer();
  288. // 向子业务写入数据,人脸数据可以一次写入
  289. mIdVerifier.writeData("ifr", params.toString(), mImageData, 0, mImageData.length);
  290. // 停止写入
  291. mIdVerifier.stopWrite("ifr");
  292. } else {
  293. showTip("请选择图片后再验证");
  294. }
  295. break;
  296. case R.id.online_camera:
  297. // 设置相机拍照后照片保存路径
  298. imageUri = Uri.fromFile(fileUri);
  299. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
  300. imageUri = FileProvider.getUriForFile(OnlineFaceDemo.this, "com.zz.fileprovider", fileUri);//通过FileProvider创建一个content类型的Uri
  301. PhotoUtils.takePicture(this, imageUri, CODE_CAMERA_REQUEST);
  302. break;
  303. case R.id.btn_modle_delete:
  304. // 人脸模型删除
  305. mModelCmd = MODEL_DEL;
  306. executeModelCommand("delete");
  307. break;
  308. case R.id.btn_identity:
  309. Intent init = new Intent(OnlineFaceDemo.this, GroupManagerActivity.class);
  310. init.putExtra("auth_id",mAuthid);
  311. init.putExtra("mfv_scenes","ifr");
  312. startActivity(init);
  313. break;
  314. default:
  315. break;
  316. }//end of switch
  317. if( ErrorCode.SUCCESS != ret ){
  318. mProDialog.dismiss();
  319. showTip( "出现错误:"+ret );
  320. }
  321. }
  322. private static final int OUTPUT_X = 480;
  323. private static final int OUTPUT_Y = 480;
  324. @Override
  325. protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  326. if (resultCode != RESULT_OK) {
  327. return;
  328. }
  329. String fileSrc = null;
  330. if (requestCode == REQUEST_PICTURE_CHOOSE) {
  331. if ("file".equals(data.getData().getScheme())) {
  332. // 有些低版本机型返回的Uri模式为file
  333. fileSrc = data.getData().getPath();
  334. } else {
  335. // Uri模型为content
  336. String[] proj = {MediaStore.Images.Media.DATA};
  337. Cursor cursor = getContentResolver().query(data.getData(), proj,
  338. null, null, null);
  339. cursor.moveToFirst();
  340. int idx = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
  341. fileSrc = cursor.getString(idx);
  342. cursor.close();
  343. }
  344. // 跳转到图片裁剪页面
  345. FaceUtil.cropPicture(this,Uri.fromFile(new File(fileSrc)));
  346. } else if (requestCode == REQUEST_CAMERA_IMAGE) {
  347. if (null == mPictureFile) {
  348. showTip("拍照失败,请重试");
  349. return;
  350. }
  351. /* fileSrc = mPictureFile.getAbsolutePath();
  352. updateGallery(fileSrc);*/
  353. // 跳转到图片裁剪页面
  354. // FaceUtil.cropPicture(this, FileProvider.getUriForFile(OnlineFaceDemo.this, "com.zz.fileprovider", new File(fileSrc)));
  355. cropImageUri = Uri.fromFile(fileCropUri);
  356. PhotoUtils.cropImageUri(this, imageUri, cropImageUri, 1, 1, OUTPUT_X, OUTPUT_Y, CODE_RESULT_REQUEST);
  357. } else if (requestCode == FaceUtil.REQUEST_CROP_IMAGE) {
  358. // 获取返回数据
  359. Bitmap bmp = data.getParcelableExtra("data");
  360. // 若返回数据不为null,保存至本地,防止裁剪时未能正常保存
  361. if(null != bmp){
  362. FaceUtil.saveBitmapToFile(OnlineFaceDemo.this, bmp);
  363. }
  364. // 获取图片保存路径
  365. fileSrc = FaceUtil.getImagePath(OnlineFaceDemo.this);
  366. // 获取图片的宽和高
  367. BitmapFactory.Options options = new BitmapFactory.Options();
  368. options.inJustDecodeBounds = true;
  369. mImage = BitmapFactory.decodeFile(fileSrc, options);
  370. // 压缩图片
  371. options.inSampleSize = Math.max(1, (int) Math.ceil(Math.max(
  372. (double) options.outWidth / 1024f,
  373. (double) options.outHeight / 1024f)));
  374. options.inJustDecodeBounds = false;
  375. mImage = BitmapFactory.decodeFile(fileSrc, options);
  376. // 若mImageBitmap为空则图片信息不能正常获取
  377. if(null == mImage) {
  378. showTip("图片信息无法正常获取!");
  379. return;
  380. }
  381. // 部分手机会对图片做旋转,这里检测旋转角度
  382. int degree = FaceUtil.readPictureDegree(fileSrc);
  383. if (degree != 0) {
  384. // 把图片旋转为正的方向
  385. mImage = FaceUtil.rotateImage(degree, mImage);
  386. }
  387. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  388. //可根据流量及网络状况对图片进行压缩
  389. mImage.compress(Bitmap.CompressFormat.JPEG, 80, baos);
  390. mImageData = baos.toByteArray();
  391. ((ImageView) findViewById(R.id.online_img)).setImageBitmap(mImage);
  392. }
  393. if (resultCode == RESULT_OK) {
  394. switch (requestCode) {
  395. //拍照完成回调
  396. case CODE_CAMERA_REQUEST:
  397. cropImageUri = Uri.fromFile(fileCropUri);
  398. PhotoUtils.cropImageUri(this, imageUri, cropImageUri, 1, 1, OUTPUT_X, OUTPUT_Y, CODE_RESULT_REQUEST);
  399. break;
  400. //访问相册完成回调
  401. case CODE_GALLERY_REQUEST:
  402. cropImageUri = Uri.fromFile(fileCropUri);
  403. Uri newUri = Uri.parse(PhotoUtils.getPath(this, data.getData()));
  404. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
  405. newUri = FileProvider.getUriForFile(this, "com.zz.fileprovider", new File(newUri.getPath()));
  406. }
  407. PhotoUtils.cropImageUri(this, newUri, cropImageUri, 1, 1, OUTPUT_X, OUTPUT_Y, CODE_RESULT_REQUEST);
  408. break;
  409. case CODE_RESULT_REQUEST:
  410. Bitmap bitmap = PhotoUtils.getBitmapFromUri(cropImageUri, this);
  411. if (bitmap != null) {
  412. ((ImageView) findViewById(R.id.online_img)).setImageBitmap(bitmap);
  413. mImage = bitmap;
  414. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  415. //可根据流量及网络状况对图片进行压缩
  416. mImage.compress(Bitmap.CompressFormat.JPEG, 80, baos);
  417. mImageData = baos.toByteArray();
  418. }
  419. break;
  420. default:
  421. }
  422. }
  423. }
  424. @Override
  425. public void finish() {
  426. if (null != mProDialog) {
  427. mProDialog.dismiss();
  428. }
  429. super.finish();
  430. }
  431. private void updateGallery(String filename) {
  432. MediaScannerConnection.scanFile(this, new String[] {filename}, null,
  433. new MediaScannerConnection.OnScanCompletedListener() {
  434. @Override
  435. public void onScanCompleted(String path, Uri uri) {
  436. }
  437. });
  438. }
  439. private void showTip(final String str) {
  440. mToast.setText(str);
  441. mToast.show();
  442. }
  443. }

其中需要我们新建一个类,为了专门完成调用系统相机和相册,也是为了替代项目中FaceUtil类,因为他不适配Android7.0

  1. public class PhotoUtils {
  2. private static final String TAG = "PhotoUtils";
  3. /**
  4. * @param activity 当前activity
  5. * @param imageUri 拍照后照片存储路径
  6. * @param requestCode 调用系统相机请求码
  7. */
  8. public static void takePicture(Activity activity, Uri imageUri, int requestCode) {
  9. //调用系统相机
  10. Intent intentCamera = new Intent();
  11. intentCamera.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
  12. //将拍照结果保存至photo_file的Uri中,不保留在相册中
  13. intentCamera.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
  14. activity.startActivityForResult(intentCamera, requestCode);
  15. }
  16. /**
  17. * @param activity 当前activity
  18. * @param requestCode 打开相册的请求码
  19. */
  20. public static void openPic(Activity activity, int requestCode) {
  21. Intent photoPickerIntent = new Intent(Intent.ACTION_GET_CONTENT);
  22. photoPickerIntent.setType("image/*");
  23. activity.startActivityForResult(photoPickerIntent, requestCode);
  24. }
  25. /**
  26. * @param activity 当前activity
  27. * @param orgUri 剪裁原图的Uri
  28. * @param desUri 剪裁后的图片的Uri
  29. * @param aspectX X方向的比例
  30. * @param aspectY Y方向的比例
  31. * @param width 剪裁图片的宽度
  32. * @param height 剪裁图片高度
  33. * @param requestCode 剪裁图片的请求码
  34. */
  35. public static void cropImageUri(Activity activity, Uri orgUri, Uri desUri, int aspectX, int aspectY, int width, int height, int requestCode) {
  36. Intent intent = new Intent("com.android.camera.action.CROP");
  37. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
  38. intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
  39. }
  40. intent.setDataAndType(orgUri, "image/*");
  41. //发送裁剪信号
  42. intent.putExtra("crop", "true");
  43. intent.putExtra("aspectX", aspectX);
  44. intent.putExtra("aspectY", aspectY);
  45. intent.putExtra("outputX", width);
  46. intent.putExtra("outputY", height);
  47. intent.putExtra("scale", true);
  48. //将剪切的图片保存到目标Uri中
  49. intent.putExtra(MediaStore.EXTRA_OUTPUT, desUri);
  50. //1-false用uri返回图片
  51. //2-true直接用bitmap返回图片(此种只适用于小图片,返回图片过大会报错)
  52. intent.putExtra("return-data", false);
  53. intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
  54. intent.putExtra("noFaceDetection", true);
  55. activity.startActivityForResult(intent, requestCode);
  56. }
  57. /**
  58. * 读取uri所在的图片
  59. *
  60. * @param uri 图片对应的Uri
  61. * @param mContext 上下文对象
  62. * @return 获取图像的Bitmap
  63. */
  64. public static Bitmap getBitmapFromUri(Uri uri, Context mContext) {
  65. try {
  66. Bitmap bitmap = MediaStore.Images.Media.getBitmap(mContext.getContentResolver(), uri);
  67. return bitmap;
  68. } catch (Exception e) {
  69. e.printStackTrace();
  70. return null;
  71. }
  72. }
  73. /**
  74. * @param context 上下文对象
  75. * @param uri 当前相册照片的Uri
  76. * @return 解析后的Uri对应的String
  77. */
  78. @SuppressLint("NewApi")
  79. public static String getPath(final Context context, final Uri uri) {
  80. final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;
  81. String pathHead = "file:///";
  82. // DocumentProvider
  83. if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
  84. // ExternalStorageProvider
  85. if (isExternalStorageDocument(uri)) {
  86. final String docId = DocumentsContract.getDocumentId(uri);
  87. final String[] split = docId.split(":");
  88. final String type = split[0];
  89. if ("primary".equalsIgnoreCase(type)) {
  90. return pathHead + Environment.getExternalStorageDirectory() + "/" + split[1];
  91. }
  92. }
  93. // DownloadsProvider
  94. else if (isDownloadsDocument(uri)) {
  95. final String id = DocumentsContract.getDocumentId(uri);
  96. final Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));
  97. return pathHead + getDataColumn(context, contentUri, null, null);
  98. }
  99. // MediaProvider
  100. else if (isMediaDocument(uri)) {
  101. final String docId = DocumentsContract.getDocumentId(uri);
  102. final String[] split = docId.split(":");
  103. final String type = split[0];
  104. Uri contentUri = null;
  105. if ("image".equals(type)) {
  106. contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
  107. } else if ("video".equals(type)) {
  108. contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
  109. } else if ("audio".equals(type)) {
  110. contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
  111. }
  112. final String selection = "_id=?";
  113. final String[] selectionArgs = new String[]{split[1]};
  114. return pathHead + getDataColumn(context, contentUri, selection, selectionArgs);
  115. }
  116. }
  117. // MediaStore (and general)
  118. else if ("content".equalsIgnoreCase(uri.getScheme())) {
  119. return pathHead + getDataColumn(context, uri, null, null);
  120. }
  121. // File
  122. else if ("file".equalsIgnoreCase(uri.getScheme())) {
  123. return pathHead + uri.getPath();
  124. }
  125. return null;
  126. }
  127. /**
  128. * Get the value of the data column for this Uri. This is useful for
  129. * MediaStore Uris, and other file-based ContentProviders.
  130. *
  131. * @param context The context.
  132. * @param uri The Uri to query.
  133. * @param selection (Optional) Filter used in the query.
  134. * @param selectionArgs (Optional) Selection arguments used in the query.
  135. * @return The value of the _data column, which is typically a file path.
  136. */
  137. private static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {
  138. Cursor cursor = null;
  139. final String column = "_data";
  140. final String[] projection = {column};
  141. try {
  142. cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null);
  143. if (cursor != null && cursor.moveToFirst()) {
  144. final int column_index = cursor.getColumnIndexOrThrow(column);
  145. return cursor.getString(column_index);
  146. }
  147. } finally {
  148. if (cursor != null)
  149. cursor.close();
  150. }
  151. return null;
  152. }
  153. /**
  154. * @param uri The Uri to check.
  155. * @return Whether the Uri authority is ExternalStorageProvider.
  156. */
  157. private static boolean isExternalStorageDocument(Uri uri) {
  158. return "com.android.externalstorage.documents".equals(uri.getAuthority());
  159. }
  160. /**
  161. * @param uri The Uri to check.
  162. * @return Whether the Uri authority is DownloadsProvider.
  163. */
  164. private static boolean isDownloadsDocument(Uri uri) {
  165. return "com.android.providers.downloads.documents".equals(uri.getAuthority());
  166. }
  167. /**
  168. * @param uri The Uri to check.
  169. * @return Whether the Uri authority is MediaProvider.
  170. */
  171. private static boolean isMediaDocument(Uri uri) {
  172. return "com.android.providers.media.documents".equals(uri.getAuthority());
  173. }
  174. }

运行项目安装app,

首先是语音听写,点击开始我们就可以说话了,他也会弹出一个对话框显示正在录音,你也可以再点击一下屏幕,它会停止录音。这个听写的准确度,一般还可以,但是对于声音相似的单词就容易错。然后使用这个听写是不需要流量的。


然后是在线语音合成,其实就是把文字语音播放出来,与语音听写相对立。这个功能需要流量,然后我们可以选择发音人。


最后一个是声纹密码,首先输入authId,这个其实就是相当于用户名,你就随便输入几个字母,然后点击获取密码,然后点击声纹注册再点击开始录音,他会让你读一段数字,这个录音要进行五次。然后点击1:1验证,再点击开始录音,这个时候他会让你读一段数字,然后他判断你是否是这个authId所代表的用户。

最后是人脸识别,首先是填写authid,然后是选图(调用系统相册)或者拍照(调用系统相机),注册。注册成功后,你就可以通过再次选图或者拍照,点击1:1验证,可以通过两次图片判断是否是同一个人。他还有人脸模型删除,和1:N验证





声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家自动化/article/detail/225470
推荐阅读
相关标签
  

闽ICP备14008679号