博主前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住也分享一下给大家,
点击跳转到网站
一、首先在AndroidManifest.xml中加入权限
<!--读取存储卡中的内容 修改或删除存储卡中的内容-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
因为这属于危险权限,需要在java代码中动态申请。
二、activity_save_image.xml 布局一个要保存的图片,一个保存按钮
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".SaveImageActivity">
<ImageView
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:scaleType="fitCenter"
android:src="@drawable/image1" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="60dp"
android:gravity="center">
<Button
android:id="@+id/btn_saveImage"
android:layout_width="150dp"
android:layout_height="50dp"
android:text="保存图片到相册"
android:textColor="@color/black" />
</LinearLayout>
</LinearLayout>
三、SaveImageActivity中我会把Android系统10之前的写法,和最新的写法都写出来,供大家参考学习。
public class SaveImageActivity extends AppCompatActivity implements View.OnClickListener {
private Button btn_saveImage;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_save_image);
btn_saveImage = findViewById(R.id.btn_saveImage);
btn_saveImage.setOnClickListener(this);
}
@Override
public void onClick(View v) {
if (v.getId() == R.id.btn_saveImage) {//对于共享区间写入的权限,在API29 Android系统10之前是需要申请的
//在API29及之后是不需要申请的,默认是允许的
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q && ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 0);
} else {
//保存图片到相册
SaveImage();
}
}
}
//请求权限后的结果回调
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == 0) {
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
//保存图片到相册
SaveImage();
} else {
Toast.makeText(this, "你拒绝了该权限,无法保存图片!", Toast.LENGTH_SHORT).show();
}
}
}
private void SaveImage() {
//获取要保存的图片的位图
// Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image1);
//API 29之前可用. API29之后该方法已经被弃用了
//MediaStore 相当于管理媒体资源的一个管理器,类似于一个数据库,对媒体资源的一个索引(包括图片 音频 视频),在里面都有索引
// if (MediaStore.Images.Media.insertImage(getContentResolver(), bitmap, "", "") == null) {
// Toast.makeText(this, "保存失败!", Toast.LENGTH_SHORT).show();
// } else {
// Toast.makeText(this, "保存成功!", Toast.LENGTH_SHORT).show();
// }
//创建一个子线程,将耗时任务在子线程中完成,防止主线程被阻塞
new Thread(new Runnable() {
@Override
public void run() {
try {
//获取要保存的图片的位图
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image1);
//创建一个保存的Uri
Uri saveUri = getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, new ContentValues());
if (TextUtils.isEmpty(saveUri.toString())) {
Looper.prepare();
Toast.makeText(SaveImageActivity.this, "保存失败!", Toast.LENGTH_SHORT).show();
Looper.loop();
return;
}
OutputStream outputStream = getContentResolver().openOutputStream(saveUri);
//将位图写出到指定的位置
//第一个参数:格式JPEG 是可以压缩的一个格式 PNG 是一个无损的格式
//第二个参数:保留原图像90%的品质,压缩10% 这里压缩的是存储大小
//第三个参数:具体的输出流
if (bitmap.compress(Bitmap.CompressFormat.JPEG, 90, outputStream)) {
Looper.prepare();
Toast.makeText(SaveImageActivity.this, "保存成功!", Toast.LENGTH_SHORT).show();
Looper.loop();
} else {
Looper.prepare();
Toast.makeText(SaveImageActivity.this, "保存失败!", Toast.LENGTH_SHORT).show();
Looper.loop();
}
} catch (Exception e) {
e.printStackTrace();
}
}
}).start();
}
}
通知图库进行刷新的方法
Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
Uri uri = Uri.fromFile(file);
intent.setData(uri);
sendBroadcast(intent);
博主已测试该功能,在Android 10 和Android 9的手机上面均能正常保存图片~ 如有不当之处,还望指正出来!