一、通过后缀名去判断。

bool IsImageByTail(const std::wstring &path)
{
std::wstring file_exten;
size_t pos = path.rfind(L'.');
if (pos == std::wstring::npos)
return false;
file_exten = path.substr(pos, std::wstring::npos);
//把file_exten转小写
for (size_t u = ; u < file_exten.length();u++)
{
if (file_exten[u] >= L'A' && file_exten[u]<='Z')
{
file_exten[u] += L'a' - L'A';
}
}
if (file_exten == L".jpg" || file_exten == L".tif"
|| file_exten == L".png" || file_exten == L".bmp"
|| file_exten == L".gif" || file_exten == L".ico")
return true;
return false;
}

优点:效率快,不用读取整个文件,无依赖,理解简单。

缺点:不准确,用户可以通过修改后缀名蒙混过关。

二、通过文件头判断

我们知道,图片文件的文件头带有图片标记信息,常见的如下:

JPEG (jpg),文件头:FFD8FF

PNG (png),文件头:89504E47

GIF (gif),文件头:47494638

TIFF (tif),文件头:49492A00

Windows Bitmap (bmp),文件头:424D

bool IsImageByHead(const std::wstring &path)
{
//读取文件首部4个字节
HANDLE hFile = CreateFile(path.c_str(), FILE_GENERIC_READ, // 打开文件,获得文件读句柄
FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, // 共享方式打开,避免其他地方需要读写此文件
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (INVALID_HANDLE_VALUE == hFile) // 文件打开失败,返回错误值
return false;
BYTE data[] = { };
DWORD readSize;
bool ok=false;
if(ReadFile(hFile, data, , &readSize, NULL))
{
if (readSize == )
{
if (data[] == 0xFF && data[]==0xD8 && data[]==0xFF)
{
ok = true;
}
else if (data[] == 0x89 && data[] == 0x50 && data[] == 0x4E && data[] == 0x47)
{
ok = true;
}
else if (data[] == 0x47 && data[] == 0x49 && data[] == 0x46 && data[] == 0x38)
{
ok = true;
}
else if (data[] == 0x49 && data[] == 0x49 && data[] == 0x2A && data[] == 0x00)
{
ok = true;
}
else if (data[] == 0x42 && data[] == 0x4D)
{
ok = true;
}
}
}
CloseHandle(hFile); // 关闭文件句柄,避免句柄泄露
return ok;
}

优点:只需要文件读取函数,实现逻辑较简单。

缺点:不准确,图片可能不完整,头部可能被伪造。

三、严格的读取文件头,匹配内部的长度、宽度与文件大小,校验值等。

这个算法偏复杂,需要对各类图片格式了如指掌,解析到位。

这里不实现了。

优点:准确无误。

缺点:实现难度复杂,需要对各类图片了如指掌。

四、通过GDI+来解析图片,判断图片的有效性

感谢GDI+,帮助我们做了解析图片的格式与内容。并提供Image类统一管理。

#include <gdiplus.h>
#pragma comment(lib,"gdiplus.lib")
bool IsImageByGDI(const std::wstring &path)
{
Gdiplus::Image image_src(path.c_str());
Gdiplus::Status status = image_src.GetLastStatus();
if (status != Gdiplus::Ok)
{
return false;
}
GUID guid;
if (image_src.GetRawFormat(&guid) != Gdiplus::Ok)
{
return false;
}
if (guid == Gdiplus::ImageFormatGIF || guid == Gdiplus::ImageFormatJPEG || guid == Gdiplus::ImageFormatPNG
|| guid == Gdiplus::ImageFormatBMP || guid == Gdiplus::ImageFormatIcon || guid == Gdiplus::ImageFormatTIFF)
{
return true;
}
return false;
}

优点:准确无误、实现简单。

缺点:需要依赖GDI+库,效率比较低。

当然,如果我们不需要准确的判断的话。上述函数可以结合使用。

如:bool isPic  =  IsImageByTail  ||   IsImageByHead。可以解决无后缀的图片的判断,并且依赖较低。

bool isPic  =  IsImageByTail  ||   IsImageByGDI 。可以通过后缀名提前过滤一遍,加快效率。

另外:上述函数稍微修改下,也可以返回具体的图片类型(到底是jpg呢还是png呢)。

另外:GDI+真的是不错的东西!

本来想更新一个GDI+系列的,感觉关注的人不多,动力不足。

从零开始学习GDI+ (一)我的第一个GDI+程序

Windows C++ 判断文件是否是图片格式的方法。的更多相关文章

  1. java判断文件是否为图片

    /** * 判断文件是否为图片<br> * <br> * @param pInput 文件名<br> * @param pImgeFlag 判断具体文件类型< ...

  2. android判断文件是否是图片文件的方法

    判断一个文件是否是图片文件的方法,采用BitmapFactory去decode然后根据返回的Options参数来确定: public static boolean isImageFile(String ...

  3. oracle 函数判断字符串是否包含图片格式

    首先是写一个分割字符串的函数,返回table类型 CREATE OR REPLACE FUNCTION fn_split (p_str IN VARCHAR2, p_delimiter IN VARC ...

  4. TensorFlow笔记五:将cifar10数据文件复原成图片格式

    cifar10数据集(http://www.cs.toronto.edu/~kriz/cifar-10-python.tar.gz)源格式是数据文件,因为训练需要转换成图片格式 转换代码: 注意文件路 ...

  5. Python判断文件是否存在的三种方法

    通常在读写文件之前,需要判断文件或目录是否存在,不然某些处理方法可能会使程序出错.所以最好在做任何操作之前,先判断文件是否存在. 这里将介绍三种判断文件或文件夹是否存在的方法,分别使用os模块.Try ...

  6. Python判断文件是否存在的三种方法【转】

    转:http://www.cnblogs.com/jhao/p/7243043.html 通常在读写文件之前,需要判断文件或目录是否存在,不然某些处理方法可能会使程序出错.所以最好在做任何操作之前,先 ...

  7. Python 判断文件是否存在的三种方法

    通常在读写文件之前,需要判断文件或目录是否存在,不然某些处理方法可能会使程序出错.所以最好在做任何操作之前,先判断文件是否存在. 这里将介绍三种判断文件或文件夹是否存在的方法,分别使用os模块.Try ...

  8. ubuntu 转换图片格式的方法(sam2p, imagemagick)

    (1) 终端:sudo apt-get install sam2p sam2p [原图片名.格式] [目标图片名.格式] 即可在同一目录下生成目标图片格式 (2) 终端: sudo apt-get i ...

  9. windows系统下压缩文件成tar.gz格式的方法

    tar.gz 是linux和unix下面比较常用的格式,几个命令就可以把文件压缩打包成tar.gz格式,然而这种格式在windows并不多见,WinRAR.WinZip等主流压缩工具可以释放解开,却不 ...

随机推荐

  1. Oracle存储结构-段区块

    段 一个段建立以后首先会分配一个区,区中包括含8个块,这时执行insert插入数据,当这个区写满后,会在分配一个区 1.一个段建立以后,Oracle如何给段分配区? 2.段分配到区以后,有了空闲空间, ...

  2. 使用oracle Sqlplus中上下键出现乱码的问题

    安装rlwrap,前提是安装readline和readline-devel yum list | grep readlineyum install -y readline.x86_64 readlin ...

  3. MaxCompute - ODPS重装上阵 第六弹 - User Defined Type

    MaxCompute(原ODPS)是阿里云自主研发的具有业界领先水平的分布式大数据处理平台, 尤其在集团内部得到广泛应用,支撑了多个BU的核心业务. MaxCompute除了持续优化性能外,也致力于提 ...

  4. mysql 递归查找所有子节点

    select dept_id from ( select t1.dept_id,t1.parent_id, if(find_in_set(parent_id, @pids) > 0, @pids ...

  5. sql 导入文件

    zai SQLQuery4.sql 文件中 --BULK INSERT Table_1 from 'D:\aaaa#azzz.txt' with(fieldterminator=',',rowterm ...

  6. eclipse切换 package explorer

  7. react-router中,<switch>

    有<Switch>标签,则其中的<Route>在路径相同的情况下,只匹配第一个,这个可以避免重复匹配: 无<Switch>标签,则其中的<Route>在 ...

  8. 论文阅读:Fast, Scalable, and Programmable Packet Scheduler in Hardware

    摘要: 随着链接速度的提高和CPU扩展速度的放缓,软件中的数据包调度会导致较低的精度和较高的CPU利用率. 通过将数据包调度卸载到诸如NIC之类的硬件,可以潜在地克服这些缺点.然而为了保持软件分组调度 ...

  9. vue开发多页面应用

    1.添加多页面配置 在工程根路径下(package.json同目录)添加添加vue.config.js配置文件,内容为: module.exports = { pages: { index: 'src ...

  10. linux下文件权限更改(转载)

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明.本文链接:https://blog.csdn.net/qq_33571752/article/d ...