Android中图片剪裁时如何预防oom
来源:互联网 发布:千兆网络测速工具 编辑:程序博客网 时间:2024/06/05 10:55
http://my.oschina.net/ryanhoo/blog/86843
译者:Ryan Hoo
来源:http://www.androidworks.com/crop_large_photos_with_android
译者按:在外企工作的半年多中花了不少时间在国外的网站上搜寻资料,其中有一些相当有含金量的文章,我会陆陆续续翻译成中文,与大家共享之。初次翻译,“信达雅”三境界恐怕只到信的层次,望大家见谅!
这篇文章相当经典而实用,想当初我做手机拍照截图的时候,大多都是在网上抄来抄去的内容,从来没有人考虑过实际项目中的需求。实际上,拍照传大图片,如果用普通方式会耗用极大的内存,Android一个App原则上的16M内存限制可以一下子被耗光。Android在拍照上有一个隐藏的设计,如果拍照图片过大,只返回一张缩略图。具体到不同手机,都是不一样的。
-------------------------------------------------------------------------------------
译文:
概述
我写这篇文章是为了发表我对MediaStore裁剪图片功能的一些简要研究。基本上,如果你要写一个应用程序,使用已有的Media Gallery并允许用户在你的应用里选取TA的图片的一部分(可选功能:人脸识别)。 可以使用一个Intent做到这个,但是也存在着相应的问题,总的来说也缺少这方面的文档告诉我们怎么实现它。 另外,这篇文章基于2.1并且在Nexus One上做了测试。 Nexus One上的实现似乎被这群人写在了这里:Media Gellery for Nexus One 。反馈
这篇文章需要使用基于我的研究所写的程序。如果你对我推荐的实现方案有所改进,请让我知道。我会相应的更新这篇文章。
Intent细节
首先,让我们探讨下Intent以及它的特点。在看了一些代码示例以后,我发现我可以很轻松的使用如下的Intent调用裁剪功能:Intent intent =
new
Intent(Intent.ACTION_GET_CONTENT,
null
);
intent.setType(“image/*”);
intent.putExtra(“crop”, “
true
”);
…
你可以在你的程序中使用使用我的代码,并且扩展它。我会将之附加在这篇文章上。
Exta Options Table for image/* crop:附加选项数据类型描述cropString发送裁剪信号aspectXintX方向上的比例aspectYintY方向上的比例outputXint裁剪区的宽outputYint裁剪区的高scaleboolean是否保留比例return-databoolean是否将数据保留在Bitmap中返回dataParcelable相应的Bitmap数据circleCropString圆形裁剪区域?MediaStore.EXTRA_OUTPUT ("output")URI将URI指向相应的file:///...,详见代码示例
现在,最令人困惑的是MediaStore.EXTRA_OUTPUT以及return-data选项。
你主要有两种方式从这个Intent中取得返回的bitmap:获取内部数据或者提供一个Uri以便程序可以将数据写入。
方法1:如果你将return-data设置为“true”,你将会获得一个与内部数据关联的Action,并且bitmap以此方式返回:(Bitmap)extras.getParcelable("data")。注意:如果你最终要获取的图片非常大,那么此方法会给你带来麻烦,所以你要控制outputX和outputY保持在较小的尺寸。鉴于此原因,在我的代码中没有使用此方法((Bitmap)extras.getParcelable("data"))。
下面是CropImage.java的源码片段:
// Return the cropped image directly or save it to the specified URI.
Bundle myExtras = getIntent().getExtras();
if
(myExtras !=
null
&& (myExtras.getParcelable(
"data"
) !=
null
|| myExtras.getBoolean(
"return-data"
)))
{
Bundle extras =
new
Bundle();
extras.putParcelable(
"data"
, croppedImage);
setResult(RESULT_OK,(
new
Intent()).setAction(
"inline-data"
).putExtras(extras));
finish();
}
但是还有一些条件,首先你需要有一个短暂的与此Uri相关联的文件地址,当然这不是个大问题(除非是那些没有sdcard的设备)。
下面是CropImage.java关于操作Uri的源码片段:
if
(mSaveUri !=
null
) {
OutputStream outputStream =
null
;
try
{
outputStream = mContentResolver.openOutputStream(mSaveUri);
if
(outputStream !=
null
) {
croppedImage.compress(mOutputFormat,
75
, outputStream);
}
}
catch
(IOException ex) {
// TODO: report error to caller
Log.e(TAG,
"Cannot open file: "
+ mSaveUri, ex);
}
finally
{
Util.closeSilently(outputStream);
}
Bundle extras =
new
Bundle();
setResult(RESULT_OK,
new
Intent(mSaveUri.toString()).putExtras(extras));
}
代码示例:
我已经附上了一些代码示例,应该可以让你测试多种配置。请让我知道它对你是否有用。
/** Called when the activity is first created. */
@Override
public
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
thiz =
this
;
setContentView(R.layout.main);
mBtn = (Button) findViewById(R.id.btnLaunch);
photo = (ImageView) findViewById(R.id.imgPhoto);
mBtn.setOnClickListener(
new
OnClickListener() {
public
void
onClick(View v) {
try
{
// Launch picker to choose photo for selected contact
Intent intent =
new
Intent(Intent.ACTION_GET_CONTENT,
null
);
intent.setType(
"image/*"
);
intent.putExtra(
"crop"
,
"true"
);
intent.putExtra(
"aspectX"
, aspectX);
intent.putExtra(
"aspectY"
, aspectY);
intent.putExtra(
"outputX"
, outputX);
intent.putExtra(
"outputY"
, outputY);
intent.putExtra(
"scale"
, scale);
intent.putExtra(
"return-data"
, return_data);
intent.putExtra(MediaStore.EXTRA_OUTPUT, getTempUri());
intent.putExtra(
"outputFormat"
,
Bitmap.CompressFormat.JPEG.toString()); <span style=
"color:#48465A;font-family:monospace;font-size:11px;line-height:normal;background-color:#EFEFEF;"
>
// lol, negative boolean noFaceDetection</span> intent.putExtra("noFaceDetection", !faceDetection);
if
(circleCrop) {
intent.putExtra(
"circleCrop"
,
true
);
}
startActivityForResult(intent, PHOTO_PICKED);
}
catch
(ActivityNotFoundException e) {
Toast.makeText(thiz, R.string.photoPickerNotFoundText,
Toast.LENGTH_LONG).show();
}
}
});
}
private
Uri getTempUri() {
return
Uri.fromFile(getTempFile());
}
private
File getTempFile() {
if
(isSDCARDMounted()) {
File f =
new
File(Environment.getExternalStorageDirectory(),
TEMP_PHOTO_FILE);
try
{
f.createNewFile();
}
catch
(IOException e) {
// TODO Auto-generated catch block
Toast.makeText(thiz, R.string.fileIOIssue, Toast.LENGTH_LONG)
.show();
}
return
f;
}
else
{
return
null
;
}
}
private
boolean
isSDCARDMounted() {
String status = Environment.getExternalStorageState();
if
(status.equals(Environment.MEDIA_MOUNTED))
return
true
;
return
false
;
}
@Override
protected
void
onActivityResult(
int
requestCode,
int
resultCode, Intent data) {
super
.onActivityResult(requestCode, resultCode, data);
switch
(requestCode) {
case
PHOTO_PICKED:
if
(resultCode == RESULT_OK) {
if
(data ==
null
) {
Log.w(TAG,
"Null data, but RESULT_OK, from image picker!"
);
Toast t = Toast.makeText(
this
, R.string.no_photo_picked,
Toast.LENGTH_SHORT);
t.show();
return
;
}
final
Bundle extras = data.getExtras();
if
(extras !=
null
) {
File tempFile = getTempFile();
// new logic to get the photo from a URI
if
(data.getAction() !=
null
) {
processPhotoUpdate(tempFile);
}
}
}
break
;
}
}
- Android中图片剪裁时如何预防oom
- android中加载图片时出现oom
- 如何在Java中进行图片剪裁
- 如何在Java中进行图片剪裁
- android 图片剪裁
- Android 剪裁图片
- 图片剪裁 Android
- android 系统图片剪裁
- android 剪裁图片
- SimpleCropView Android图片剪裁
- android 自定义图片剪裁
- Android剪裁图片
- Android照片墙应用实现预防OOM
- android中图片内存管理-防止OOM
- android编程之图片剪裁
- Android图片剪裁的实现
- Android图片的剪裁处理
- Android图片剪裁库:uCrop
- bzoj3514 GERALD07
- android骚年的年终总结和规划
- Android ProgressBar 样式【大全】
- sequence training
- eclipse版本
- Android中图片剪裁时如何预防oom
- android Button源码分析
- JavaMail邮件传输
- Android ListView嵌套Button,Button事件覆盖item事件解决办法
- 学习理论之经验风险最小化——Andrew Ng机器学习笔记(七)
- 小米抢购限流峰值系统架构历年演进历程
- python代码片段——不断尝试执行cmd
- cout输出字符串地址 http://cache.baiducontent.com/c?m=9f65cb4a8c8507ed4fece7631049c0666e04de3e69c0d0622e95
- Android 对话框(Dialog)【大全】