2015-02-10 96 views
27

如何在不降低android图像质量的情况下缩小图像大小。
我在我的项目中有一个ImageView,我必须在其中放入更大尺寸的图像。
我是android新手。
请帮忙。在android中减少图像大小而不损失其质量

+0

我已经使用剪裁。但。裁剪图像损害其质量..裁剪后,我可以添加图像到imageview.but它降低图像的质量..但是当我正在拍摄更大的图像..图像不显示.. – 2015-02-10 05:54:58

+0

现在我怎么能减少图像的大小与你失去它的质量..可以提供一些示例代码..plzz帮助...预先感谢.. – 2015-02-10 05:56:40

回答

48

下面的代码为宽高比为灰度图像的工作:

Bitmap bitmapImage = BitmapFactory.decodeFile("Your path"); 
int nh = (int) (bitmapImage.getHeight() * (512.0/bitmapImage.getWidth())); 
Bitmap scaled = Bitmap.createScaledBitmap(bitmapImage, 512, nh, true); 
your_imageview.setImageBitmap(scaled); 

压缩图像,而不会失去质量例如WhatsApp

public String compressImage(String imageUri) { 

     String filePath = getRealPathFromURI(imageUri); 
     Bitmap scaledBitmap = null; 

     BitmapFactory.Options options = new BitmapFactory.Options(); 

//  by setting this field as true, the actual bitmap pixels are not loaded in the memory. Just the bounds are loaded. If 
//  you try the use the bitmap here, you will get null. 
     options.inJustDecodeBounds = true; 
     Bitmap bmp = BitmapFactory.decodeFile(filePath, options); 

     int actualHeight = options.outHeight; 
     int actualWidth = options.outWidth; 

//  max Height and width values of the compressed image is taken as 816x612 

     float maxHeight = 816.0f; 
     float maxWidth = 612.0f; 
     float imgRatio = actualWidth/actualHeight; 
     float maxRatio = maxWidth/maxHeight; 

//  width and height values are set maintaining the aspect ratio of the image 

     if (actualHeight > maxHeight || actualWidth > maxWidth) { 
      if (imgRatio < maxRatio) { 
       imgRatio = maxHeight/actualHeight; 
       actualWidth = (int) (imgRatio * actualWidth); 
       actualHeight = (int) maxHeight; 
      } else if (imgRatio > maxRatio) { 
       imgRatio = maxWidth/actualWidth; 
       actualHeight = (int) (imgRatio * actualHeight); 
       actualWidth = (int) maxWidth; 
      } else { 
       actualHeight = (int) maxHeight; 
       actualWidth = (int) maxWidth; 

      } 
     } 

//  setting inSampleSize value allows to load a scaled down version of the original image 

     options.inSampleSize = calculateInSampleSize(options, actualWidth, actualHeight); 

//  inJustDecodeBounds set to false to load the actual bitmap 
     options.inJustDecodeBounds = false; 

//  this options allow android to claim the bitmap memory if it runs low on memory 
     options.inPurgeable = true; 
     options.inInputShareable = true; 
     options.inTempStorage = new byte[16 * 1024]; 

     try { 
//   load the bitmap from its path 
      bmp = BitmapFactory.decodeFile(filePath, options); 
     } catch (OutOfMemoryError exception) { 
      exception.printStackTrace(); 

     } 
     try { 
      scaledBitmap = Bitmap.createBitmap(actualWidth, actualHeight, Bitmap.Config.ARGB_8888); 
     } catch (OutOfMemoryError exception) { 
      exception.printStackTrace(); 
     } 

     float ratioX = actualWidth/(float) options.outWidth; 
     float ratioY = actualHeight/(float) options.outHeight; 
     float middleX = actualWidth/2.0f; 
     float middleY = actualHeight/2.0f; 

     Matrix scaleMatrix = new Matrix(); 
     scaleMatrix.setScale(ratioX, ratioY, middleX, middleY); 

     Canvas canvas = new Canvas(scaledBitmap); 
     canvas.setMatrix(scaleMatrix); 
     canvas.drawBitmap(bmp, middleX - bmp.getWidth()/2, middleY - bmp.getHeight()/2, new Paint(Paint.FILTER_BITMAP_FLAG)); 

//  check the rotation of the image and display it properly 
     ExifInterface exif; 
     try { 
      exif = new ExifInterface(filePath); 

      int orientation = exif.getAttributeInt(
        ExifInterface.TAG_ORIENTATION, 0); 
      Log.d("EXIF", "Exif: " + orientation); 
      Matrix matrix = new Matrix(); 
      if (orientation == 6) { 
       matrix.postRotate(90); 
       Log.d("EXIF", "Exif: " + orientation); 
      } else if (orientation == 3) { 
       matrix.postRotate(180); 
       Log.d("EXIF", "Exif: " + orientation); 
      } else if (orientation == 8) { 
       matrix.postRotate(270); 
       Log.d("EXIF", "Exif: " + orientation); 
      } 
      scaledBitmap = Bitmap.createBitmap(scaledBitmap, 0, 0, 
        scaledBitmap.getWidth(), scaledBitmap.getHeight(), matrix, 
        true); 
     } catch (IOException e) { 
      e.printStackTrace(); 
     } 

     FileOutputStream out = null; 
     String filename = getFilename(); 
     try { 
      out = new FileOutputStream(filename); 

//   write the compressed bitmap at the destination specified by filename. 
      scaledBitmap.compress(Bitmap.CompressFormat.JPEG, 80, out); 

     } catch (FileNotFoundException e) { 
      e.printStackTrace(); 
     } 

     return filename; 

    } 

    public String getFilename() { 
     File file = new File(Environment.getExternalStorageDirectory().getPath(), "MyFolder/Images"); 
     if (!file.exists()) { 
      file.mkdirs(); 
     } 
     String uriSting = (file.getAbsolutePath() + "/" + System.currentTimeMillis() + ".jpg"); 
     return uriSting; 

    } 

    private String getRealPathFromURI(String contentURI) { 
     Uri contentUri = Uri.parse(contentURI); 
     Cursor cursor = getContentResolver().query(contentUri, null, null, null, null); 
     if (cursor == null) { 
      return contentUri.getPath(); 
     } else { 
      cursor.moveToFirst(); 
      int index = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA); 
      return cursor.getString(index); 
     } 
    } 

    public int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) { 
     final int height = options.outHeight; 
     final int width = options.outWidth; 
     int inSampleSize = 1; 

     if (height > reqHeight || width > reqWidth) { 
      final int heightRatio = Math.round((float) height/(float) reqHeight); 
      final int widthRatio = Math.round((float) width/(float) reqWidth); 
      inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio; 
     } 
     final float totalPixels = width * height; 
     final float totalReqPixelsCap = reqWidth * reqHeight * 2; 
     while (totalPixels/(inSampleSize * inSampleSize) > totalReqPixelsCap) { 
      inSampleSize++; 
     } 

     return inSampleSize; 
    } 
+0

@Jitty Aandyan如果图像大小和尺寸太小或太大,你的代码将如何反应...为什么816x612是硬编码的,我们可以通过编程来计算 – Akshay 2015-03-25 13:50:27

+1

基本上上面的代码用于在保持纵横比的情况下减小图像的大小。最重要的是每个图像都应该具有特定的高度和宽度,这些都是由您的应用程序的要求定义的。如果您使用高质量图像作为配置文件图片,则可以将最大高度最大宽度更改为300,以便尊重您的实施。而且,如果压缩前的实际高度和宽度大于要求,则可以检查压缩前的实际高度和宽度。 – 2015-03-27 07:52:31

+0

如何使它适用于所有尺寸的图像以及具有不同分辨率的所有设备, – Akshay 2015-09-01 12:23:01

0

创建类CompressFile图片

public class CompressFile { 
    public static File getCompressedImageFile(File file, Context mContext) { 
     try { 
      BitmapFactory.Options o = new BitmapFactory.Options(); 
      o.inJustDecodeBounds = true; 

      if (getFileExt(file.getName()).equals("png") || getFileExt(file.getName()).equals("PNG")) { 
       o.inSampleSize = 6; 
      } else { 
       o.inSampleSize = 6; 
      } 

      FileInputStream inputStream = new FileInputStream(file); 
      BitmapFactory.decodeStream(inputStream, null, o); 
      inputStream.close(); 

      // The new size we want to scale to 
      final int REQUIRED_SIZE = 100; 

      // Find the correct scale value. It should be the power of 2. 
      int scale = 1; 
      while (o.outWidth/scale/2 >= REQUIRED_SIZE && 
        o.outHeight/scale/2 >= REQUIRED_SIZE) { 
       scale *= 2; 
      } 

      BitmapFactory.Options o2 = new BitmapFactory.Options(); 
      o2.inSampleSize = scale; 
      inputStream = new FileInputStream(file); 

      Bitmap selectedBitmap = BitmapFactory.decodeStream(inputStream, null, o2); 

      ExifInterface ei = new ExifInterface(file.getAbsolutePath()); 
      int orientation = ei.getAttributeInt(ExifInterface.TAG_ORIENTATION, 
        ExifInterface.ORIENTATION_UNDEFINED); 

      switch (orientation) { 

       case ExifInterface.ORIENTATION_ROTATE_90: 
        rotateImage(selectedBitmap, 90); 
        break; 

       case ExifInterface.ORIENTATION_ROTATE_180: 
        rotateImage(selectedBitmap, 180); 
        break; 

       case ExifInterface.ORIENTATION_ROTATE_270: 
        rotateImage(selectedBitmap, 270); 
        break; 

       case ExifInterface.ORIENTATION_NORMAL: 

       default: 
        break; 
      } 
      inputStream.close(); 


      // here i override the original image file 
      File folder = new File(Environment.getExternalStorageDirectory() + "/FolderName"); 
      boolean success = true; 
      if (!folder.exists()) { 
       success = folder.mkdir(); 
      } 
      if (success) { 
       File newFile = new File(new File(folder.getAbsolutePath()), file.getName()); 
       if (newFile.exists()) { 
        newFile.delete(); 
       } 
       FileOutputStream outputStream = new FileOutputStream(newFile); 

       if (getFileExt(file.getName()).equals("png") || getFileExt(file.getName()).equals("PNG")) { 
        selectedBitmap.compress(Bitmap.CompressFormat.PNG, 100, outputStream); 
       } else { 
        selectedBitmap.compress(Bitmap.CompressFormat.JPEG, 100, outputStream); 
       } 

       return newFile; 
      } else { 
       return null; 
      } 
     } catch (Exception e) { 
      return null; 
     } 
    } 

    public static String getFileExt(String fileName) { 
     return fileName.substring(fileName.lastIndexOf(".") + 1, fileName.length()); 
    } 

    public static Bitmap rotateImage(Bitmap source, float angle) { 
     Matrix matrix = new Matrix(); 
     matrix.postRotate(angle); 
     return Bitmap.createBitmap(source, 0, 0, source.getWidth(), source.getHeight(), 
       matrix, true); 
    } 
} 
相关问题