C#无损高质量压缩图片实现代码

2024-06-11 21:32

本文主要是介绍C#无损高质量压缩图片实现代码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近,项目上涉及到了图像压缩,发现原有的图像压缩功能,虽然保证了图像的大小300K以内,但是压缩后的图像看的不在清晰,并且,限定了图片的Height或者是Width。

在CSDN上看到了一个压缩算法:C#无损高质量压缩图片代码

进过测试这个算法,发现,将原始图像的大小进行对半处理,然后迭代跳转压缩质量参数,可以得到不错的效果。

修改后的算法如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

/// <summary>

/// 无损压缩图片

/// </summary>

/// <param name="sFile">原图片地址</param>

/// <param name="dFile">压缩后保存图片地址</param>

/// <param name="flag">压缩质量(数字越小压缩率越高)1-100</param>

/// <param name="size">压缩后图片的最大大小</param>

/// <param name="sfsc">是否是第一次调用</param>

/// <returns></returns>

public static bool CompressImage(string sFile, string dFile, int flag = 90, int size = 300, bool sfsc = true)

{

Image iSource = Image.FromFile(sFile);

ImageFormat tFormat = iSource.RawFormat;

//如果是第一次调用,原始图像的大小小于要压缩的大小,则直接复制文件,并且返回true

FileInfo firstFileInfo = new FileInfo(sFile);

if (sfsc == true && firstFileInfo.Length < size * 1024)

{

firstFileInfo.CopyTo(dFile);

return true;

}

int dHeight = iSource.Height / 2;

int dWidth = iSource.Width / 2;

int sW = 0, sH = 0;

//按比例缩放

Size tem_size = new Size(iSource.Width, iSource.Height);

if (tem_size.Width > dHeight || tem_size.Width > dWidth)

{

if ((tem_size.Width * dHeight) > (tem_size.Width * dWidth))

{

sW = dWidth;

sH = (dWidth * tem_size.Height) / tem_size.Width;

}

else

{

sH = dHeight;

sW = (tem_size.Width * dHeight) / tem_size.Height;

}

}

else

{

sW = tem_size.Width;

sH = tem_size.Height;

}

Bitmap ob = new Bitmap(dWidth, dHeight);

Graphics g = Graphics.FromImage(ob);

g.Clear(Color.WhiteSmoke);

g.CompositingQuality = System.Drawing.Drawing2D.CompositingQuality.HighQuality;

g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;

g.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.HighQualityBicubic;

g.DrawImage(iSource, new Rectangle((dWidth - sW) / 2, (dHeight - sH) / 2, sW, sH), 0, 0, iSource.Width, iSource.Height, GraphicsUnit.Pixel);

g.Dispose();

//以下代码为保存图片时,设置压缩质量

EncoderParameters ep = new EncoderParameters();

long[] qy = new long[1];

qy[0] = flag;//设置压缩的比例1-100

EncoderParameter eParam = new EncoderParameter(System.Drawing.Imaging.Encoder.Quality, qy);

ep.Param[0] = eParam;

try

{

ImageCodecInfo[] arrayICI = ImageCodecInfo.GetImageEncoders();

ImageCodecInfo jpegICIinfo = null;

for (int x = 0; x < arrayICI.Length; x++)

{

if (arrayICI[x].FormatDescription.Equals("JPEG"))

{

jpegICIinfo = arrayICI[x];

break;

}

}

if (jpegICIinfo != null)

{

ob.Save(dFile, jpegICIinfo, ep);//dFile是压缩后的新路径

FileInfo fi = new FileInfo(dFile);

if (fi.Length > 1024 * size)

{

flag = flag - 10;

CompressImage(sFile, dFile, flag, size, false);

}

}

else

{

ob.Save(dFile, tFormat);

}

return true;

}

catch

{

return false;

}

finally

{

iSource.Dispose();

ob.Dispose();

}

}

效果图如下:

第一张的大小是2.82M,尺寸是3680*4640。

 

第二张的大小是274KB,尺寸是1740*2320,清晰度方面还是不错的。

 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

 

 

 

/// <summary>

        /// 图像缩略图处理

        /// </summary>

        /// <param name="bytes">图像源数据</param>

        /// <param name="compression">压缩质量 1-100</param>

        /// <param name="thumbWidth">缩略图的宽</param>

        /// <param name="thumbHeight">缩略图的高</param>

        /// <returns></returns>

        public static byte[] ConvertToThumbnail(byte[] bytes, int compression = 100, int thumbWidth = 0, int thumbHeight = 0)

        {

            byte[] bs = null;

 

            try

            {

                if (bytes != null)

                {

                    using (MemoryStream ms = new MemoryStream(bytes))

                    {

                        using (Bitmap srcimg = new Bitmap(ms))

                        {

                            if (thumbWidth == 0 && thumbHeight == 0)

                            {

                                thumbWidth = srcimg.Width;

                                thumbHeight = srcimg.Height;

                            }

                            using (Bitmap dstimg = new Bitmap(thumbWidth, thumbHeight))//图片压缩质量

                            {

                                //从Bitmap创建一个System.Drawing.Graphics对象,用来绘制高质量的缩小图。

                                using (Graphics gr = Graphics.FromImage(dstimg))

                                {

                                    //把原始图像绘制成上面所设置宽高的缩小图

                                    Rectangle rectDestination = new Rectangle(0, 0, thumbWidth, thumbHeight);

                                    gr.Clear(Color.WhiteSmoke);

                                    gr.CompositingQuality = CompositingQuality.HighQuality;

                                    gr.SmoothingMode = SmoothingMode.HighQuality;

                                    gr.InterpolationMode = InterpolationMode.HighQualityBicubic;

                                    gr.DrawImage(srcimg, rectDestination, 0, 0, srcimg.Width, srcimg.Height, GraphicsUnit.Pixel);

 

                                    EncoderParameters ep = new EncoderParameters(1);

                                    ep.Param[0] = new EncoderParameter(System.Drawing.Imaging.Encoder.Quality, compression);//设置压缩的比例1-100

                                    ImageCodecInfo[] arrayICI = ImageCodecInfo.GetImageEncoders();

                                    ImageCodecInfo jpegICIinfo = arrayICI.FirstOrDefault(t => t.FormatID == System.Drawing.Imaging.ImageFormat.Png.Guid);

                                    using (MemoryStream dstms = new MemoryStream())

                                    {

                                        if (jpegICIinfo != null)

                                        {

                                            dstimg.Save(dstms, jpegICIinfo, ep);

                                        }

                                        else

                                        {

                                            dstimg.Save(dstms, System.Drawing.Imaging.ImageFormat.Png);//保存到内存里

                                        }

                                        bs = new Byte[dstms.Length];

                                        dstms.Position = 0;

                                        dstms.Read(bs, 0, bs.Length);

                                    }

                                }

                            }

                        }

                    }

                }

            }

            catch (Exception ex)

            {

                LogManager.DefaultLogger.Error(LogConvert.ParseWebEx(ex), string.Concat("ConvertToThumbnail error.", thumbWidth, " ", thumbHeight));

            }

            return bs;

        }

 

有时候需要一些图片,但是太大了,又有限制,所以想到如何把一张图片的内存大小给缩小,这样就OK了。

1.代码如下:

借鉴:https://docs.microsoft.com/zh-cn/dotnet/api/system.drawing.imaging.encoderparameter?view=netframework-4.6

using System;
using System.Drawing;
using System.Drawing.Imaging;
//这里面有示例文档代码
//https://docs.microsoft.com/zh-cn/dotnet/api/system.drawing.imaging.encoderparameter?view=netframework-4.6
namespace WindowsFormsApp1 {
    class MyTool {
        //根据原图,得到压缩图片并保存在桌面,返回压缩图路径
        public static String compressImage(String  bmpPath , int quality) {
            //原图路径
            Bitmap bmp = new Bitmap(bmpPath);
            ImageCodecInfo codecInfo = GetEncoder(bmp.RawFormat); //图片编解码信息
            System.Drawing.Imaging.Encoder encoder = System.Drawing.Imaging.Encoder.Quality;
            EncoderParameters encoderParameters = new EncoderParameters(1);
            EncoderParameter encoderParameter = new EncoderParameter(encoder , quality);
            encoderParameters.Param[0] = encoderParameter; //编码器参数
            //压缩图路径
            ImageFormat format = bmp.RawFormat;
            String newFilePath = String.Empty; //压缩图所在路径
            // Guid.NewGuid().ToString()
            //GUID 是一个 128 位整数 (16 个字节),它可用于跨所有计算机和网络中,任何唯一标识符是必需的。 此类标识符具有重复的可能性非常小
            String deskPath = Environment.GetFolderPath(Environment.SpecialFolder.Desktop);
            if (format.Equals(ImageFormat.Jpeg)) {
                newFilePath = deskPath + @"\" + Guid.NewGuid().ToString() + ".jpeg";
            }
            else if (format.Equals(ImageFormat.Png)) {
                newFilePath = deskPath + @"\" + Guid.NewGuid().ToString() + ".png";
            }
            else if (format.Equals(ImageFormat.Bmp)) {
                newFilePath = deskPath + @"\" + Guid.NewGuid().ToString() + ".bmp";
            }
            else if (format.Equals(ImageFormat.Gif)) {
                newFilePath = deskPath + @"\" + Guid.NewGuid().ToString() + ".gif";
            }
            else if (format.Equals(ImageFormat.Icon)) {
                newFilePath = deskPath + @"\" + Guid.NewGuid().ToString() + ".icon";
            }
            else {
                newFilePath = deskPath + @"\" + Guid.NewGuid().ToString() + ".jpg";
            }
            bmp.Save(newFilePath , codecInfo , encoderParameters); //保存压缩图
            return newFilePath; //返回压缩图路径
        }
 
        private static ImageCodecInfo GetEncoder(ImageFormat rawFormat) {
            ImageCodecInfo[] codecs = ImageCodecInfo.GetImageDecoders();
            foreach (ImageCodecInfo codec in codecs) {
                if (codec.FormatID == rawFormat.Guid) {
                    return codec;
                }
            }
            return null;
        }
    }
}
然后调用:

private void button1_Click_1(object sender, EventArgs e) {
            using (OpenFileDialog ofd = new OpenFileDialog()) {
                ofd.Filter = "图片|*.jpg;*.png;*.gif;*.jpeg;*.bmp";
                if (ofd.ShowDialog() == DialogResult.OK) {
                    String b = MyTool.compressImage(ofd.FileName , 50);
                }
            }
        }
2.结果

我试了把一张大于2M的图片压缩成了400多KB。
 

 

这篇关于C#无损高质量压缩图片实现代码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1052327

相关文章

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

C# 中变量未赋值能用吗,各种类型的初始值是什么

对于一个局部变量,如果未赋值,是不能使用的 对于属性,未赋值,也能使用有系统默认值,默认值如下: 对于 int 类型,默认值是 0;对于 int? 类型,默认值是 null;对于 bool 类型,默认值是 false;对于 bool? 类型,默认值是 null;对于 string 类型,默认值是 null;对于 string? 类型,哈哈,没有这种写法,会出错;对于 DateTime 类型,默

uniapp接入微信小程序原生代码配置方案(优化版)

uniapp项目需要把微信小程序原生语法的功能代码嵌套过来,无需把原生代码转换为uniapp,可以配置拷贝的方式集成过来 1、拷贝代码包到src目录 2、vue.config.js中配置原生代码包直接拷贝到编译目录中 3、pages.json中配置分包目录,原生入口组件的路径 4、manifest.json中配置分包,使用原生组件 5、需要把原生代码包里的页面修改成组件的方

JAVA读取MongoDB中的二进制图片并显示在页面上

1:Jsp页面: <td><img src="${ctx}/mongoImg/show"></td> 2:xml配置: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

17.用300行代码手写初体验Spring V1.0版本

1.1.课程目标 1、了解看源码最有效的方式,先猜测后验证,不要一开始就去调试代码。 2、浓缩就是精华,用 300行最简洁的代码 提炼Spring的基本设计思想。 3、掌握Spring框架的基本脉络。 1.2.内容定位 1、 具有1年以上的SpringMVC使用经验。 2、 希望深入了解Spring源码的人群,对 Spring有一个整体的宏观感受。 3、 全程手写实现SpringM

通过SSH隧道实现通过远程服务器上外网

搭建隧道 autossh -M 0 -f -D 1080 -C -N user1@remotehost##验证隧道是否生效,查看1080端口是否启动netstat -tuln | grep 1080## 测试ssh 隧道是否生效curl -x socks5h://127.0.0.1:1080 -I http://www.github.com 将autossh 设置为服务,隧道开机启动

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

vue项目集成CanvasEditor实现Word在线编辑器

CanvasEditor实现Word在线编辑器 官网文档:https://hufe.club/canvas-editor-docs/guide/schema.html 源码地址:https://github.com/Hufe921/canvas-editor 前提声明: 由于CanvasEditor目前不支持vue、react 等框架开箱即用版,所以需要我们去Git下载源码,拿到其中两个主

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std