关于.net中获取图像缩略图的函数GetThumbnailImage的一些认识。

在很多图像软件中,打开一幅图像的时候都会显示其缩略图,在看图软件中这样的需求更为常见。如何快速的获取缩略图的信息并提供给用户查看,是个值得研究的问题。在我所研究过的图像格式中,只有JPG和PSD两种格式可能内嵌了图像自身的缩略图信息。

  在.net中,图像处理方面的内容主要是借助于GDI+的平板化API函数实现的。为了获取GDI+能支持的那几种格式的缩略图,可以调用Bitmap或者Image类的GetThumbnailImage函数。用Reflecor反编译后知道,这个函数的主要实现代码如下所示:

public Image GetThumbnailImage(int thumbWidth, int thumbHeight, GetThumbnailImageAbort callback, IntPtr callbackData)
{
IntPtr zero = IntPtr.Zero;
int status = SafeNativeMethods.Gdip.GdipGetImageThumbnail(new HandleRef(this, this.nativeImage), thumbWidth, thumbHeight, out zero, callback, callbackData);
if (status != 0)
{
throw SafeNativeMethods.Gdip.StatusException(status);
}
return CreateImageObject(zero);
}

  那么其实他也是直接调用GdipGetImageThumbnail函数。

下面我们主要通过实验说说这个函数的实质和其可应用的场合以及不应该应用的场合。

  为了测试公平,我们选用VB6作为测试语言,这有两个原因:(1)因为VB6直接调用GDI+的API函数很方便,也可以降低.net中创建各种对象所用的时间。(2)我在C#中调用Bitmap.FromFile读取文件的时间比VB6中使用同样的API要慢很多,不知道为什么。

结论1: 该函数首先判断图像是否内嵌了缩略图,如果有,则直接读取他,然后再将获得的缩略图缩放到用户调用时指定的大小。如果没有,则从图像数据中抽样填充到缩略图数据中,至于抽样算法,这个没有研究,也许是线性插值吧。

验证前的准备工作:

     (1)  一副4000*3000的数码照片,JPG格式,内嵌了缩略图信息(如何验证:可以用很多查看EXIF信息的软件查看),可在此处下载

(2)  一副4000*3000的照片,JPG格式,没有内嵌缩略图信息(如何验证:可以用很多查看EXIF信息的软件查看),可在此处下载

(3) 一副4000*3000的照片,Png格式,由于无法上传大于5MB的文件,请朋友自行用工具转换。

对上述三幅图像进行获取缩略图的操作,具体代码如下:

 '第一步:加载图像
Elapse = GetTickCount
GdipLoadImageFromFile StrPtr(FileName), Bitmap
Result = Result + "加载图像用时: " & GetTickCount - Elapse & " 毫秒。" + vbCrLf GdipGetImageWidth Bitmap, Width
GdipGetImageHeight Bitmap, Height
GetFitSize Width, Height, 600, 450, FitX, FitY, FitWidth, FitHeight '第二步:获取缩略图
Elapse = GetTickCount
GdipGetImageThumbnail Bitmap, FitWidth, FitHeight, Thumb
Result = Result + "获取缩略图用时: " & GetTickCount - Elapse & " 毫秒。" + vbCrLf GdipCreateFromHDC Hdc, Graphics '第三步:绘制缩略图
Elapse = GetTickCount
GdipDrawImageRect Graphics, Thumb, FitX, FitY, FitWidth, FitHeight
Result = Result + "绘制缩略图: " & GetTickCount - Elapse & " 毫秒。" + vbCrLf GdipDisposeImage Bitmap
GdipDisposeImage Thumb
GdipDeleteGraphics Graphics

我们依次查看结果:

             图1 : 内嵌了缩略图的JPG图像

                                   图2: 未内嵌缩略图的JPG图像

                                 图3: PNG图像

  上述缩略图的大小设置为600*450。

通过上面3个测试结果图的比较,可以明显看到:  内嵌了缩略图的JPG图像获得最后的缩略图很模糊,但是速度相当的块,而未内嵌了缩略图的JPG图像以及PNG图像获得的缩略图非常的清晰,但是耗时很多。因此我们可以初步的判断如果内嵌了缩略图,则GdipGetImageThumbnail会直接从内嵌的数据中进行缩放。为了进一步验证这一点,我生成了一副缩略图和原图完全不配套的JPG图像,来验证这一点,可从此处下载:

处理结果如下图:

可见,执行速度还是不错的,缩略图的结果却是错误的,但是和我们嵌入的缩略图却是一致的。

附带说一个问题:不知道大家注意到没有,上述代码中  GdipLoadImageFromFile 函数执行的时间都很短,而基本相同的函数在C#的Bitmap.FromFile函数中对于上述测试图像都要200多ms,不知为什么,附上Bitmap.FromFile的源码:

public static Image FromFile(string filename, bool useEmbeddedColorManagement)
{
int num;
if (!File.Exists(filename))
{
IntSecurity.DemandReadFileIO(filename);
throw new FileNotFoundException(filename);
}
filename = Path.GetFullPath(filename);
IntPtr zero = IntPtr.Zero;
if (useEmbeddedColorManagement)
{
num = SafeNativeMethods.Gdip.GdipLoadImageFromFileICM(filename, out zero);
}
else
{
num = SafeNativeMethods.Gdip.GdipLoadImageFromFile(filename, out zero);
}
if (num != 0)
{
throw SafeNativeMethods.Gdip.StatusException(num);
}
num = SafeNativeMethods.Gdip.GdipImageForceValidation(new HandleRef(null, zero));
if (num != 0)
{
SafeNativeMethods.Gdip.GdipDisposeImage(new HandleRef(null, zero));
throw SafeNativeMethods.Gdip.StatusException(num);
}
Image image = CreateImageObject(zero);
EnsureSave(image, filename, null);
return image;
}
public static Image FromFile(string filename, bool useEmbeddedColorManagement)
{
int num;
if (!File.Exists(filename))
{
IntSecurity.DemandReadFileIO(filename);
throw new FileNotFoundException(filename);
}
filename = Path.GetFullPath(filename);
IntPtr zero = IntPtr.Zero;
if (useEmbeddedColorManagement)
{
num = SafeNativeMethods.Gdip.GdipLoadImageFromFileICM(filename, out zero);
}
else
{
num = SafeNativeMethods.Gdip.GdipLoadImageFromFile(filename, out zero);
}
if (num != 0)
{
throw SafeNativeMethods.Gdip.StatusException(num);
}
num = SafeNativeMethods.Gdip.GdipImageForceValidation(new HandleRef(null, zero));
if (num != 0)
{
SafeNativeMethods.Gdip.GdipDisposeImage(new HandleRef(null, zero));
throw SafeNativeMethods.Gdip.StatusException(num);
}
Image image = CreateImageObject(zero);
EnsureSave(image, filename, null);
return image;
}

上述源码中多了一些错误处理的代码,但那些代码明显不会太耗时间。这也是我这里用VB6做测试的原因。

结论2:GetThumbnailImage不适合于做快速的图像缩放预览之类的工作,但是却是选择单开单个图像预览时的好选择。

由以上图片测试结果可以看出,GetThumbnailImage是无法胜任任意大小预览模式的,但是对于大哥图像预览时,大部分大小都只有160*120大小的预览窗口的图像,确实非常合适的。

结论3:C#下的Bitmap或者Image类的GetThumbnailImage函数不适合于做预览工作,原因就是他不如我在VB6下工作的快,特别是对于那些已经内嵌了缩略图的图像。如果是用C#做,我可能会像类似于VB中这样,直接调用GDI+的API函数。

  测试源码下载: http://files.cnblogs.com/Imageshop/ThumbNail.rar

附在的说一下: JPG的EXIF信息中的缩略图格式其实也是JPG格式,这也可以看成为什么JPG内部不一定非要内嵌缩略图的原因,不然缩略图本身格式也是JPG,那缩略图里有要嵌入缩略图....想想吧,会出现什么。

后记: 用了下美图秀秀,在打开8000*6000这样尺寸的JPG进行预览时,初次打开的速度就很快,没感觉到有延迟,并且图像质量还可以,这个的算法过程期待有高人指点下。

*****************************基本上我不提供源代码,但是我会尽量用文字把对应的算法描述清楚或提供参考文档**************************

*******************************因为靠自己的努力和实践写出来的效果才真正是自己的东西,人一定要靠自己******************************

********************************如果你真的要源代码,那你得把你的腰包变得更瘦一点,这就是偷懒的惩罚*******************************

 
 

.net中获取图像缩略图的函数GetThumbnailImage的更多相关文章

  1. asp.net中生成缩略图并添加版权实例代码

    这篇文章介绍了asp.net中生成缩略图并添加版权实例代码,有需要的朋友可以参考一下 复制代码代码如下: //定义image类的对象 Drawing.Image image,newimage; //图 ...

  2. SQL中Round(),Floor(),Ceiling()函数的浅析

    项目中的一个功能模块上用到了标量值函数,函数中又有ceiling()函数的用法,自己找了一些资料,对SQL中这几个函数做一个简单的记录,方便自己学习.有不足之处欢迎拍砖补充 1.round()函数遵循 ...

  3. avascript中的this与函数讲解

    徐某某 一个半路出家的野生程序员 javascript中的this与函数讲解 前言 javascript中没有块级作用域(es6以前),javascript中作用域分为函数作用域和全局作用域.并且,大 ...

  4. PHP中有关正则表达式的函数集锦

    之前学正则表达式的目的是想从网上抓取点小说啊,文档啊,还有获取相应的视频连接然后批量下载.当时初学PHP根本不知道PHP有专门抓包的工具,就像Simple_html_dom.php(在我的其他博文中有 ...

  5. SQL SERVER中用户定义标量函数(scalar user defined function)的性能问题

    用户定义函数(UDF)分类  SQL SERVER中的用户定义函数(User Defined Functions 简称UDF)分为标量函数(Scalar-Valued Function)和表值函数(T ...

  6. mysql中bit_count和bit_or函数的含义

    翻阅mysql手册时,看到有个示例使用了bit_or方法来去除重复的数据,一开始没看明白,后来看明白之后感觉非常巧妙.示例要实现的功能就是计算每月有几天有访问,先把示例摘录在这里. 1 2 3 4 5 ...

  7. C#中的日期处理函数

    C#中的日期处理函数 //2013年4月24日 this.TextBox6.Text = System.DateTime.Now.ToString("D"); //2013-4-2 ...

  8. 在Excel中使用频率最高的函数的功能和使用方法

    在Excel中使用频率最高的函数的功能和使用方法,按字母排序: 1.ABS函数 函数名称:ABS 主要功能:求出相应数字的绝对值. 使用格式:ABS(number) 参数说明:number代表需要求绝 ...

  9. Loadrunner中web_find和web_reg_find函数的使用与区别

    总结一下Loadrunner中的检查点函数,主要介绍两个函数:web_find()和web_reg_find():这两个函数均用于内容的查找,但两者也有本质的区别,具体介绍如下:一.web_find( ...

随机推荐

  1. Spring该讲座

    看看今天Spring. 国内搞Java开发的朋友们.对Spring一定不会陌生. Spring的历史? 谈起Spring.就会想起Ejb2.0.虽然.现实中有非常多基于Ejb2.0的成功系统,可是Ej ...

  2. CentOS修改用户密码方法

    CentOS修改用户密码方法 CentOS修改用户密码方法 1. 普通用户 a. 获取超级用户root权限 命令:su或者su -或者su - root b. passwd 用户名 2. 超级用户 a ...

  3. IOS中TableView的使用(1) -创建一个简单的tableView

    创建一个简单的tableView: #import <UIKit/UIKit.h> /*tableView 一定要遵守这两个协议: UITableViewDataSource,UITabl ...

  4. Java设计模式论述

    为何须要设计模式: 模式是做事的方法,是实现目标,研磨技术的方法.这样的对高效技术不懈追求的思想,广泛见于诸多领域,比如制作精美佳肴的过程.对于不论什么一种迈向成熟的全新技艺,身处这个行业的人都须要寻 ...

  5. PHP通过OpenSSL生成证书、密钥并且加密解密数据,以及公钥,私钥和数字签名的理解

    一.公钥加密假设一下,我找了两个数字,一个是1,一个是2.我喜欢2这个数字,就保留起来,不告诉你们(私钥),然后我告诉大家,1是我的公钥. 我有一个文件,不能让别人看,我就用1加密了.别人找到了这个文 ...

  6. Oralce 处理字符串函数

    原文:Oralce 处理字符串函数 平常我们用Oracle主要有两种字符串类型1.char始终为固定的长度,如果设置了长度小于char列的值,则Oracle会自动用空格填充的.当比较char时,Ora ...

  7. Premiere Pro CC问题集,不断更新

    1.Premiere Pro CC不好用? 是的.原因如下: 1.1 Adobe公司不注重用户体验,不注重工作流程,导致这款软件的用户体验很差,设计也很烂.对比Adobe公司当年用户体验最好的软件 F ...

  8. 快速构建Windows 8风格应用21-构建简单媒体播放器

    原文:快速构建Windows 8风格应用21-构建简单媒体播放器 本篇博文主要介绍如何构建一个简单的媒体播放器. <快速构建Windows 8风格应用20-MediaElement>博文中 ...

  9. leetcode第31题--Longest Valid Parentheses

    Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...

  10. jquery-validate的用法

    默认校验规则 (1)required:true               必输字段(2)remote:"check.php"          使用ajax方法调用check.p ...