活动介绍

package com.chinaiatb.mrdoctor.utils; import android.Manifest; import android.app.Activity; import android.content.ActivityNotFoundException; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.database.Cursor; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.PixelFormat; import android.net.Uri; import android.os.Build; import android.os.Bundle; import android.os.Environment; import android.os.Handler; import android.provider.MediaStore; import androidx.annotation.NonNull; import androidx.core.content.FileProvider; import android.provider.OpenableColumns; import android.provider.Settings; import android.text.TextUtils; import android.util.Log; import android.view.Gravity; import android.view.View; import android.view.WindowManager; import android.webkit.MimeTypeMap; import android.widget.TextView; import android.widget.Toast; import com.blankj.utilcode.util.LogUtils; import com.chinaiatb.mrdoctor.R; import com.chinaiatb.mrdoctor.base.ui.widget.ActionSheetDialog; import com.chinaiatb.mrdoctor.base.utils.FileUtils; import com.chinaiatb.mrdoctor.base.utils.PermissionUtils; import com.chinaiatb.mrdoctor.base.utils.ToastUtil; import com.chinaiatb.mrdoctor.configs.GlobalConstant; import com.chinaiatb.mrdoctor.configs.ParamsConfigs; import com.chinaiatb.mrdoctor.ui.test.friend.view.SelectImageActivity; import com.chinaiatb.mrdoctor.ui.test.model.Image; import com.google.android.material.snackbar.Snackbar; import com.huawei.hms.utils.IOUtils; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Date; import java.util.List; import java.util.Locale; import static com.blankj.utilcode.util.StringUtils.getString; import static com.chinaiatb.mrdoctor.configs.IntentParameter.TAG; /** * description: * 2020/12/10. * auth:lihe */ public class AddPictureFileUtils { private ActionSheetDialog sheetDialog; private Activity mActivity; public AddPictureFileUtils(Activity activity, int singleOrMulti, boolean needCrop){ this.mActivity = activity; this.singleOrMulti = singleOrMulti; if(needCrop){ FILE_SAVEPATH = getSDPath(mActivity.getBaseContext()); // SELECT_FILE_SAVEPATH = getSDPath(mActivity.getBaseContext()); TEMP_DIR_PATH = ParamsConfigs.TEMP_SAVE_PATH_11; } } public AddPictureFileUtils(Activity activity, int singleOrMulti){ this.mActivity = activity; this.singleOrMulti = singleOrMulti; } private final static String AUTHORITY_NAME = "com.chinaiatb.mrdoctor.fileprovider"; private String FILE_SAVEPATH = ParamsConfigs.PICTURE_SAVE_PATH; private String TEMP_DIR_PATH = ParamsConfigs.TEMP_SAVE_PATH; private String SELECT_FILE_SAVEPATH = ParamsConfigs.SELECT_FILE_SAVE_PATH; private String FileFlag; private Uri origUri;//原始uri private Uri cropUri;//剪裁后的uri private Uri selectFileUri; private File uploadPicFile;//上传照片 private String uploadPicPath;//上传照片路径 private File uploadSelectFile;//上传文件 private String uploadSelectFilePath;//上传文件路径 private List<String> selectedFileNames = new ArrayList<>(); private File uploadOriPicFile;//原始照片 private String uploadOriPicPath;//原始照片路径 private String fileType; private boolean needCutPicture;//标记是否需要剪裁 private int singleOrMulti = SINGLE_CHOOSE;//标记单选还是多选 public static final int SINGLE_CHOOSE = 1;//照片单选f public static final int MULTI_CHOOSE = 2;//照片多选 /** 请求相册 */ public static final int REQUEST_CODE_GETIMAGE_BYSDCARD = 0; /** 请求相机 */ public static final int REQUEST_CODE_GETIMAGE_BYCAMERA = 1; /** 请求裁剪 */ public static final int REQUEST_CODE_GETIMAGE_BYCROP = 2; /** 请求文件 */ public static final int REQUEST_CODE_GETFILE = 3; private final int CROP = 200; /** 请求存储权限 */ private static final int REQUEST_CODE_STORAGE_PERMISSION = 4; /** * 检查相机存储权限 */ public void checkPremission(boolean needCutPicture,String fileFlag) { FileFlag = fileFlag; this.needCutPicture = needCutPicture; CharSequence[] items = { getString(R.string.str_img_from_camera), getString(R.string.str_img_from_album) }; imageChooseItem(items); } /* * 获取SDK=30及以上的存储路径 * */ public static String getSDPath(Context context) { File sdDir = null; boolean sdCardExist = Environment.getExternalStorageState().equals( Environment.MEDIA_MOUNTED);// 判断sd卡是否存在 if (sdCardExist) { if (Build.VERSION.SDK_INT>=29){ //Android10之后 sdDir = context.getExternalFilesDir(null); }else { sdDir = Environment.getExternalStorageDirectory();// 获取SD卡根目录 } } else { sdDir = Environment.getRootDirectory();// 获取跟目录 } return sdDir.toString(); } public void checkCameraPermissions(){ String[] permissions = new String[]{ // Manifest.permission.WRITE_EXTERNAL_STORAGE, // Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.CAMERA}; //这里的this不是上下文,是Activity对象! PermissionUtils.getInstance().chekPermissions(mActivity, permissions, permissionsCameraResult,1); // ToastUtil.showLongToastTop("用于扫码登录、扫码签名、患者问题答复时添加图片或视频信息"); } //创建监听权限的接口对象 PermissionUtils.IPermissionsResult permissionsCameraResult = new PermissionUtils.IPermissionsResult() { @Override public void passPermissons() { onSheetDialogClick(1); } @Override public void forbitPermissons() { ToastUtil.showShortToast("没有权限!"); } }; public void checkFilePermissions(){ String[] permissions = new String[]{ Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.READ_EXTERNAL_STORAGE, }; //这里的this不是上下文,是Activity对象! PermissionUtils.getInstance().chekPermissions(mActivity, permissions, permissionsFileResult,1); } PermissionUtils.IPermissionsResult permissionsFileResult = new PermissionUtils.IPermissionsResult() { @Override public void passPermissons() { onSheetDialogClick(2); } @Override public void forbitPermissons() { ToastUtil.showShortToast("没有权限!"); } }; public void checkAlbumPermissions(){ String[] permissions = new String[]{ Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.READ_EXTERNAL_STORAGE, }; //这里的this不是上下文,是Activity对象! PermissionUtils.getInstance().chekPermissions(mActivity, permissions, permissionsAlbumResult,1); } //创建监听权限的接口对象 PermissionUtils.IPermissionsResult permissionsAlbumResult = new PermissionUtils.IPermissionsResult() { @Override public void passPermissons() { onSheetDialogClick(0); } @Override public void forbitPermissons() { ToastUtil.showShortToast("没有权限!"); } }; public void initDialog(){ if(null != sheetDialog){ return; } sheetDialog = new ActionSheetDialog(mActivity).builder(); sheetDialog.addSheetItem(getString(R.string.str_img_from_camera), null, new ActionSheetDialog.OnSheetItemClickListener() { @Override public void onClick(int which) { if(GlobalConstant.isAtRoomActivity){//还在视频过程中,不能拍照 ToastUtil.showShortToast("视频通话中,无法使用此功能"); }else{ checkCameraPermissions(); } } }); sheetDialog.addSheetItem(getString(R.string.str_img_from_album), null, new ActionSheetDialog.OnSheetItemClickListener() { @Override public void onClick(int which) { checkAlbumPermissions(); } }); //if (FileFlag=="1"){ if ("1".equals(FileFlag)){ sheetDialog.addSheetItem(getString(R.string.str_file_select), null, new ActionSheetDialog.OnSheetItemClickListener() { @Override public void onClick(int which) { checkFilePermissions(); } });} } /** * 相机选择和拍照 * @param items */ private void imageChooseItem(CharSequence[] items) { mActivity.runOnUiThread(new Runnable() { @Override public void run() { initDialog(); if (mActivity == null || mActivity.isDestroyed() || mActivity.isFinishing()) { return; } sheetDialog.show(); } }); } private void onSheetDialogClick(int position){ //判断是否挂载了SD卡 String storageState = Environment.getExternalStorageState(); if(storageState.equals(Environment.MEDIA_MOUNTED)){ File savedir = new File(FILE_SAVEPATH); File tempdir = new File(TEMP_DIR_PATH); if (!savedir.exists()) { savedir.mkdirs(); } if (!tempdir.exists()) { tempdir.mkdirs(); } } else{ ToastUtil.showShortToast("SD卡未挂载,不能操作照片"); return; } //输出裁剪的临时文件 String timeStamp = new SimpleDateFormat("yyyyMMddHHmmss", Locale.getDefault()).format(new Date()); //照片命名 String origFileName = "byt_" + timeStamp + ".jpg"; String cropFileName = "byt_crop_" + timeStamp + ".jpg"; uploadOriPicPath = FILE_SAVEPATH + origFileName; uploadOriPicFile = new File(uploadOriPicPath); //裁剪头像的绝对路径 uploadPicPath = FILE_SAVEPATH + cropFileName; uploadPicFile = new File(uploadPicPath); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { origUri = FileProvider.getUriForFile(mActivity, AUTHORITY_NAME, uploadOriPicFile); }else{ origUri = Uri.fromFile(uploadOriPicFile); } cropUri = Uri.fromFile(uploadPicFile); //相册选图 if(position == 0) { if(singleOrMulti == SINGLE_CHOOSE){ startImagePick(); }else{ startActivity(); } } //手机拍照 else if(position == 1){ startActionCamera(); } //文件管理 else if(position == 2) { startActionFile(); } } private ArrayList<Image> mSelectImages = new ArrayList<>(); public static final int SELECT_IMAGE_REQUEST = 0x0011; private void startActivity() { Intent intent = new Intent(mActivity, SelectImageActivity.class); intent.putParcelableArrayListExtra("selected_images", mSelectImages); mActivity.startActivityForResult(intent, SELECT_IMAGE_REQUEST); } /** * 选择图片并直接裁剪 */ private void startImagePick() { // // 检查权限是否已授予 // if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { // if (!Environment.isExternalStorageManager()) { // // 如果没有权限,打开设置页面让用户授予权限 // Intent intent = new Intent(Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSION); // mActivity.startActivityForResult(intent, REQUEST_CODE_STORAGE_PERMISSION); // return; // } // } // // 否则,执行正常的图片选择操作 Intent intent = new Intent(Intent.ACTION_PICK,null); intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*"); mActivity.startActivityForResult(Intent.createChooser(intent, "选择图片"),REQUEST_CODE_GETIMAGE_BYSDCARD); } /** * 相机拍照 */ private void startActionCamera() { Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); intent.putExtra(MediaStore.EXTRA_OUTPUT, selectFileUri/*this.getCameraTempFile()*/); // 添加这一句表示对目标应用临时授权该Uri所代表的文件 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION); } mActivity.startActivityForResult(intent, REQUEST_CODE_GETIMAGE_BYCAMERA); } private void startActionFile() { Intent intent = new Intent(Intent.ACTION_GET_CONTENT); intent.setType("application/pdf,application/msword,application/vnd.openxmlformats-officedocument.wordprocessingml.document,application/vnd.ms-excel,application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"); intent.addCategory(Intent.CATEGORY_OPENABLE); // 添加可打开的文件分类 mActivity.startActivityForResult(Intent.createChooser(intent, "选择文件"), REQUEST_CODE_GETFILE); } /** * 拍照后裁剪 * @param data 原始图片 */ private void startActionCrop(Uri data, boolean fromCamera) { Intent intent = new Intent("com.android.camera.action.CROP"); intent.setDataAndType(data, "image/*"); if (fromCamera && Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION); } intent.putExtra("return-data", false); intent.putExtra("crop", "true"); intent.putExtra("aspectX", 1);// 裁剪框比例 intent.putExtra("aspectY", 1); intent.putExtra("outputX", CROP); intent.putExtra("outputY", CROP); intent.putExtra("scale", true); intent.putExtra("scaleUpIfNeeded", true);// 去黑边 intent.putExtra(MediaStore.EXTRA_OUTPUT, cropUri); mActivity.startActivityForResult(intent, REQUEST_CODE_GETIMAGE_BYCROP); } private void startActionCrop2(File targetPhoto) { Intent intent = new Intent("com.android.camera.action.CROP"); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { // intent.setAction(Intent.ACTION_OPEN_DOCUMENT); intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION); intent.setDataAndType(getUriByFileForN(targetPhoto), "image/*"); }else{ intent.setDataAndType(Uri.fromFile(targetPhoto), "image/*"); } intent.putExtra(MediaStore.EXTRA_OUTPUT, cropUri); intent.putExtra("return-data", false); intent.putExtra("crop", "true"); intent.putExtra("aspectX", 1);// 裁剪框比例 intent.putExtra("aspectY", 1); intent.putExtra("outputX", CROP); intent.putExtra("outputY", CROP); intent.putExtra("scale", true); intent.putExtra("scaleUpIfNeeded", true);// 去黑边 mActivity.startActivityForResult(intent, REQUEST_CODE_GETIMAGE_BYCROP); } private void startActionCrop3(Uri sourceUri) { uploadPicFile = FileUtils.createImageFile(mActivity.getApplication(), true); Intent intent = new Intent("com.android.camera.action.CROP"); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { // Android N 及以上版本,确保裁剪图片时可以访问文件 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION); intent.setDataAndType(sourceUri, "image/*"); } // 设置裁剪输出文件的路径,针对不同版本需要做不同的处理 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { // Android 10 及以上版本,需要通过 FileProvider 获取 content:// URI intent.putExtra(MediaStore.EXTRA_OUTPUT, FileUtils.uri); // FileUtils.uri 是通过 FileProvider 提供的 URI } else { // 对于 Android 9 以下版本,使用 Uri.fromFile 来处理 file:// URI Uri imgCropUri = Uri.fromFile(uploadPicFile); intent.putExtra(MediaStore.EXTRA_OUTPUT, imgCropUri); } //intent.putExtra("return-data", true); // 设置为 true 来返回裁剪数据 // 设置裁剪的相关参数 intent.putExtra("return-data", false); // 不返回裁剪数据,返回文件 URI intent.putExtra("crop", "true"); intent.putExtra("aspectX", 1); // 裁剪框的比例 intent.putExtra("aspectY", 1); intent.putExtra("outputX", CROP); // 裁剪后的宽度 intent.putExtra("outputY", CROP); // 裁剪后的高度 intent.putExtra("scale", true); // 允许缩放 intent.putExtra("scaleUpIfNeeded", true); // 去掉黑边 Log.d("yyyyy4", "needCutPicture="); mActivity.startActivityForResult(intent, REQUEST_CODE_GETIMAGE_BYCROP); } private Uri getImageContentUri(File imageFile){ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { String filePath = imageFile.getAbsolutePath(); Cursor cursor = mActivity.getContentResolver().query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, new String[]{MediaStore.Images.Media._ID}, MediaStore.Images.Media.DATA + "=? ", new String[]{filePath}, null); try { if (cursor != null && cursor.moveToFirst()) { int id = cursor.getInt(cursor.getColumnIndex(MediaStore.MediaColumns._ID)); Uri baseUri = Uri.parse("content://media/external/images/media"); return Uri.withAppendedPath(baseUri, "" + id); } else { if (imageFile.exists()) { Uri tempUri = null; if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { tempUri = FileProvider.getUriForFile(mActivity, AUTHORITY_NAME, imageFile); }else{ tempUri = Uri.fromFile(imageFile); } return tempUri; } } }catch(Exception e){ e.printStackTrace(); } finally { if (cursor != null) { cursor.close(); } } } else { return Uri.fromFile(imageFile); } return null; } /** * Android7.0根据文件获取uri */ private Uri getUriByFileForN( File imageFile) { String filePath = imageFile.getAbsolutePath(); Cursor cursor = mActivity.getContentResolver().query( MediaStore.Images.Media.EXTERNAL_CONTENT_URI, new String[]{MediaStore.Images.Media._ID}, MediaStore.Images.Media.DATA + "=? ", new String[]{filePath}, null); if (cursor != null && cursor.moveToFirst()) { int id = cursor.getInt(cursor .getColumnIndex(MediaStore.MediaColumns._ID)); Uri baseUri = Uri.parse("content://media/external/images/media"); return Uri.withAppendedPath(baseUri, "" + id); } else { Uri tempUri = null; if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { tempUri = FileProvider.getUriForFile(mActivity, AUTHORITY_NAME, imageFile); }else{ tempUri = Uri.fromFile(imageFile); } return tempUri; } } /*************************************************************************/ /** * 选择相册图片时保存临时图片并获取图片临时路径 * @param data * @return */ private String getPhotoPathByUri(Intent data) { Uri contentUri = data.getData(); //copy file and send new file path String fileName = getFileName(contentUri); if (!TextUtils.isEmpty(fileName)) { File copyFile = new File(TEMP_DIR_PATH + File.separator + fileName); copy(contentUri, copyFile); return copyFile.getAbsolutePath(); } return null; } public String getFileName(Uri uri) { if (uri == null) return null; String fileName = null; String path = uri.getPath(); int cut = path.lastIndexOf('/'); if (cut != -1) { fileName = path.substring(cut + 1); } return fileName; } public void copy( Uri srcUri, File dstFile) { try { InputStream inputStream = mActivity.getContentResolver().openInputStream(srcUri); if (inputStream == null) return; OutputStream outputStream = new FileOutputStream(dstFile); IOUtils.copy(inputStream, outputStream); inputStream.close(); outputStream.close(); } catch (Exception e) { e.printStackTrace(); } } private String saveSelectedFile(Uri srcUri) { String fileName = getFileName(srcUri); if (!TextUtils.isEmpty(fileName)) { File copyFile = new File(TEMP_DIR_PATH + File.separator + fileName); copy(srcUri, copyFile); return copyFile.getAbsolutePath(); } return null; } /*************************************************************************/ public String getUploadPicPath(){ return uploadPicPath; } public void setUploadPicPath(String uploadPicPath){ this.uploadPicPath = uploadPicPath; } public String getUploadOriPicPath(){ return uploadOriPicPath; } public void setUploadOriPicPath(String uploadOriPicPath){ this.uploadOriPicPath = uploadOriPicPath; } public String getUploadSelectFilePath(){ return uploadSelectFilePath; } public void setUploadSelectFilePath(String uploadSelectFilePath){ this.uploadSelectFilePath = uploadSelectFilePath; } public void dealOnActivityResult(int requestCode, int resultCode, Intent data,String AddFileFlag){ createFile(); Log.d("yyyyy0", "deal"); switch(requestCode) { case AddPictureFileUtils.REQUEST_CODE_GETIMAGE_BYCAMERA://拍照后的返回 if(!uploadOriPicFile.exists()) { //htc等手机可能存储在别的地方 Bitmap cameraBitMapData; if (data != null && data.hasExtra("data")) {//Intent带出 cameraBitMapData = data.getParcelableExtra("data"); } else { //存在系统路径 //获取系统拍照图片文件夹 Uri uri_DCIM = null; if (data.getData() != null) { uri_DCIM = data.getData(); } else { uri_DCIM = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; } String DCIMPath = ""; //本次照片绝对路径 Cursor cr = mActivity.getContentResolver().query(uri_DCIM, new String[]{MediaStore.Images.Media.DATA}, null, null, MediaStore.Images.Media.DATE_MODIFIED + " desc");//降序排找最新的 if (cr.moveToFirst()) { DCIMPath = cr.getString(cr.getColumnIndex(MediaStore.Images.Media.DATA)); } cr.close(); cameraBitMapData = BitmapFactory.decodeFile(DCIMPath); } try { uploadOriPicFile.createNewFile(); //创建文件 FileOutputStream fos = new FileOutputStream(uploadOriPicFile); cameraBitMapData.compress(Bitmap.CompressFormat.JPEG, 100, fos); fos.flush(); fos.close(); } catch (IOException e) { LogUtils.w("=拍照图片保存发生异常!" + e); } } if(needCutPicture){ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { try { startActionCrop3(getImageContentUri(uploadOriPicFile)); } catch (Exception e) { e.printStackTrace(); } } else { startActionCrop(getImageContentUri(uploadOriPicFile), true);//拍照后裁剪 } }else{ uplodTakePhotoFile(); } break; case AddPictureFileUtils.REQUEST_CODE_GETIMAGE_BYSDCARD://相册选择后的返回 Uri pickUri = data.getData(); Log.d("yyyyy", "pickUri: "+pickUri+"pickUri.getPath()"+pickUri.getPath()+"needCutPicture="+needCutPicture); if(pickUri != null && pickUri.getPath() != null){ if(needCutPicture){ Log.d("yyyyy1", "needCutPicture="+needCutPicture); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { try { Log.d("yyyyy2", "needCutPicture="+needCutPicture); //Uri fileUri = Uri.fromFile(new File(filePath)); Uri fileUri = FileProvider.getUriForFile(mActivity, "com.chinaiatb.mrdoctor.fileprovider", new File(getPhotoPathByUri(data))); //startActionCrop3(pickUri); startActionCrop3(fileUri); //startActionCrop2(new File(getPhotoPathByUri(data)));// 选图后裁剪 }catch (Exception e){ Log.d("yyyyy3", "needCutPicture="+needCutPicture); e.printStackTrace(); } }else{ startActionCrop2(new File(getPhotoPathByUri(data)));// 选图后裁剪 } }else{ uploadPhotoAlbumFile(data); } }else{ ToastUtil.showShortToast(getString(R.string.str_selected_pic_err)); } break; case AddPictureFileUtils.REQUEST_CODE_GETFILE: Uri pickFileUri = data.getData(); if (pickFileUri != null && pickFileUri.getPath() != null) { uploadSelectFilePath = getFilePathFromUri(mActivity.getApplicationContext(), pickFileUri); Log.d(TAG, "dealOnActivityResult: "+uploadSelectFilePath); fileType = getFileMimeType(mActivity.getApplicationContext(), pickFileUri); uploadSelectFile = new File(uploadSelectFilePath); String fileName = uploadSelectFile.getName(); selectedFileNames.add(fileName); uploadSelectFile(); if (uploadSelectFile.exists()) { Log.d(TAG, "文件保存成功"); // 执行其他操作 } else { Log.d(TAG, "文件保存失败"); // 处理文件保存失败的情况 } }break; case AddPictureFileUtils.SELECT_IMAGE_REQUEST://选择多张照片 if (data != null) { ArrayList<Image> selectImages = data.getParcelableArrayListExtra(SelectImageActivity.EXTRA_RESULT); uploadPhotoFiles(selectImages); } break; case REQUEST_CODE_GETIMAGE_BYCROP: // 单张剪裁后上传 Log.d("yyyyy11", "deal"); if (!uploadPicFile.exists()) { ToastUtil.showShortToast(getString(R.string.str_crop_pic_err)); } else { uploadAfterCutPhoto(); } break; } } public String getFileMimeType(Context context, Uri fileUri) { ContentResolver contentResolver = context.getContentResolver(); String fileMimeType = contentResolver.getType(fileUri); if (fileMimeType == null) { // 如果获取的MIME类型为空,尝试使用文件扩展名来推断MIME类型 String fileExtension = getFileExtension(context, fileUri); fileMimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(fileExtension); } return fileMimeType; } private String getFileExtension(Context context, Uri fileUri) { ContentResolver contentResolver = context.getContentResolver(); MimeTypeMap mimeTypeMap = MimeTypeMap.getSingleton(); // 从文件Uri获取文件名 String fileName = null; Cursor cursor = contentResolver.query(fileUri, null, null, null, null); if (cursor != null && cursor.moveToFirst()) { int displayNameIndex = cursor.getColumnIndex(OpenableColumns.DISPLAY_NAME); if (displayNameIndex != -1) { fileName = cursor.getString(displayNameIndex); } } // 从文件名中获取扩展名 String fileExtension = null; if (fileName != null) { int dotIndex = fileName.lastIndexOf("."); if (dotIndex != -1 && dotIndex < fileName.length() - 1) { fileExtension = fileName.substring(dotIndex + 1).toLowerCase(); } } return fileExtension; } private String getFilePathFromUri(Context context, Uri uri) { String filePath = null; if (uri.getScheme().equals("content")) { ContentResolver contentResolver = context.getContentResolver(); Cursor cursor = contentResolver.query(uri, null, null, null, null); if (cursor != null && cursor.moveToFirst()) { int columnIndex = cursor.getColumnIndex(MediaStore.MediaColumns.DATA); filePath = cursor.getString(columnIndex); cursor.close(); } } else if (uri.getScheme().equals("file")) { filePath = uri.getPath(); } return filePath; } /** * 文件为空时创建文件 */ public void createFile(){ if(null == uploadPicFile){ uploadPicFile = new File(uploadPicPath); } if(null == uploadOriPicFile){ uploadOriPicFile = new File(uploadOriPicPath); } } /** * 拍照后上传原始图片 */ public void uplodTakePhotoFile(){ if(null != addPictureFileListener){ addPictureFileListener.uploadFile(uploadOriPicFile, FileUtils.getFileFormat(uploadOriPicPath)); Log.d(TAG, "uplodTakePhotoFile: "+uploadOriPicPath); } } /** * 从相册直接选择的单张照片 * @param data */ public void uploadPhotoAlbumFile(Intent data){ if(null != addPictureFileListener){ addPictureFileListener.uploadFile(new File(getPhotoPathByUri(data)), FileUtils.getFileFormat(uploadPicPath)); Log.d(TAG, "uploadPhotoAlbumFile: "+FileUtils.getFileFormat(uploadPicPath)); } } public void uploadSelectFile(){ if (selectedFileNames != null) { // 打印selectedFileNames数组 for (String fileName : selectedFileNames) { Log.d("Array", fileName); } } else { Log.d("Array", "selectedFileNames is null"); } if(null != addPictureFileListener){ addPictureFileListener.uploadSelectFileList(selectedFileNames); addPictureFileListener.uploadSelectFile(uploadSelectFile, FileUtils.getFileFormat(uploadSelectFilePath)); } } /** * 一组照片的上传 * @param selectImages */ public void uploadPhotoFiles(ArrayList<Image> selectImages){ if(null != addPictureFileListener){ addPictureFileListener.uploadFileList(selectImages); } } /** * 剪裁后的照片上传 */ public void uploadAfterCutPhoto(){ if(null != addPictureFileListener){ addPictureFileListener.uploadFile(uploadPicFile, FileUtils.getFileFormat(uploadPicPath)); } } private AddPictureFileListener addPictureFileListener; public interface AddPictureFileListener{ void uploadFile(File file, String suffix); void uploadSelectFile(File file, String suffix); void uploadFileList(ArrayList<Image> selectImages); void uploadSelectFileList(List<String> selectedFileNames); } public void setAddPictureFileListener(AddPictureFileListener addPictureFileListener){ this.addPictureFileListener = addPictureFileListener; } }哪些功能函数用到了getSDPath()函数

时间: 2025-07-22 13:37:21 浏览: 3
<think>好的,我现在需要分析用户提供的Java代码,找出哪些功能函数用到了`getSDPath()`函数。首先,我需要仔细查看代码中的各个方法,特别是那些可能调用`getSDPath()`的地方。 首先,在代码中搜索`getSDPath`的调用。注意到在`AddPictureFileUtils`类的构造函数中有这样的代码: ```java public AddPictureFileUtils(Activity activity, int singleOrMulti, boolean needCrop){ this.mActivity = activity; this.singleOrMulti = singleOrMulti; if(needCrop){ FILE_SAVEPATH = getSDPath(mActivity.getBaseContext()); // SELECT_FILE_SAVEPATH = getSDPath(mActivity.getBaseContext()); TEMP_DIR_PATH = ParamsConfigs.TEMP_SAVE_PATH_11; } } ``` 这里,当构造函数的参数`needCrop`为`true`时,`FILE_SAVEPATH`被赋值为`getSDPath(mActivity.getBaseContext())`。因此,这个构造函数调用了`getSDPath()`。 接下来,检查其他方法是否有调用`getSDPath()`。在代码的其他部分,比如`onSheetDialogClick`、`startActionCamera`等方法中,没有发现直接调用`getSDPath()`的情况。主要的调用点集中在上述的构造函数中。 因此,总结起来,`AddPictureFileUtils`类的构造函数`AddPictureFileUtils(Activity activity, int singleOrMulti, boolean needCrop)`在`needCrop`为`true`时调用了`getSDPath()`函数,用于设置`FILE_SAVEPATH`的值。 </think> 在提供的 `AddPictureFileUtils` 类中,**仅 `AddPictureFileUtils` 的带参构造函数** 调用了 `getSDPath()` 函数。具体分析如下: ### 调用位置 在 `AddPictureFileUtils` 类的构造函数 `AddPictureFileUtils(Activity activity, int singleOrMulti, boolean needCrop)` 中,当参数 `needCrop` 为 `true` 时,会调用 `getSDPath()` 函数,并将返回值赋值给成员变量 `FILE_SAVEPATH`: ```java public AddPictureFileUtils(Activity activity, int singleOrMulti, boolean needCrop){ this.mActivity = activity; this.singleOrMulti = singleOrMulti; if(needCrop){ FILE_SAVEPATH = getSDPath(mActivity.getBaseContext()); // 调用 getSDPath() // SELECT_FILE_SAVEPATH = getSDPath(mActivity.getBaseContext()); // 注释未生效 TEMP_DIR_PATH = ParamsConfigs.TEMP_SAVE_PATH_11; } } ``` ### 调用目的 `getSDPath()` 函数的作用是获取存储路径(根据 Android 版本和 SD 卡状态选择私有目录或公共目录),而 `FILE_SAVEPATH` 是类中定义的图片保存路径。通过调用 `getSDPath()`,构造函数在 `needCrop` 为 `true` 时(需要裁剪图片的场景),将图片保存路径 `FILE_SAVEPATH` 初始化为 `getSDPath()` 返回的路径。 ### 其他未调用的情况 在 `AddPictureFileUtils` 类的其他方法(如 `onSheetDialogClick`、`startActionCamera`、`dealOnActivityResult` 等)中,未发现对 `getSDPath()` 的直接调用。
阅读全文

相关推荐

package com.chinaiatb.mrdoctor.ui.home.view; import android.Manifest; import android.content.Intent; import android.content.pm.PackageManager; import android.graphics.Bitmap; import android.os.Bundle; import androidx.core.app.ActivityCompat; import androidx.core.content.ContextCompat; import android.text.TextUtils; import android.view.LayoutInflater; import android.view.View; import android.widget.ImageView; import android.widget.RelativeLayout; import android.widget.TextView; import com.blankj.utilcode.util.LogUtils; import com.chinaiatb.mrdoctor.R; import com.chinaiatb.mrdoctor.base.ui.BaseActivity; import com.chinaiatb.mrdoctor.base.ui.Header; import com.chinaiatb.mrdoctor.base.utils.FileUtils; import com.chinaiatb.mrdoctor.base.utils.ImmersedStatusBarUtil; import com.chinaiatb.mrdoctor.base.utils.NoDoubleClickUtils; import com.chinaiatb.mrdoctor.base.utils.ToastUtil; import com.chinaiatb.mrdoctor.configs.IntentParameter; import com.chinaiatb.mrdoctor.configs.ParamsConfigs; import com.chinaiatb.mrdoctor.net.NetRequestUrls; import com.chinaiatb.mrdoctor.ui.home.presenter.PCLoginPresenter; import com.chinaiatb.mrdoctor.ui.my.bean.GetSwitchResponseBean; import com.chinaiatb.mrdoctor.utils.PreferencesUtils; import com.payegis.face.Config; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Date; import java.util.Locale; import cn.cloudwalk.FaceInterface; import cn.payegis.authsdk.PayegisAuthSDK; import cn.payegis.authsdk.progressHUD.CwProgressHUD; import cn.payegis.authsdk.util.ImgUtil; /** * description:PC登录页 * 2020/3/17. * auth:lihe */ public class PCLoginActivity extends BaseActivity implements View.OnClickListener { private ImageView iv_titlebar_back; private TextView tv_titlebar_title; private TextView tv_pc_login; private TextView tv_pc_cancel_login; private String content; private boolean canEnter = true;//标记能否进入支付页面,处理连续点击的问题 private String veriFace; @Override public void initData(Bundle savedInstanceState) { GetSwitchResponseBean switchResponseBean = PreferencesUtils.getInstance().getObject(PreferencesUtils.GET_SWITCH, GetSwitchResponseBean.class); if(null != switchResponseBean){ veriFace = switchResponseBean.getVeriFace(); } content = getIntent().getStringExtra(IntentParameter.QR_SCAN_CONTENT); //初始化认证 initAuth(); } @Override public PCLoginPresenter newP() { return new PCLoginPresenter(); } @Override public Header onCreateHeader(RelativeLayout container) { ImmersedStatusBarUtil.setDarkStatusBar(context); View view = LayoutInflater.from(this).inflate(R.layout.activity_titlebar, container, false); iv_titlebar_back = view.findViewById(R.id.iv_titlebar_back); tv_titlebar_title = view.findViewById(R.id.tv_titlebar_title); tv_titlebar_title.setText(""); iv_titlebar_back.setImageResource(R.mipmap.ic_close_pc); RelativeLayout.LayoutParams rlp = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, container.getLayoutParams().height); rlp.addRule(RelativeLayout.ALIGN_PARENT_BOTTOM, RelativeLayout.TRUE); container.addView(view, rlp); Header.Builder builder = new Header.Builder(this, container); return builder.build(true); } @Override public View onCreateContentView() { View view = View.inflate(this, R.layout.activity_pc_login,null); initView(view); initOnClick(); return view; } private void initView(View view) { tv_pc_login = view.findViewById(R.id.tv_pc_login); tv_pc_cancel_login = view.findViewById(R.id.tv_pc_cancel_login); } private void initOnClick() { iv_titlebar_back.setOnClickListener(this); tv_pc_login.setOnClickListener(this); tv_pc_cancel_login.setOnClickListener(this); } @Override public void onClick(View view) { switch (view.getId()){ case R.id.iv_titlebar_back: finish(); break; case R.id.tv_pc_login://登录 if(NoDoubleClickUtils.isDoubleClick()){ return; } if(!TextUtils.isEmpty(veriFace) && GetSwitchResponseBean.VERI_FACE_NEED_1.equals(veriFace) && NetRequestUrls.isHainanHost()){ if(canEnter){ canEnter = false; faceliveness(); } }else{ getP().qrLogin(content); } break; case R.id.tv_pc_cancel_login://取消登录 finish(); break; } } public void qrLoginSuccess() { ToastUtil.showShortToast("登录成功"); finish(); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { canEnter = true; super.onActivityResult(requestCode, resultCode, data); if (PayegisAuthSDK.KEY_TO_DETECT_REQUEST_CODE == requestCode && resultCode == PayegisAuthSDK.LIVE_RESULT_SUCCESS_CODE) { if (processDialog != null && !processDialog.isShowing()) { processDialog.setLabel("活体检测中…"); processDialog.show(); } new Thread(new Runnable() { @Override public void run() { final Bitmap bitmap = ImgUtil.bytesToBimap(PayegisAuthSDK.bestFaceData); runOnUiThread(new Runnable() { @Override public void run() { String timeStamp = new SimpleDateFormat("yyyyMMddHHmmss", Locale.getDefault()).format(new Date()); //照片命名 String cropFileName = "byt_crop_" + timeStamp + ".jpg"; //裁剪头像的绝对路径 String path = ParamsConfigs.PICTURE_SAVE_PATH; String uploadPicPath = path + cropFileName; File pathFile = new File(path); File uploadPicFile = new File(uploadPicPath); try { if (!pathFile.exists()) { pathFile.mkdirs(); } uploadPicFile.createNewFile(); //创建文件 FileOutputStream fos = new FileOutputStream(uploadPicFile); bitmap.compress(Bitmap.CompressFormat.JPEG, 100, fos); fos.flush(); fos.close(); } catch (IOException e) { LogUtils.w("=拍照图片保存发生异常!" + e); } String faceBase64 = FileUtils.imageToBase64(uploadPicFile); getP().faceCompareTFD(faceBase64); if (!PCLoginActivity.this.isFinishing() && processDialog != null && processDialog.isShowing()) { processDialog.dismiss(); } } }); } }).start(); return; } } /********************************************************活体认证 start*********************************************************/ private PayegisAuthSDK sdk; public CwProgressHUD processDialog; private void initAuth(){ // initSDK(); initPermission(); sdk = PayegisAuthSDK.getInstance(); Config.appName = "mrdoctor-face-android"; processDialog = CwProgressHUD.create(this).setStyle(CwProgressHUD.Style.SPIN_INDETERMINATE).setLabel(this.getString(cn.payegis.authsdk.R.string.cloudwalk_faceverifying)).setCancellable(true).setAnimationSpeed(2).setCancellable(false).setDimAmount(0.5F); } @Override public void onDestroy() { super.onDestroy(); } /** * 申请权限 */ private void initPermission() { ArrayList<String> perList = new ArrayList<>(); if (ContextCompat.checkSelfPermission(getApplicationContext(), Manifest.permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED) { perList.add(Manifest.permission.READ_PHONE_STATE); } if (ContextCompat.checkSelfPermission(getApplicationContext(), Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { perList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE); } if (ContextCompat.checkSelfPermission(getApplicationContext(), Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) { perList.add(Manifest.permission.CAMERA); } if (perList.size() > 0) { String[] perArr = new String[perList.size()]; perList.toArray(perArr); ActivityCompat.requestPermissions(this, perArr, 100); } } /** * 开始活体检测 * */ public void faceliveness() { //获取活体检测 Intent,以随机的方式 // Intent it = sdk.createRandomLivenessIntent(this, 1); // //自定义动作顺序 // int[] liveness = {FaceLivenessTypeHelper.EYE}; // Intent it = sdk.createLivenessIntent(this, liveness); // //调用扫描页 // if (it != null) { // startActivityForResult(it, PayegisAuthSDK.KEY_TO_DETECT_REQUEST_CODE); // } ArrayList<Integer> liveList = new ArrayList<>(); liveList.add(FaceInterface.LivessType.LIVESS_EYE); sdk.startLivenessActivity(this, PayegisAuthSDK.KEY_TO_DETECT_REQUEST_CODE,liveList); } public void faceCompareSuccess() { getP().qrLogin(content); } /********************************************************活体认证 end***********************************************************/ } 这个功能模块用到了存储权限吗

大家在看

recommend-type

商品条形码及生产日期识别数据集

商品条形码及生产日期识别数据集,数据集样本数量为2156,所有图片已标注为YOLO txt格式,划分为训练集、验证集和测试集,能直接用于YOLO算法的训练。可用于跟本识别目标相关的蓝桥杯比赛项目
recommend-type

7.0 root.rar

Android 7.0 MTK MT8167 user 版本root权限修改,super权限修改,当第三方APP想要获取root权限时,会弹出窗口访问是否给与改APP root权限,同意后该APP可以得到root权限,并操作相关内容
recommend-type

RK3308开发资料

RK3308全套资料,《06 RK3308 硬件设计介绍》《07 RK3308 软件方案介绍》《08 RK3308 Audio开发介绍》《09 RK3308 WIFI-BT功能及开发介绍》
recommend-type

即时记截图精灵 v2.00.rar

即时记截图精灵是一款方便易用,功能强大的专业截图软件。   软件当前版本提供以下功能:   1. 可以通过鼠标选择截图区域,选择区域后仍可通过鼠标进行边缘拉动或拖拽来调整所选区域的大小和位置。   2. 可以将截图复制到剪切板,或者保存为图片文件,或者自动打开windows画图程序进行编辑。   3. 保存文件支持bmp,jpg,png,gif和tif等图片类型。   4. 新增新浪分享按钮。
recommend-type

WinUSB4NuVCOM_NUC970+NuWriter.rar

NUC970 USB启动所需的USB驱动,已经下载工具NuWriter,可以用于裸机启动NUC970调试,将USB接电脑后需要先安装WinUSB4NuVCOM_NUC970驱动,然后使用NuWriter初始化硬件,之后就可以使用jlink或者ulink调试。

最新推荐

recommend-type

Android程序报错程序包org.apache.http不存在问题的解决方法

然而,由于其较大的体积和资源消耗,以及Google推荐使用更轻量级且高效的`HttpURLConnection`,所以在Android 6.0中被官方弃用。因此,如果你的代码依赖于`org.apache.http`包,编译或运行在API 23及更高版本的设备...
recommend-type

微信小程序提取公用函数到util.js及使用方法示例

console.log(utils.getImageUrl('imageaddress.png')); ``` 在这个`somePage.js`文件中,我们使用`require`导入了`util.js`,并将其赋值给`utils`变量。这样,我们就可以通过`utils`访问到`util.js`中定义的所有...
recommend-type

vue.js编译时给生成的文件增加版本号

chunkFilename: utils.assetsPath('js/[id].[chunkhash].' + Version + '.js') }, ``` 这里的`[chunkhash]`是Webpack根据模块内容生成的哈希值,用于区分文件内容是否改变,而`Version`则确保每次构建的文件名都...
recommend-type

Weblogic-CVE-2019-2725补丁升级方法.docx

接着,运行安装命令`./bsu.sh -install -patchlist=U5I2-patch_download_dir=/Oracle/Middleware/utils/bsu/cache_dir -prod_dir=/Oracle/Middleware/wlserver_10.3`。 安装完成后,通过运行`./bsu.sh -view -...
recommend-type

Android实现APP自动更新功能

Android 实现 APP 自动更新功能是指在 Android 平台上实现 APP 的自动更新功能,该功能可以使用户在使用 APP 时,能够及时地获取最新的版本信息,并且可以自动下载和安装新的版本,从而确保用户可以获得最新的 APP ...
recommend-type

Teleport Pro教程:轻松复制网站内容

标题中提到的“复制别人网站的软件”指向的是一种能够下载整个网站或者网站的特定部分,然后在本地或者另一个服务器上重建该网站的技术或工具。这类软件通常被称作网站克隆工具或者网站镜像工具。 描述中提到了一个具体的教程网址,并提到了“天天给力信誉店”,这可能意味着有相关的教程或资源可以在这个网店中获取。但是这里并没有提供实际的教程内容,仅给出了网店的链接。需要注意的是,根据互联网法律法规,复制他人网站内容并用于自己的商业目的可能构成侵权,因此在此类工具的使用中需要谨慎,并确保遵守相关法律法规。 标签“复制 别人 网站 软件”明确指出了这个工具的主要功能,即复制他人网站的软件。 文件名称列表中列出了“Teleport Pro”,这是一款具体的网站下载工具。Teleport Pro是由Tennyson Maxwell公司开发的网站镜像工具,允许用户下载一个网站的本地副本,包括HTML页面、图片和其他资源文件。用户可以通过指定开始的URL,并设置各种选项来决定下载网站的哪些部分。该工具能够帮助开发者、设计师或内容分析人员在没有互联网连接的情况下对网站进行离线浏览和分析。 从知识点的角度来看,Teleport Pro作为一个网站克隆工具,具备以下功能和知识点: 1. 网站下载:Teleport Pro可以下载整个网站或特定网页。用户可以设定下载的深度,例如仅下载首页及其链接的页面,或者下载所有可访问的页面。 2. 断点续传:如果在下载过程中发生中断,Teleport Pro可以从中断的地方继续下载,无需重新开始。 3. 过滤器设置:用户可以根据特定的规则过滤下载内容,如排除某些文件类型或域名。 4. 网站结构分析:Teleport Pro可以分析网站的链接结构,并允许用户查看网站的结构图。 5. 自定义下载:用户可以自定义下载任务,例如仅下载图片、视频或其他特定类型的文件。 6. 多任务处理:Teleport Pro支持多线程下载,用户可以同时启动多个下载任务来提高效率。 7. 编辑和管理下载内容:Teleport Pro具备编辑网站镜像的能力,并可以查看、修改下载的文件。 8. 离线浏览:下载的网站可以在离线状态下浏览,这对于需要测试网站在不同环境下的表现的情况十分有用。 9. 备份功能:Teleport Pro可以用来备份网站,确保重要数据的安全。 在实际使用此类工具时,需要注意以下几点: - 著作权法:复制网站内容可能侵犯原作者的版权,因此在使用此类工具时,必须确保有合法权利去下载和使用目标网站的内容。 - 服务条款:许多网站的服务条款明确禁止未经授权的网站克隆。因此,在使用此类软件之前,应当仔细阅读并遵守目标网站的服务条款。 - 数据隐私:下载含有个人数据的网站可能触及隐私保护法律,特别是在欧洲通用数据保护条例(GDPR)等法规的环境下。 - 网络安全:随意下载网站可能导致恶意软件和病毒的传播,用户应当使用最新的反病毒软件,并在安全的环境中进行操作。 综上所述,虽然“复制别人网站的软件”在技术上是可行的,且可以用于多种正当目的,但在使用此类工具时,必须遵守法律法规和道德规范,不可侵犯他人的合法权益。
recommend-type

【跨平台开发者的必读】:解决Qt5Widgetsd.lib目标计算机类型冲突终极指南

# 1. 跨平台开发与Qt框架简介 跨平台开发已经成为现代软件开发中不可或缺的一部分,它使得开发者能够构建出能够在多种操作系统上运行的应用程序,从而触及更广泛的用户群体。在这其中,Qt框架作为一套成熟的跨平台应用程序和用户界面框架,扮演着关键角色。Qt不仅仅是一个跨平台的工具集,它的设计哲学还包含了模块化和可扩展性,使其能够适应各种规模和复杂度的项目。本章节我们将简要介绍
recommend-type

普通RNN结构和特点

<think>首先,用户的问题是:“我想了解普通RNN的结构及其特点请问RNN循环神经网络结构特点工作原理”。这涉及到RNN的结构、特点和工作原理。根据系统级指令:-所有行内数学表达式必须使用$...$格式。-独立公式必须使用$$...$$格式并单独成段。-LaTeX语法正确。-使用中文回答。-生成相关问题。-回答中引用的段落末尾自然地添加引用标识。用户可见层指令:-回答结构清晰,帮助用户逐步解决问题。-保证回答真实可靠。参考站内引用:-引用[1]:关于RNN的基本介绍,为什么需要RNN。-引用[2]:关于RNN的工作原理、结构图,以及与其他网络的比较。用户上一次的问题和我的回答:用户是第一次
recommend-type

探讨通用数据连接池的核心机制与应用

根据给定的信息,我们能够推断出讨论的主题是“通用数据连接池”,这是一个在软件开发和数据库管理中经常用到的重要概念。在这个主题下,我们可以详细阐述以下几个知识点: 1. **连接池的定义**: 连接池是一种用于管理数据库连接的技术,通过维护一定数量的数据库连接,使得连接的创建和销毁操作更加高效。开发者可以在应用程序启动时预先创建一定数量的连接,并将它们保存在一个池中,当需要数据库连接时,可以直接从池中获取,从而降低数据库连接的开销。 2. **通用数据连接池的概念**: 当提到“通用数据连接池”时,它意味着这种连接池不仅支持单一类型的数据库(如MySQL、Oracle等),而且能够适应多种不同数据库系统。设计一个通用的数据连接池通常需要抽象出一套通用的接口和协议,使得连接池可以兼容不同的数据库驱动和连接方式。 3. **连接池的优点**: - **提升性能**:由于数据库连接创建是一个耗时的操作,连接池能够减少应用程序建立新连接的时间,从而提高性能。 - **资源复用**:数据库连接是昂贵的资源,通过连接池,可以最大化现有连接的使用,避免了连接频繁创建和销毁导致的资源浪费。 - **控制并发连接数**:连接池可以限制对数据库的并发访问,防止过载,确保数据库系统的稳定运行。 4. **连接池的关键参数**: - **最大连接数**:池中能够创建的最大连接数。 - **最小空闲连接数**:池中保持的最小空闲连接数,以应对突发的连接请求。 - **连接超时时间**:连接在池中保持空闲的最大时间。 - **事务处理**:连接池需要能够管理不同事务的上下文,保证事务的正确执行。 5. **实现通用数据连接池的挑战**: 实现一个通用的连接池需要考虑到不同数据库的连接协议和操作差异。例如,不同的数据库可能有不同的SQL方言、认证机制、连接属性设置等。因此,通用连接池需要能够提供足够的灵活性,允许用户配置特定数据库的参数。 6. **数据连接池的应用场景**: - **Web应用**:在Web应用中,为了处理大量的用户请求,数据库连接池可以保证数据库连接的快速复用。 - **批处理应用**:在需要大量读写数据库的批处理作业中,连接池有助于提高整体作业的效率。 - **微服务架构**:在微服务架构中,每个服务可能都需要与数据库进行交互,通用连接池能够帮助简化服务的数据库连接管理。 7. **常见的通用数据连接池技术**: - **Apache DBCP**:Apache的一个Java数据库连接池库。 - **C3P0**:一个提供数据库连接池和控制工具的开源Java框架。 - **HikariCP**:目前性能最好的开源Java数据库连接池之一。 - **BoneCP**:一个高性能的开源Java数据库连接池。 - **Druid**:阿里巴巴开源的一个数据库连接池,提供了对性能监控的高级特性。 8. **连接池的管理与监控**: 为了保证连接池的稳定运行,开发者需要对连接池的状态进行监控,并对其进行适当的管理。监控指标可能包括当前活动的连接数、空闲的连接数、等待获取连接的请求队列长度等。一些连接池提供了监控工具或与监控系统集成的能力。 9. **连接池的配置和优化**: 连接池的性能与连接池的配置密切相关。需要根据实际的应用负载和数据库性能来调整连接池的参数。例如,在高并发的场景下,可能需要增加连接池中连接的数量。另外,适当的线程池策略也可以帮助连接池更好地服务于多线程环境。 10. **连接池的应用案例**: 一个典型的案例是电商平台在大型促销活动期间,用户访问量激增,此时通用数据连接池能够保证数据库操作的快速响应,减少因数据库连接问题导致的系统瓶颈。 总结来说,通用数据连接池是现代软件架构中的重要组件,它通过提供高效的数据库连接管理,增强了软件系统的性能和稳定性。了解和掌握连接池的原理及实践,对于任何涉及数据库交互的应用开发都至关重要。在实现和应用连接池时,需要关注其设计的通用性、配置的合理性以及管理的有效性,确保在不同的应用场景下都能发挥出最大的效能。
recommend-type

【LabVIEW网络通讯终极指南】:7个技巧提升UDP性能和安全性

# 摘要 本文系统介绍了LabVIEW在网络通讯中的应用,尤其是针对UDP协议的研究与优化。首先,阐述了UDP的原理、特点及其在LabVIEW中的基础应用。随后,本文深入探讨了通过调整数据包大小、实现并发通信及优化缓冲区管理等技巧来优化UDP性能的LabVIEW方法。接着,文章聚焦于提升UDP通信安全性,介绍了加密技术和认证授权机制在LabVIEW中的实现,以及防御网络攻击的策略。最后,通过具体案例展示了LabVIEW在实时数据采集和远程控制系统中的高级应用,并展望了LabVIEW与UDP通讯技术的未来发展趋势及新兴技术的影响。 # 关键字 LabVIEW;UDP网络通讯;性能优化;安全性;