Repository: guizhigang/LGImageCompressor Branch: master Commit: 3080b2d1cb42 Files: 37 Total size: 54.1 KB Directory structure: gitextract_la4y5atw/ ├── .gitignore ├── .idea/ │ ├── compiler.xml │ ├── copyright/ │ │ └── profiles_settings.xml │ ├── gradle.xml │ └── runConfigurations.xml ├── README.md ├── app/ │ ├── .gitignore │ ├── build.gradle │ ├── proguard-rules.pro │ └── src/ │ ├── androidTest/ │ │ └── java/ │ │ └── gui/ │ │ └── com/ │ │ └── lgimagecompressor/ │ │ └── ApplicationTest.java │ ├── main/ │ │ ├── AndroidManifest.xml │ │ ├── java/ │ │ │ └── gui/ │ │ │ └── com/ │ │ │ └── lgimagecompressor/ │ │ │ ├── BasicCompressActivity.java │ │ │ ├── CompressServiceListener.java │ │ │ ├── CompressServiceParam.java │ │ │ ├── Constanse.java │ │ │ ├── LGImgCompressor.java │ │ │ ├── LGImgCompressorIntentService.java │ │ │ ├── LGImgCompressorService.java │ │ │ ├── MainActivity.java │ │ │ └── ServiceCompressActivity.java │ │ └── res/ │ │ ├── layout/ │ │ │ ├── activity_basic_compress.xml │ │ │ ├── activity_intent_service_compress.xml │ │ │ └── activity_main.xml │ │ ├── values/ │ │ │ ├── colors.xml │ │ │ ├── dimens.xml │ │ │ ├── strings.xml │ │ │ └── styles.xml │ │ └── values-w820dp/ │ │ └── dimens.xml │ └── test/ │ └── java/ │ └── gui/ │ └── com/ │ └── lgimagecompressor/ │ ├── ExampleUnitTest.java │ └── LGImgCompressorServiceTest.java ├── build.gradle ├── gradle/ │ └── wrapper/ │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradle.properties ├── gradlew ├── gradlew.bat └── settings.gradle ================================================ FILE CONTENTS ================================================ ================================================ FILE: .gitignore ================================================ # OSX *.DS_Store # Gradle files build/ .gradle/ */build/ # IDEA *.iml .idea/.name .idea/encodings.xml .idea/inspectionProfiles/Project_Default.xml .idea/inspectionProfiles/profiles_settings.xml .idea/misc.xml .idea/modules.xml .idea/scopes/scope_settings.xml .idea/vcs.xml .idea/workspace.xml .idea/libraries # Built application files *.apk *.ap_ # Files for the Dalvik VM *.dex # Java class files *.class # Local configuration file (sdk path, etc) local.properties # Log Files *.log ================================================ FILE: .idea/compiler.xml ================================================ ================================================ FILE: .idea/copyright/profiles_settings.xml ================================================ ================================================ FILE: .idea/gradle.xml ================================================ ================================================ FILE: .idea/runConfigurations.xml ================================================ ================================================ FILE: README.md ================================================ # LGImageCompressor android图片压缩的处理 ================================================ FILE: app/.gitignore ================================================ /build ================================================ FILE: app/build.gradle ================================================ apply plugin: 'com.android.application' android { compileSdkVersion 23 buildToolsVersion "23.0.2" defaultConfig { applicationId "gui.com.lgimagecompressor" minSdkVersion 14 targetSdkVersion 23 versionCode 1 versionName "1.0" } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } } dependencies { compile fileTree(dir: 'libs', include: ['*.jar']) testCompile 'junit:junit:4.12' compile 'com.android.support:appcompat-v7:23.4.0' } ================================================ FILE: app/proguard-rules.pro ================================================ # Add project specific ProGuard rules here. # By default, the flags in this file are appended to flags specified # in /Users/guizhigang/Library/Android/sdk/tools/proguard/proguard-android.txt # You can edit the include path and order by changing the proguardFiles # directive in build.gradle. # # For more details, see # http://developer.android.com/guide/developing/tools/proguard.html # Add any project specific keep options here: # If your project uses WebView with JS, uncomment the following # and specify the fully qualified class name to the JavaScript interface # class: #-keepclassmembers class fqcn.of.javascript.interface.for.webview { # public *; #} ================================================ FILE: app/src/androidTest/java/gui/com/lgimagecompressor/ApplicationTest.java ================================================ package gui.com.lgimagecompressor; import android.app.Application; import android.test.ApplicationTestCase; /** * Testing Fundamentals */ public class ApplicationTest extends ApplicationTestCase { public ApplicationTest() { super(Application.class); } } ================================================ FILE: app/src/main/AndroidManifest.xml ================================================ ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/BasicCompressActivity.java ================================================ package gui.com.lgimagecompressor; import android.content.Intent; import android.content.pm.PackageManager; import android.graphics.Bitmap; import android.net.Uri; import android.os.Environment; import android.provider.MediaStore; import android.support.annotation.NonNull; import android.support.v4.app.ActivityCompat; import android.support.v4.content.ContextCompat; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.ImageView; import android.widget.TextView; import android.widget.Toast; import java.io.File; import java.io.IOException; import java.text.SimpleDateFormat; import java.util.Date; public class BasicCompressActivity extends AppCompatActivity implements LGImgCompressor.CompressListener{ private final String TAG = MainActivity.class.getSimpleName(); private ImageView imageView; private TextView imageInfo; private final static int CAMERA_REQESTCODE = 100; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_basic_compress); imageInfo = (TextView) findViewById(R.id.image_info); imageView = (ImageView) findViewById(R.id.image_view); imageView.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { requestPermission(); } }); } //处理6.0动态权限问题 private void requestPermission() { if (ContextCompat.checkSelfPermission(this, android.Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions( this, new String[]{android.Manifest.permission.WRITE_EXTERNAL_STORAGE}, CAMERA_REQESTCODE); } else { takePictureFormCamera(); } } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { if (requestCode == CAMERA_REQESTCODE) { if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { takePictureFormCamera(); } else { Toast.makeText(this, "需要允许写入权限来存储图片", Toast.LENGTH_LONG).show(); } } } private File imageFile; private void takePictureFormCamera() { Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); String timeStamp = new SimpleDateFormat("yyyyMMddHHmmss").format(new Date()); String fileName = timeStamp + "_"; File fileDir = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES); imageFile = null; try { imageFile = File.createTempFile(fileName, ".jpg", fileDir); } catch (IOException e) { e.printStackTrace(); } intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(imageFile)); startActivityForResult(intent, CAMERA_REQESTCODE); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (resultCode == RESULT_OK) { if (requestCode == CAMERA_REQESTCODE) { LGImgCompressor.getInstance(this).withListener(this). starCompress(Uri.fromFile(imageFile).toString(), 600, 800, 100); // LGImgCompressor.getInstance(this).withListener(this). // starCompressWithDefault(Uri.fromFile(imageFile).toString()); } } } @Override public void onCompressStart() { Log.d(TAG, "onCompressStart"); } @Override public void onCompressEnd(LGImgCompressor.CompressResult compressResult) { Log.d(TAG, "onCompressEnd outPath:" + compressResult.getOutPath()); if (compressResult.getStatus() == LGImgCompressor.CompressResult.RESULT_ERROR)//压缩失败 return; File file = new File(compressResult.getOutPath()); Bitmap bitmap = null; try { bitmap = MediaStore.Images.Media.getBitmap(getContentResolver(), Uri.fromFile(file)); imageView.setImageBitmap(bitmap); float imageFileSize = file.length() / 1024f; imageInfo.setText("image info width:" + bitmap.getWidth() + " \nheight:" + bitmap.getHeight() + " \nsize:" + imageFileSize + "kb" + "\nimagePath:" + file.getAbsolutePath()); } catch (IOException e) { e.printStackTrace(); } } } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/CompressServiceListener.java ================================================ package gui.com.lgimagecompressor; import java.util.ArrayList; /** * Created by guizhigang on 16/5/28. */ public interface CompressServiceListener { void onCompressServiceStart(); void onCompressServiceEnd(ArrayList compressResults); } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/CompressServiceParam.java ================================================ package gui.com.lgimagecompressor; import android.os.Parcel; import android.os.Parcelable; public class CompressServiceParam implements Parcelable { private int outWidth; private int outHeight; private int maxFileSize; private String srcImageUri; public CompressServiceParam() { } protected CompressServiceParam(Parcel in) { outWidth = in.readInt(); outHeight = in.readInt(); maxFileSize = in.readInt(); srcImageUri = in.readString(); } public static final Creator CREATOR = new Creator() { @Override public CompressServiceParam createFromParcel(Parcel in) { return new CompressServiceParam(in); } @Override public CompressServiceParam[] newArray(int size) { return new CompressServiceParam[size]; } }; public int getOutWidth() { return outWidth; } public void setOutWidth(int outWidth) { this.outWidth = outWidth; } public int getOutHeight() { return outHeight; } public void setOutHeight(int outHeight) { this.outHeight = outHeight; } public int getMaxFileSize() { return maxFileSize; } public void setMaxFileSize(int maxFileSize) { this.maxFileSize = maxFileSize; } public String getSrcImageUri() { return srcImageUri; } public void setSrcImageUri(String srcImageUri) { this.srcImageUri = srcImageUri; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeInt(outWidth); dest.writeInt(outHeight); dest.writeInt(maxFileSize); dest.writeString(srcImageUri); } } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/Constanse.java ================================================ package gui.com.lgimagecompressor; /** * Created by guizhigang on 16/5/28. */ public abstract class Constanse { public static final String COMPRESS_PARAM = "gui.com.lgimagecompressor.extra.PARAM"; public static final String ACTION_COMPRESS_BROADCAST = "gui.com.lgimagecompressor.message.broadcast"; public static final String KEY_COMPRESS_PROCCESSING = "gui.com.lgimagecompressor.message.proccessing"; public static final String KEY_COMPRESS_FLAG = "gui.com.lgimagecompressor.message.flag"; public static final String KEY_COMPRESS_RESULT = "gui.com.lgimagecompressor.message.result"; public static final int FLAG_BEGAIIN = 0; public static final int FLAG_END = 1; } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/LGImgCompressor.java ================================================ package gui.com.lgimagecompressor; import android.content.Context; import android.database.Cursor; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Matrix; import android.media.ExifInterface; import android.net.Uri; import android.os.AsyncTask; import android.os.Environment; import android.os.Parcel; import android.os.Parcelable; import android.provider.MediaStore; import java.io.BufferedOutputStream; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; /** * Created by guizhigang on 16/5/25. */ public class LGImgCompressor { private static LGImgCompressor instance = null; private Context context; private CompressListener compressListener; private static final int DEFAULT_OUTWIDTH = 720; private static final int DEFAULT_OUTHEIGHT = 1080; private static final int DEFAULT_MAXFILESIZE = 1024;//KB private LGImgCompressor(Context context) { this.context = context; } public static LGImgCompressor getInstance(Context context) { if (instance == null) { synchronized (LGImgCompressor.class) { if (instance == null) instance = new LGImgCompressor(context.getApplicationContext()); } } return instance; } public LGImgCompressor withListener(CompressListener compressListener) { this.compressListener = compressListener; return this; } /** * 通过uri地址获取文件路径 * @param uri * @return */ private String getFilePathFromUri(String uri) { Uri pathUri = Uri.parse(uri); Cursor cursor = context.getContentResolver().query(pathUri, null, null, null, null); if (cursor == null) { return pathUri.getPath(); } else { cursor.moveToFirst(); int index = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA); String str = cursor.getString(index); cursor.close(); return str; } } /** * Can't compress a recycled bitmap * @param srcImageUri 原始图片的uri路径 * @param outWidth 期望的输出图片的宽度 * @param outHeight 期望的输出图片的高度 * @param maxFileSize 期望的输出图片的最大占用的存储空间 * @return */ public String compressImage(String srcImageUri, int outWidth, int outHeight, int maxFileSize) { String srcImagePath = getFilePathFromUri(srcImageUri); //进行大小缩放来达到压缩的目的 BitmapFactory.Options options = new BitmapFactory.Options(); options.inJustDecodeBounds = true; BitmapFactory.decodeFile(srcImagePath, options); //根据原始图片的宽高比和期望的输出图片的宽高比计算最终输出的图片的宽和高 float srcWidth = options.outWidth; float srcHeight = options.outHeight; float maxWidth = outWidth; float maxHeight = outHeight; float srcRatio = srcWidth / srcHeight; float outRatio = maxWidth / maxHeight; float actualOutWidth = srcWidth; float actualOutHeight = srcHeight; if (srcWidth > maxWidth || srcHeight > maxHeight) { //如果输入比率小于输出比率,则最终输出的宽度以maxHeight为准() //比如输入比为10:20 输出比是300:10 如果要保证输出图片的宽高比和原始图片的宽高比相同,则最终输出图片的高为10 //宽度为10/20 * 10 = 5 最终输出图片的比率为5:10 和原始输入的比率相同 //同理如果输入比率大于输出比率,则最终输出的高度以maxHeight为准() //比如输入比为20:10 输出比是5:100 如果要保证输出图片的宽高比和原始图片的宽高比相同,则最终输出图片的宽为5 //高度需要根据输入图片的比率计算获得 为5 / 20/10= 2.5 最终输出图片的比率为5:2.5 和原始输入的比率相同 if (srcRatio < outRatio) { actualOutHeight = maxHeight; actualOutWidth = actualOutHeight * srcRatio; } else if (srcRatio > outRatio) { actualOutWidth = maxWidth; actualOutHeight = actualOutWidth / srcRatio; } else { actualOutWidth = maxWidth; actualOutHeight = maxHeight; } } options.inSampleSize = computSampleSize(options, actualOutWidth, actualOutHeight); options.inJustDecodeBounds = false; Bitmap scaledBitmap = null; try { scaledBitmap = BitmapFactory.decodeFile(srcImagePath, options); } catch (OutOfMemoryError e) { e.printStackTrace(); } if (scaledBitmap == null) { return null;//压缩失败 } //生成最终输出的bitmap Bitmap actualOutBitmap = Bitmap.createScaledBitmap(scaledBitmap, (int) actualOutWidth, (int) actualOutHeight, true); if(actualOutBitmap != scaledBitmap) scaledBitmap.recycle(); //处理图片旋转问题 ExifInterface exif = null; try { exif = new ExifInterface(srcImagePath); int orientation = exif.getAttributeInt( ExifInterface.TAG_ORIENTATION, 0); Matrix matrix = new Matrix(); if (orientation == ExifInterface.ORIENTATION_ROTATE_90) { matrix.postRotate(90); } else if (orientation == ExifInterface.ORIENTATION_ROTATE_180) { matrix.postRotate(180); } else if (orientation == ExifInterface.ORIENTATION_ROTATE_270) { matrix.postRotate(270); } actualOutBitmap = Bitmap.createBitmap(actualOutBitmap, 0, 0, actualOutBitmap.getWidth(), actualOutBitmap.getHeight(), matrix, true); } catch (IOException e) { e.printStackTrace(); return null; } //进行有损压缩 ByteArrayOutputStream baos = new ByteArrayOutputStream(); int options_ = 100; actualOutBitmap.compress(Bitmap.CompressFormat.JPEG, options_, baos);//质量压缩方法,把压缩后的数据存放到baos中 (100表示不压缩,0表示压缩到最小) int baosLength = baos.toByteArray().length; while (baosLength / 1024 > maxFileSize) {//循环判断如果压缩后图片是否大于maxMemmorrySize,大于继续压缩 baos.reset();//重置baos即让下一次的写入覆盖之前的内容 options_ = Math.max(0, options_ - 10);//图片质量每次减少10 actualOutBitmap.compress(Bitmap.CompressFormat.JPEG, options_, baos);//将压缩后的图片保存到baos中 baosLength = baos.toByteArray().length; if (options_ == 0)//如果图片的质量已降到最低则,不再进行压缩 break; } actualOutBitmap.recycle(); //将bitmap保存到指定路径 FileOutputStream fos = null; String filePath = getOutputFileName(srcImagePath); try { fos = new FileOutputStream(filePath); //包装缓冲流,提高写入速度 BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(fos); bufferedOutputStream.write(baos.toByteArray()); bufferedOutputStream.flush(); } catch (FileNotFoundException e) { return null; } catch (IOException e) { return null; } finally { if (baos != null) { try { baos.close(); } catch (IOException e) { e.printStackTrace(); } } if (fos != null) { try { fos.close(); } catch (IOException e) { e.printStackTrace(); } } } return filePath; } private int computSampleSize(BitmapFactory.Options options, float reqWidth, float reqHeight) { float srcWidth = options.outWidth;//20 float srcHeight = options.outHeight;//10 int sampleSize = 1; if (srcWidth > reqWidth || srcHeight > reqHeight) { int withRatio = Math.round(srcWidth / reqWidth); int heightRatio = Math.round(srcHeight / reqHeight); sampleSize = Math.min(withRatio, heightRatio); } return sampleSize; } private String getOutputFileName(String srcFilePath) { File srcFile = new File(srcFilePath); File file = new File(Environment.getExternalStorageDirectory().getPath(), "LGImgCompressor/Images"); if (!file.exists()) { file.mkdirs(); } String uriSting = (file.getAbsolutePath() + File.separator + srcFile.getName()); return uriSting; } public void starCompress(String srcImageUri, int outWidth, int outHeight, int maxFileSize) { new CompressTask().execute(srcImageUri, "" + outWidth, "" + outHeight, "" + maxFileSize); } public void starCompressWithDefault(String srcImageUri) { new CompressTask().execute(srcImageUri, "" + DEFAULT_OUTWIDTH, "" + DEFAULT_OUTHEIGHT, "" + DEFAULT_MAXFILESIZE); } public static class CompressResult implements Parcelable{ public static final int RESULT_OK = 0; public static final int RESULT_ERROR = 1; private int status = RESULT_OK; private String srcPath; private String outPath; public CompressResult(){ } protected CompressResult(Parcel in) { status = in.readInt(); srcPath = in.readString(); outPath = in.readString(); } public static final Creator CREATOR = new Creator() { @Override public CompressResult createFromParcel(Parcel in) { return new CompressResult(in); } @Override public CompressResult[] newArray(int size) { return new CompressResult[size]; } }; @Override public int describeContents() { return 0; } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeInt(status); dest.writeString(srcPath); dest.writeString(outPath); } public int getStatus() { return status; } public void setStatus(int status) { this.status = status; } public String getSrcPath() { return srcPath; } public void setSrcPath(String srcPath) { this.srcPath = srcPath; } public String getOutPath() { return outPath; } public void setOutPath(String outPath) { this.outPath = outPath; } } /** * 压缩结果回到监听类 */ public interface CompressListener { void onCompressStart(); void onCompressEnd(CompressResult imageOutPath); } private class CompressTask extends AsyncTask { @Override protected CompressResult doInBackground(String... params) { String path = params[0]; int outWidth = Integer.parseInt(params[1]); int outHeight = Integer.parseInt(params[2]); int maxFileSize = Integer.parseInt(params[3]); CompressResult compressResult = new CompressResult(); String outPutPath = null; try { outPutPath = compressImage(path, outWidth, outHeight, maxFileSize); }catch (Exception e){ } compressResult.setSrcPath(path); compressResult.setOutPath(outPutPath); if(outPutPath == null){ compressResult.setStatus(CompressResult.RESULT_ERROR); } return compressResult; } @Override protected void onPreExecute() { if (compressListener != null) { compressListener.onCompressStart(); } } @Override protected void onPostExecute(CompressResult compressResult) { if (compressListener != null) { compressListener.onCompressEnd(compressResult); } } } } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/LGImgCompressorIntentService.java ================================================ package gui.com.lgimagecompressor; import android.app.IntentService; import android.content.Intent; import android.content.Context; import android.util.Log; import java.util.ArrayList; public class LGImgCompressorIntentService extends IntentService { private final String TAG = LGImgCompressorIntentService.class.getSimpleName(); private static final String ACTION_COMPRESS = "gui.com.lgimagecompressor.action.COMPRESS"; private ArrayList compressResults = new ArrayList<>();//存储压缩任务的返回结果 public LGImgCompressorIntentService() { super("LGImgCompressorIntentService"); setIntentRedelivery(false);//避免出异常后service重新启动 } @Override public void onCreate() { super.onCreate(); Intent intent = new Intent(Constanse.ACTION_COMPRESS_BROADCAST); intent.putExtra(Constanse.KEY_COMPRESS_FLAG,Constanse.FLAG_BEGAIIN); sendBroadcast(intent); Log.d(TAG,"onCreate..."); } @Override public void onDestroy() { super.onDestroy(); Intent intent = new Intent(Constanse.ACTION_COMPRESS_BROADCAST); intent.putExtra(Constanse.KEY_COMPRESS_FLAG,Constanse.FLAG_END); intent.putParcelableArrayListExtra(Constanse.KEY_COMPRESS_RESULT,compressResults); sendBroadcast(intent); compressResults.clear(); Log.d(TAG,"onDestroy..."); } public static void startActionCompress(Context context, CompressServiceParam param) { Intent intent = new Intent(context, LGImgCompressorIntentService.class); intent.setAction(ACTION_COMPRESS); intent.putExtra(Constanse.COMPRESS_PARAM, param); context.startService(intent); } @Override protected void onHandleIntent(Intent intent) { if (intent != null) { final String action = intent.getAction(); if (ACTION_COMPRESS.equals(action)) { final CompressServiceParam param1 = intent.getParcelableExtra(Constanse.COMPRESS_PARAM); handleActionCompress(param1); } } } private void handleActionCompress(CompressServiceParam param) { int outwidth = param.getOutWidth(); int outHieight = param.getOutHeight(); int maxFileSize = param.getMaxFileSize(); String srcImageUri = param.getSrcImageUri(); LGImgCompressor.CompressResult compressResult = new LGImgCompressor.CompressResult(); String outPutPath = null; try { outPutPath = LGImgCompressor.getInstance(this).compressImage(srcImageUri, outwidth, outHieight, maxFileSize); } catch (Exception e) { } compressResult.setSrcPath(srcImageUri); compressResult.setOutPath(outPutPath); if (outPutPath == null) { compressResult.setStatus(LGImgCompressor.CompressResult.RESULT_ERROR); } compressResults.add(compressResult); } } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/LGImgCompressorService.java ================================================ package gui.com.lgimagecompressor; import android.app.Service; import android.content.Intent; import android.os.IBinder; import android.util.Log; import java.util.ArrayList; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class LGImgCompressorService extends Service { private static final String TAG = "GImgCompressorService"; private ArrayList compressResults = new ArrayList<>(); public LGImgCompressorService() { } @Override public void onCreate() { super.onCreate(); Log.d(TAG,"onCreate..."); // executorService = Executors.newCachedThreadPool(); executorService = Executors.newFixedThreadPool(10); Intent intent = new Intent(Constanse.ACTION_COMPRESS_BROADCAST); intent.putExtra(Constanse.KEY_COMPRESS_FLAG,Constanse.FLAG_BEGAIIN); sendBroadcast(intent); } @Override public void onDestroy() { super.onDestroy(); Log.d(TAG,"onDestroy..."); Intent intent = new Intent(Constanse.ACTION_COMPRESS_BROADCAST); intent.putExtra(Constanse.KEY_COMPRESS_FLAG,Constanse.FLAG_END); intent.putParcelableArrayListExtra(Constanse.KEY_COMPRESS_RESULT,compressResults); sendBroadcast(intent); compressResults.clear(); executorService.shutdownNow(); } @Override public int onStartCommand(Intent intent, int flags, int startId) { doCompressImages(intent,startId); return Service.START_NOT_STICKY; } private int taskNumber; private ExecutorService executorService; private final Object lock = new Object(); private void doCompressImages(final Intent intent,final int taskId){ final ArrayList paramArrayList = intent.getParcelableArrayListExtra(Constanse.COMPRESS_PARAM); synchronized (lock){ taskNumber += paramArrayList.size(); } //如果paramArrayList过大,为了避免"The application may be doing too much work on its main thread"的问题,将任务的创建和执行统一放在后台线程中执行 new Thread(new Runnable() { @Override public void run() { for (int i = 0; i < paramArrayList.size(); ++i){ executorService.execute(new CompressTask(paramArrayList.get(i),taskId)); } } }).start(); } private class CompressTask implements Runnable{ private CompressServiceParam param; private int taskId ; private CompressTask(CompressServiceParam compressServiceParam,int taskId){ this.param = compressServiceParam; this.taskId = taskId; } @Override public void run() { Log.d(TAG,taskId + " do compress begain..." + Thread.currentThread().getId()); int outwidth = param.getOutWidth(); int outHieight = param.getOutHeight(); int maxFileSize = param.getMaxFileSize(); String srcImageUri = param.getSrcImageUri(); LGImgCompressor.CompressResult compressResult = new LGImgCompressor.CompressResult(); String outPutPath = null; try { outPutPath = LGImgCompressor.getInstance(LGImgCompressorService.this).compressImage( srcImageUri, outwidth, outHieight, maxFileSize); } catch (Exception e) { } compressResult.setSrcPath(srcImageUri); compressResult.setOutPath(outPutPath); if (outPutPath == null) { compressResult.setStatus(LGImgCompressor.CompressResult.RESULT_ERROR); } Log.d(TAG,taskId + " do compress end..." + Thread.currentThread().getId()); synchronized (lock){ compressResults.add(compressResult); taskNumber--; if(taskNumber <= 0){ stopSelf(taskId); } } } } @Override public IBinder onBind(Intent intent) { throw null; } } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/MainActivity.java ================================================ package gui.com.lgimagecompressor; import android.content.Intent; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.view.View; public class MainActivity extends AppCompatActivity { private final String TAG = MainActivity.class.getSimpleName(); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); findViewById(R.id.basic_test).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(MainActivity.this,BasicCompressActivity.class); startActivity(intent); } }); findViewById(R.id.from_service).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(MainActivity.this,ServiceCompressActivity.class); startActivity(intent); } }); } } ================================================ FILE: app/src/main/java/gui/com/lgimagecompressor/ServiceCompressActivity.java ================================================ package gui.com.lgimagecompressor; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.database.Cursor; import android.net.Uri; import android.os.Environment; import android.provider.MediaStore; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.TextView; import java.io.File; import java.util.ArrayList; public class ServiceCompressActivity extends AppCompatActivity { private final String TAG = ServiceCompressActivity.class.getSimpleName(); private long serviceStartTime; private CompressingReciver reciver; private TextView infoView; private class CompressingReciver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { Log.d(TAG, "onReceive:" + Thread.currentThread().getId()); int flag = intent.getIntExtra(Constanse.KEY_COMPRESS_FLAG,-1); Log.d(TAG," flag:" + flag); if(flag == Constanse.FLAG_BEGAIIN){ Log.d(TAG, "onCompressServiceStart"); serviceStartTime = System.currentTimeMillis(); updateInfo("compress begain..."); return; } if(flag == Constanse.FLAG_END){ ArrayList compressResults = (ArrayList)intent.getSerializableExtra(Constanse.KEY_COMPRESS_RESULT); Log.d(TAG, compressResults.size() + "compressed done"); Log.d(TAG, "compress " + compressResults.size() + " files used total time:" + (System.currentTimeMillis() - serviceStartTime)); updateInfo(compressResults.size() + " files compressed done \nused total time:" + (System.currentTimeMillis() - serviceStartTime) + "ms"); } } } private void updateInfo(String message){ infoView.setText(message); } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_intent_service_compress); reciver = new CompressingReciver(); IntentFilter intentFilter = new IntentFilter(Constanse.ACTION_COMPRESS_BROADCAST); registerReceiver(reciver, intentFilter); infoView = (TextView)findViewById(R.id.compress_info); final int maxSize = 0; findViewById(R.id.intent_service).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { ArrayList compressFiles = getImagesPathFormAlbum(); Log.d(TAG, compressFiles.size() + "compresse begain"); int size = compressFiles.size() > 10 ? 10:compressFiles.size(); for (int i = 0; i < compressFiles.size(); ++i) { Uri uri = compressFiles.get(i); CompressServiceParam param = new CompressServiceParam(); param.setOutHeight(800); param.setOutWidth(600); param.setMaxFileSize(400); param.setSrcImageUri(uri.toString()); LGImgCompressorIntentService.startActionCompress(ServiceCompressActivity.this, param); } } }); findViewById(R.id.service).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { ArrayList compressFiles = getImagesPathFormAlbum(); int size = compressFiles.size() > 10 ? 10:compressFiles.size(); ArrayList tasks = new ArrayList(compressFiles.size()); for (int i = 0; i < compressFiles.size(); ++i) { Uri uri = compressFiles.get(i); CompressServiceParam param = new CompressServiceParam(); param.setOutHeight(800); param.setOutWidth(600); param.setMaxFileSize(400); param.setSrcImageUri(uri.toString()); tasks.add(param); } Log.d(TAG, compressFiles.size() + "compresse begain"); Intent intent = new Intent(ServiceCompressActivity.this, LGImgCompressorService.class); intent.putParcelableArrayListExtra(Constanse.COMPRESS_PARAM, tasks); startService(intent); } }); } private ArrayList getImagesPathFormAlbum() { ArrayList paths = new ArrayList<>(); //selection: 指定查询条件 String selection = MediaStore.Images.Media.MIME_TYPE + "=? or " + MediaStore.Images.Media.MIME_TYPE + "=?"; //定义selection参数匹配值 String[] selectionArgs = {"image/jpeg", "image/png"}; Cursor cursor = getContentResolver().query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, null, selection, selectionArgs, MediaStore.Images.Media.DATE_MODIFIED); if (cursor != null && cursor.moveToFirst()) { do { long id = cursor.getInt(cursor.getColumnIndex(MediaStore.Images.Media._ID)); String title = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.TITLE)); String url = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA)); long size = (int) cursor.getLong(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.SIZE)); long lastModified = (int) cursor.getLong(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATE_MODIFIED)); //排除size为0的无效文件 if (size != 0) { Uri uri = Uri.parse(url); paths.add(uri); } } while (cursor.moveToNext()); cursor.close(); } return paths; } @Override protected void onDestroy() { super.onDestroy(); if(reciver != null){ unregisterReceiver(reciver); } } } ================================================ FILE: app/src/main/res/layout/activity_basic_compress.xml ================================================ ================================================ FILE: app/src/main/res/layout/activity_intent_service_compress.xml ================================================