1、經過後綴名去判斷。
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 = 0; 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; }
優勢:效率快,不用讀取整個文件,無依賴,理解簡單。html
缺點:不許確,用戶能夠經過修改後綴名矇混過關。 算法
2、經過文件頭判斷
咱們知道,圖片文件的文件頭帶有圖片標記信息,常見的以下:函數
JPEG (jpg),文件頭:FFD8FF學習
PNG (png),文件頭:89504E47 ui
GIF (gif),文件頭:47494638 spa
TIFF (tif),文件頭:49492A00 code
Windows Bitmap (bmp),文件頭:424Dorm
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[4] = { 0 }; DWORD readSize; bool ok=false; if(ReadFile(hFile, data, 4, &readSize, NULL)) { if (readSize == 4) { if (data[0] == 0xFF && data[1]==0xD8 && data[2]==0xFF) { ok = true; } else if (data[0] == 0x89 && data[1] == 0x50 && data[2] == 0x4E && data[3] == 0x47) { ok = true; } else if (data[0] == 0x47 && data[1] == 0x49 && data[2] == 0x46 && data[3] == 0x38) { ok = true; } else if (data[0] == 0x49 && data[1] == 0x49 && data[2] == 0x2A && data[3] == 0x00) { ok = true; } else if (data[0] == 0x42 && data[1] == 0x4D) { ok = true; } } } CloseHandle(hFile); // 關閉文件句柄,避免句柄泄露 return ok; }
優勢:只須要文件讀取函數,實現邏輯較簡單。htm
缺點:不許確,圖片可能不完整,頭部可能被僞造。blog
3、嚴格的讀取文件頭,匹配內部的長度、寬度與文件大小,校驗值等。
這個算法偏複雜,須要對各種圖片格式瞭如指掌,解析到位。
這裏不實現了。
優勢:準確無誤。
缺點:實現難度複雜,須要對各種圖片瞭如指掌。
4、經過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+系列的,感受關注的人很少,動力不足。
原文出處:https://www.cnblogs.com/xuhuajie/p/11721912.html