为什么输出图像文件大于使用imagick的输入?

时间:2014-04-15 09:21:19

标签: php imagick

为什么输出图像比使用imagick的输入更大? 原jpg:http://www.persunmall.com/images/jm3.jpg

/**
 * example
 --------------------------------------
        'jm3.jpg' => 119 k   (453*680)
        'l.jpg' => 275 k   (400*600)        // why the output jpg is much bigger than input jpg?

 */
$jy_image=new jy_image
$jy_image->output_image('jm3.jpg','l.jpg',400,0);

下面的课程:

<?php 
class jy_image {
    private $image = null;
    private $type = null;

    // 构造函数
    public function __construct() {
    }


    // 析构函数
    public function __destruct() {
        if ($this->image !== null)
            $this->image->destroy();
    }

    // 载入图像
    public function open($path) {
        $this->image = new Imagick($path);
        if ($this->image) {
            $this->type = strtolower($this->image->getImageFormat());   // jpeg
        }
        return $this->image;
    }


    public function crop($x = 0, $y = 0, $width = null, $height = null) {
        if ($width == null)
            $width = $this->image->getImageWidth() - $x;
        if ($height == null)
            $height = $this->image->getImageHeight() - $y;
        if ($width <= 0 || $height <= 0)
            return;

        if ($this->type == 'gif') {
            $image  = $this->image;
            $canvas = new Imagick();

            $images = $image->coalesceImages();
            foreach ($images as $frame) {
                $img = new Imagick();
                $img->readImageBlob($frame);
                $img->cropImage($width, $height, $x, $y);

                $canvas->addImage($img);
                $canvas->setImageDelay($img->getImageDelay());
                $canvas->setImagePage($width, $height, 0, 0);
            }

            $image->destroy();
            $this->image = $canvas;
        } else {
            $this->image->cropImage($width, $height, $x, $y);
        }
    }

    /*
     * 更改图像大小
     $fit: 适应大小方式
     'force': 把图片强制变形成 $width X $height 大小
     'scale': 按比例在安全框 $width X $height 内缩放图片, 输出缩放后图像大小 不完全等于 $width X $height
     'scale_fill': 按比例在安全框 $width X $height 内缩放图片,安全框内没有像素的地方填充色, 使用此参数时可设置背景填充色 $bg_color = array(255,255,255)(红,绿,蓝, 透明度) 透明度(0不透明-127完全透明))
     其它: 智能模能 缩放图像并载取图像的中间部分 $width X $height 像素大小
     $fit = 'force','scale','scale_fill' 时: 输出完整图像
     $fit = 图像方位值 时, 输出指定位置部分图像 
     字母与图像的对应关系如下:

     north_west   north   north_east

     west         center        east

     south_west   south   south_east

     */
    public function resize_to($width = 100, $height = 100, $fit = 'center', $fill_color = array(255, 255, 255, 0)) {

        // 保证其中一边,另外一边相应缩放
        if($width==0 || $height==0){
            $image = $this->image;
            $src_width = $image->getImageWidth();
            $src_height = $image->getImageHeight();
            if($width>0 && $height==0){
                $width=($width>$src_width)?$src_width:$width;
                $height = $src_height*($width/$src_width);
            }
            else if($width==0 && $height>0){
                $height=($height>$src_height)?$src_height:$height;
                $width = $src_width*($height/$src_height);
            }
            $fit='force';
        }

        switch ($fit) {
            case 'force':   // 强制缩放,会变形
                if ($this->type == 'gif') {
                    $image  = $this->image;
                    $canvas = new Imagick();

                    $images = $image->coalesceImages();
                    foreach ($images as $frame) {
                        $img = new Imagick();
                        $img->readImageBlob($frame);
                        $img->thumbnailImage($width, $height, false);

                        $canvas->addImage($img);
                        $canvas->setImageDelay($img->getImageDelay());
                    }
                    $image->destroy();
                    $this->image = $canvas;
                } else {
                    $this->image->thumbnailImage($width, $height, false);
                }
                break;
            case 'scale':
                if ($this->type == 'gif') {
                    $image  = $this->image;
                    $images = $image->coalesceImages();
                    $canvas = new Imagick();
                    foreach ($images as $frame) {
                        $img = new Imagick();
                        $img->readImageBlob($frame);
                        $img->thumbnailImage($width, $height, true);

                        $canvas->addImage($img);
                        $canvas->setImageDelay($img->getImageDelay());
                    }
                    $image->destroy();
                    $this->image = $canvas;
                } else {
                    $this->image->thumbnailImage($width, $height, true);
                }
                break;
            case 'scale_fill':      // 填充白色
                $size       = $this->image->getImagePage();
                $src_width  = $size['width'];
                $src_height = $size['height'];

                $x = 0;
                $y = 0;

                $dst_width  = $width;
                $dst_height = $height;

                if ($src_width * $height > $src_height * $width) {
                    $dst_height = intval($width * $src_height / $src_width);
                    $y          = intval(($height - $dst_height) / 2);
                } else {
                    $dst_width = intval($height * $src_width / $src_height);
                    $x         = intval(($width - $dst_width) / 2);
                }

                $image  = $this->image;
                $canvas = new Imagick();

                $color = 'rgba(' . $fill_color[0] . ',' . $fill_color[1] . ',' . $fill_color[2] . ',' . $fill_color[3] . ')';
                if ($this->type == 'gif') {
                    $images = $image->coalesceImages();
                    foreach ($images as $frame) {
                        $frame->thumbnailImage($width, $height, true);

                        $draw = new ImagickDraw();
                        $draw->composite($frame->getImageCompose(), $x, $y, $dst_width, $dst_height, $frame);

                        $img = new Imagick();
                        $img->newImage($width, $height, $color, 'gif');
                        $img->drawImage($draw);

                        $canvas->addImage($img);
                        $canvas->setImageDelay($img->getImageDelay());
                        $canvas->setImagePage($width, $height, 0, 0);
                    }
                } else {
                    $image->thumbnailImage($width, $height, true);

                    $draw = new ImagickDraw();
                    $draw->composite($image->getImageCompose(), $x, $y, $dst_width, $dst_height, $image);

                    $canvas->newImage($width, $height, $color, $this->get_type());
                    $canvas->drawImage($draw);
                    $canvas->setImagePage($width, $height, 0, 0);
                }
                $image->destroy();
                $this->image = $canvas;
                break;
            default:
                $size       = $this->image->getImagePage();
                $src_width  = $size['width'];
                $src_height = $size['height'];

                $crop_x = 0;
                $crop_y = 0;

                $crop_w = $src_width;
                $crop_h = $src_height;

                if ($src_width * $height > $src_height * $width) {
                    $crop_w = intval($src_height * $width / $height);   // 100*50 =>50*100 输出:25*50
                } else {
                    $crop_h = intval($src_width * $height / $width);    //50*100 =>100*50 输出:50*25
                }

                switch ($fit) {
                    case 'north_west':
                        $crop_x = 0;
                        $crop_y = 0;
                        break;
                    case 'north':
                        $crop_x = intval(($src_width - $crop_w) / 2);
                        $crop_y = 0;
                        break;
                    case 'north_east':
                        $crop_x = $src_width - $crop_w;
                        $crop_y = 0;
                        break;
                    case 'west':
                        $crop_x = 0;
                        $crop_y = intval(($src_height - $crop_h) / 2);
                        break;
                    case 'center':
                        $crop_x = intval(($src_width - $crop_w) / 2);
                        $crop_y = intval(($src_height - $crop_h) / 2);
                        break;
                    case 'east':
                        $crop_x = $src_width - $crop_w;
                        $crop_y = intval(($src_height - $crop_h) / 2);
                        break;
                    case 'south_west':
                        $crop_x = 0;
                        $crop_y = $src_height - $crop_h;
                        break;
                    case 'south':
                        $crop_x = intval(($src_width - $crop_w) / 2);
                        $crop_y = $src_height - $crop_h;
                        break;
                    case 'south_east':
                        $crop_x = $src_width - $crop_w;
                        $crop_y = $src_height - $crop_h;
                        break;
                    default:
                        $crop_x = intval(($src_width - $crop_w) / 2);
                        $crop_y = intval(($src_height - $crop_h) / 2);
                }

                $image  = $this->image;
                $canvas = new Imagick();

                if ($this->type == 'gif') {
                    $images = $image->coalesceImages();
                    foreach ($images as $frame) {
                        $img = new Imagick();
                        $img->readImageBlob($frame);
                        $img->cropImage($crop_w, $crop_h, $crop_x, $crop_y);
                        $img->thumbnailImage($width, $height, true);

                        $canvas->addImage($img);
                        $canvas->setImageDelay($img->getImageDelay());
                        $canvas->setImagePage($width, $height, 0, 0);
                    }
                } else {
                    $image->cropImage($crop_w, $crop_h, $crop_x, $crop_y);
                    $image->thumbnailImage($width, $height, true);
                    $canvas->addImage($image);
                    $canvas->setImagePage($width, $height, 0, 0);
                }
                $image->destroy();
                $this->image = $canvas;
        }

    }




    // 添加水印图片
    public function add_watermark($path, $x = 0, $y = 0) {

        $watermark = new Imagick($path);

        // 默认水印位置 by jimmy 2014-4-11
        if(empty($x) && empty($y)){
            $im_w= $this->image->getImageWidth();
            $im_h = $this->image->getImageHeight();
            $wa_w = $watermark->getImageWidth();
            $wa_h = $watermark->getImageHeight();
            $x=($im_w-$wa_w)/2;
            $y=($im_h-$wa_h-50);
        }


        $draw      = new ImagickDraw();
        $draw->composite($watermark->getImageCompose(), $x, $y, $watermark->getImageWidth(), $watermark->getimageheight(), $watermark);

        if ($this->type == 'gif') {
            $image  = $this->image;
            $canvas = new Imagick();
            $images = $image->coalesceImages();
            foreach ($image as $frame) {
                $img = new Imagick();
                $img->readImageBlob($frame);
                $img->drawImage($draw);

                $canvas->addImage($img);
                $canvas->setImageDelay($img->getImageDelay());
            }
            $image->destroy();
            $this->image = $canvas;
        } else {
            $this->image->drawImage($draw);
        }
    }


    // 添加水印文字
    public function add_text($text, $x = 0, $y = 0, $angle = 0, $style = array()) {
        $draw = new ImagickDraw();
        if (isset($style['font']))
            $draw->setFont($style['font']);
        if (isset($style['font_size']))
            $draw->setFontSize($style['font_size']);
        if (isset($style['fill_color']))
            $draw->setFillColor($style['fill_color']);
        if (isset($style['under_color']))
            $draw->setTextUnderColor($style['under_color']);

        if ($this->type == 'gif') {
            foreach ($this->image as $frame) {
                $frame->annotateImage($draw, $x, $y, $angle, $text);
            }
        } else {
            $this->image->annotateImage($draw, $x, $y, $angle, $text);
        }
    }


    // 保存到指定路径
    public function save_to($path) {
        $this->image->stripImage();
        if ($this->type == 'gif') {
            $this->image->writeImages($path, true);
        } else {
            $this->image->writeImage($path);
        }
    }

    // 输出图像
    public function output($header = true) {
        if ($header)
            header('Content-type: ' . $this->type);
        echo $this->image->getImagesBlob();
    }


    public function get_width() {
        $size = $this->image->getImagePage();
        return $size['width'];
    }

    public function get_height() {
        $size = $this->image->getImagePage();
        return $size['height'];
    }

    // 设置图像类型, 默认与源类型一致
    public function set_type($type = 'png') {
        $this->type = $type;
        $this->image->setImageFormat($type);
    }

    // 获取源图像类型
    public function get_type() {
        return $this->type;
    }


    // 当前对象是否为图片
    public function is_image() {
        if ($this->image)
            return true;
        else
            return false;
    }



    public function thumbnail($width = 100, $height = 100, $fit = true) {
        $this->image->thumbnailImage($width, $height, $fit);
    } // 生成缩略图 $fit为真时将保持比例并在安全框 $width X $height 内生成缩略图片

    /*
    添加一个边框
    $width: 左右边框宽度
    $height: 上下边框宽度
    $color: 颜色: RGB 颜色 'rgb(255,0,0)' 或 16进制颜色 '#FF0000' 或颜色单词 'white'/'red'...
    */
    public function border($width, $height, $color = 'rgb(220, 220, 220)') {
        $color = new ImagickPixel();
        $color->setColor($color);
        $this->image->borderImage($color, $width, $height);
    }

    public function blur($radius, $sigma) {
        $this->image->blurImage($radius, $sigma);
    } // 模糊
    public function gaussian_blur($radius, $sigma) {
        $this->image->gaussianBlurImage($radius, $sigma);
    } // 高斯模糊
    public function motion_blur($radius, $sigma, $angle) {
        $this->image->motionBlurImage($radius, $sigma, $angle);
    } // 运动模糊
    public function radial_blur($radius) {
        $this->image->radialBlurImage($radius);
    } // 径向模糊

    public function add_noise($type = null) {
        $this->image->addNoiseImage($type == null ? imagick::NOISE_IMPULSE : $type);
    } // 添加噪点

    public function level($black_point, $gamma, $white_point) {
        $this->image->levelImage($black_point, $gamma, $white_point);
    } // 调整色阶
    public function modulate($brightness, $saturation, $hue) {
        $this->image->modulateImage($brightness, $saturation, $hue);
    } // 调整亮度、饱和度、色调

    public function charcoal($radius, $sigma) {
        $this->image->charcoalImage($radius, $sigma);
    } // 素描
    public function oil_paint($radius) {
        $this->image->oilPaintImage($radius);
    } // 油画效果

    public function flop() {
        $this->image->flopImage();
    } // 水平翻转
    public function flip() {
        $this->image->flipImage();
    } // 垂直翻转

    /**
     * 压缩图片 by jimmy 2014-4-11
     * ----------------------------------------
     */
    function compress($quality=100){
        $this->image->setImageFormat('JPEG');
        $this->image->setImageCompression(Imagick::COMPRESSION_JPEG);
        $q = $this->image->getImageCompressionQuality();
        $q = $q* $quality/100;
        if($q==0){
            $q = $quality;
        }
        $this->image->setImageCompressionQuality($q);
        $this->image->stripImage();
    }

    /**
     * 综合各个步骤输出 by jimmy 2014-4-11
     * ----------------------------------------
     */
    function output_image($srcFile,$destFile,$w=0,$h=0,$quality=100,$waterFile='',$fit="scale_fill"){
        $this->open($srcFile);
        $this->compress($quality);
        if(!empty($waterFile)){
            $this->add_watermark($waterFile);
        }
        if($w+$h>0){
            $this->resize_to($w,$h,$fit);
        }
        $this->save_to($destFile);
    }

}

1 个答案:

答案 0 :(得分:1)

首先,从你的评论&#34;甚至质量= 80,l.jpg比jm3.jpg更大#34;我认为这是错误的。以较低质量重新保存文件会产生以下结果。

119,300 = original image          
266,008 = Resized at 100% 
245,145 = Resized at 100% + despeckle
 97,536 = Resized at 90% quality
 88,342 = Resized at 90% quality with despeckle

我还添加了Imagick::despeckle,因为调整大小会引入斑点。

然而,你要问的基本问题&#34;为什么重新保存JPEG图像会增加文件大小,质量相同?&#34;

简单的答案是,因为它是一种有损格式,在将图像保存为JPEG后,新图像中可能会有更多信息。为了以相同的质量保存图像,需要对附加信息进行编码,这会占用额外的空间。

考虑一个简单的正弦波;它包含的信息非常少。然后想象我们将其保存为图像&#39;具有有损格式,可以修剪正弦波的末端,例如:

enter image description here

无论您考虑新形状的哪种方式,它都需要编码以保留新形状的更多信息,因此以100%质量重新保存它会占用原始有损保存图像的更多空间。

以较低的质量重新保存会产生较小的图像,但会迅速丢失图像中的信息,因为每次保存都是近似值的近似值。

TL:DR将PNG用于所有中间格式,并仅使用JPG作为客户的最终输出。