不用第三方解码库取得图片宽高 附完整C++算法实现代码
在特定的应用场景下,有时候我们只是想获取图片的宽高,
但不想通过解码图片才取得这个信息。
预先知道图片的宽高信息,进而提速图片加载,预处理等相关操作以提升体验。
在stackoverflow有一篇相关讨论。
Get Image size WITHOUT loading image into memory
http://stackoverflow.com/questions/15800704/python-get-image-size-without-loading-image-into-memory/
不加图片到内存,而取得图像的大小。
这个技巧具有一定的实用价值,博主据此进行了相应的编码。
项目地址:https://github.com/cpuimage/image_size
实现了 常用图片格式(png,jpeg,ico,bmp,gif) 不采用第三方解码库,解析得到图像宽高的函数get_image_size_without_decode_image。
bool get_image_size_without_decode_image(const char* file_path, int*width, int*height,int *filesize);
完整代码:
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <stdbool.h>
#include <stdint.h> #if defined(_MSC_VER) && (_MSC_VER >= 1310) /*Visual Studio: A few warning types are not desired here.*/
#pragma warning( disable : 4996 ) /*VS does not like fopen, but fopen_s is not standard C so unusable here*/
#endif /*_MSC_VER */ unsigned long byteswap_ulong(unsigned long i)
{
unsigned int j;
j = (i << );
j += (i << ) & 0x00FF0000;
j += (i >> ) & 0x0000FF00;
j += (i >> );
return j;
} inline int Abs(int x) {
return (x ^ (x >> )) - (x >> );
} unsigned short byteswap_ushort(unsigned short i)
{
unsigned short j;
j = (i << );
j += (i >> );
return j;
} //Get Image size WITHOUT loading image into memory
//ref: http://stackoverflow.com/questions/15800704/python-get-image-size-without-loading-image-into-memory/ bool get_image_size_without_decode_image(const char* file_path, int*width, int*height, int * file_size)
{
bool has_image_size = false;
*height = -;
*width = -;
*file_size = -;
FILE * fp = fopen(file_path, "rb");
if (fp == NULL)
return has_image_size;
struct stat st;
char sigBuf[];
if (fstat(fileno(fp), &st) < )
{
fclose(fp);
return has_image_size;
}
else
{
*file_size = st.st_size;
}
if (fread(&sigBuf, , , fp) < )
{
fclose(fp);
return has_image_size;
}
char* png_signature = "\211PNG\r\n\032\n";
char* ihdr_signature = "IHDR";
char* gif87_signature = "GIF87a";
char* gif89_signature = "GIF89a";
char* jpeg_signature = "\377\330";
char* bmp_signature = "BM";
if ((*file_size >= ) && (memcmp(sigBuf, gif87_signature, strlen(gif87_signature)) == || memcmp(sigBuf, gif89_signature, strlen(gif89_signature)) == ))
{
// image type: gif
unsigned short* size_info = (unsigned short*)(sigBuf + );
*width = size_info[];
*height = size_info[];
has_image_size = true;
}
else if ((*file_size >= ) && (memcmp(sigBuf, png_signature, strlen(png_signature)) == && memcmp(sigBuf + , ihdr_signature, strlen(ihdr_signature)) == ))
{
// image type: png
unsigned long* size_info = (unsigned long*)(sigBuf + );
*width = byteswap_ulong(size_info[]);
*height = byteswap_ulong(size_info[]);
has_image_size = true;
}
else if ((*file_size >= ) && (memcmp(sigBuf, png_signature, strlen(png_signature)) == ))
{
// image type: old png
unsigned long* size_info = (unsigned long*)(sigBuf + );
*width = byteswap_ulong(size_info[]);
*height = byteswap_ulong(size_info[]);
has_image_size = true;
}
else if ((*file_size >= ) && (memcmp(sigBuf, jpeg_signature, strlen(jpeg_signature)) == ))
{
// image type: jpeg
fseek(fp, , SEEK_SET);
char b = ;
fread(&sigBuf, , , fp);
fread(&b, , , fp);
int w = -;
int h = -;
while (b && ((unsigned char)b & 0xff) != 0xDA) {
while (((unsigned char)b & 0xff) != 0xFF)
{
fread(&b, , , fp);
}
while (((unsigned char)b & 0xff) == 0xFF) {
fread(&b, , , fp);
}
if (((unsigned char)b & 0xff) >= 0xC0 && ((unsigned char)b & 0xff) <= 0xC3)
{
fread(&sigBuf, , , fp);
fread(&sigBuf, , , fp);
unsigned short* size_info = (unsigned short*)(sigBuf);
h = byteswap_ushort(size_info[]);
w = byteswap_ushort(size_info[]);
}
else
{
unsigned short chunk_size = ;
fread(&chunk_size, , , fp);
if (fseek(fp, byteswap_ushort(chunk_size) - , SEEK_CUR) != )
break;
}
fread(&b, , , fp);
}
if (w != - && h != -)
{
*width = w;
*height = h;
}
has_image_size = true;
}
else if ((*file_size >= ) && (memcmp(sigBuf, bmp_signature, strlen(bmp_signature)) == ))
{
// image type: bmp
unsigned int header_size = (*(sigBuf + ));
if (header_size == )
{
unsigned short* size_info = (unsigned short*)(sigBuf + );
*width = size_info[];
*height = size_info[];
}
else if (header_size >= )
{
unsigned int* size_info = (unsigned int*)(sigBuf + );
*width = size_info[];
*height = Abs((size_info[]));
}
has_image_size = true;
}
else if (*file_size >= )
{
// image type: ico
fseek(fp, , SEEK_SET);
unsigned short format = -;
unsigned short reserved = -;
fread(&reserved, , , fp);
fread(&format, , , fp);
if (reserved == && format == )
{
unsigned short num = -;
fread(&num, , , fp);
if (num > )
{
printf("this is a muti-ico file.");
}
else
{
char w = , h = ;
fread(&w, , , fp);
fread(&h, , , fp);
*width = (int)((unsigned char)w & 0xff);
*height = (int)((unsigned char)h & 0xff);
}
}
has_image_size = true;
}
if (fp != NULL)
fclose(fp);
return has_image_size;
} int main(int argc, char ** argv) { const char * pic_file = "lookmanodeps.png";
int w = ;
int h = ;
int filesize = ;
get_image_size_without_decode_image(pic_file, &w, &h, &filesize);
printf("file:%s\nfilesize:%d\nwidth:%d\nheight%d\n", pic_file, filesize, w, h);
getchar();
}
传入图片的位置,输出对应的宽高,高和宽 为-1时,就是解析失败了。
代码比较简单,不多注释了。
若有其他相关问题或者需求可以邮件联系俺探讨。
邮箱地址是:
gaozhihan@vip.qq.com
不用第三方解码库取得图片宽高 附完整C++算法实现代码的更多相关文章
- 不用第三方解码码取得图片宽高 附完整C++算法实现代码
在特定的应用场景下,有时候我们只是想获取图片的宽高, 但不想通过解码图片才取得这个信息. 预先知道图片的宽高信息,进而提速图片加载,预处理等相关操作以提升体验. 在stackoverflow有一篇相关 ...
- JS快速获取图片宽高的方法
快速获取图片的宽高其实是为了预先做好排版样式布局做准备,通过快速获取图片宽高的方法比onload方法要节省很多时间,甚至一分钟以上都有可能,并且这种方法适用主流浏览器包括IE低版本浏览器. 我们一步一 ...
- 转载:JS快速获取图片宽高的方法
快速获取图片的宽高其实是为了预先做好排版样式布局做准备,通过快速获取图片宽高的方法比onload方法要节省很多时间,甚至一分钟以上都有可能,并且这种方法适用主流浏览器包括IE低版本浏览器. 我们一步一 ...
- php 图片上传的公共方法(按图片宽高缩放或原图)
写的用于图片上传的公共方法类调用方法: $upload_name='pic';$type = 'logo_val';$file_name = 'logo_' . $user_id .create_st ...
- css3圆形头像(当图片宽高不相等时)
1.图片宽高相等,width:300px: height:300px; 把他变成宽高100px的圆形头像 img{width:100px; height:100px; border-radius:50 ...
- JS实现图片宽高的等比缩放
关于图片宽高的等比缩放,其实需求就是让图片自适应父容器的宽高,并且是等比缩放图片,使图片不变形. 例如,需要实现如下的效果: 要实现上面的效果,需要知道图片的宽高,父容器的宽高,然后计算缩放后的宽高. ...
- js不需要知道图片宽高的懒加载方法(经过实际测试,不加宽高仍然是无法正常加载的,设置height:auto,height:100%,仍然显示高度为0)
js不需要知道图片宽高的懒加载方法 懒加载是如何实现的? - 简书https://www.jianshu.com/p/e86c61468285找到一个不需要知道图片宽高的懒加载方法了(经过实际测试,不 ...
- css+background实现 图片宽高自适应,拉伸裁剪不变形
图片宽高不固定 ,一样实现自适应,拉伸裁剪不变形,适应各大兼容性. 下面咱们在网上找两张宽高不一样的照片: No.1 ...
- 写个js动态调整图片宽高 (原创)
<body style="TEXT-ALIGN: center;"> <div id="testID" style="backgro ...
随机推荐
- 记录python接口自动化测试--从excel中读取params参数传入requests请求不生效问题的解决过程(第七目)
在第六目把主函数写好了,先来运行一下主函数 从截图中可以看到,请求参数打印出来了,和excel中填写的一致 但是每个接口的返回值却都是400,提示参数没有传进去,开始不知道是什么原因(因为excel中 ...
- alpha-咸鱼冲刺day5-紫仪
总汇链接 一,合照 emmmmm.自然还是没有的. 二,项目燃尽图 三,项目进展 !!!QAQ可以做到跟数据库交互了!!!!先来撒花花!(然后继续甲板) 四,问题困难 日常啥都不会,百度真心玩一年 ...
- Linux下关闭Tomcat残留线程
ps -ef | grep tomcat kill -9 {pid}
- UWP 页面间传递参数(常见类型string、int以及自定义类型)
这是一篇很基础的,大佬就不要看了,也不要喷,谢谢
- RAID 损坏后如何对物理硬盘做完整镜像
"磁盘阵列是由很多价格较便宜的磁盘,组合成一个容量巨大的磁盘组,利用个别磁盘提供数据所产生加成效果提升整个磁盘系统效能.利用这项技术,将数据切割成许多区段,分别存放在各个硬盘上." ...
- Java面试题合集(二)
接下来几篇文章准备系统整理一下有关Java的面试题,分为基础篇,javaweb篇,框架篇,数据库篇,多线程篇,并发篇,算法篇等等,陆续更新中.其他方面如前端后端等等的面试题也在整理中,都会有的. 注: ...
- JAVA_SE基础——编码规范&代码编写规则
这次我来给大家说明下编码规范&代码编写规则 ↓ 编码规范可以帮助程序员在编程时注意一些细节问题,提高程序的可读性,让程序员能够尽快地理解新的代码,并帮助大家编写出规范的利于维护的Java代码 ...
- LeetCode & Q14-Longest Common Prefix-Easy
String Description: Write a function to find the longest common prefix string amongst an array of st ...
- DataContractJsonSerializer序列化时间类型时转换为UTC溢出问题
问题描述 如下一个实体类,含有非空时间类型属性,默认是C#的最小时间,在使用DataContractJsonSerializer将该类对象序列化成JSON时,抛出异常信息:System.Runtime ...
- Linux:日期用法,及格式定义
在shell脚本中经常会需要获取当前日期的地方,linux的系统时间在shell里是可以直接调用系统变量: 获取今天时期---`date +%Y%m%d` 或 `date +%F` 或 $(date ...