/*格式组成典型的BMP图像文件由四部分组成:
1:位图头文件数据结构,它包含BMP图像文件的类型、显示内容等信息;
2:位图信息数据结构,它包含有BMP图像的宽、高、压缩方法,以及定义颜色等信息;
3:调色板,这个部分是可选的,有些位图需要调色板,有些位图,比如真彩色图(24位的BMP)就不需要调色板;
4:位图数据,这部分的内容根据BMP位图使用的位数不同而不同,在24位图中直接使用RGB,而其他的小于24位的使用调色板中颜色索引值。
*/
function imagebmp(&$im, $filename = '', $bit = 8, $compression = 0)
{
if (!in_array($bit, array(1, 4, 8, 16, 24, 32)))
{
$bit = 8; //记录每个像素所占计算机字节的位数,默认为8位
}
else if ($bit == 32) // todo:32 bit
{
$bit = 24;
}
$bits = pow(2, $bit); //表示待创建的图像一共由$bits种颜色组成 // 将图像调整为调色板图像,便于以后绘图,如果是24位BMP位图,那么就不需要这一步
/*调色板是被保存在一个RGBQUAD结构的数组中,该结构指出了每一种颜色的红、绿、蓝的分量值。
*位数组中的每一个索引都对应于一个调色板项(即一个RGBQUAD结构),应用程序将根据这种对
*应关系,将像素索引值转换为像素RGB值(真实的像素颜色)
* */
imagetruecolortopalette($im, true, $bits);
$width = imagesx($im);
$height = imagesy($im);
$colors_num = imagecolorstotal($im); //返回的一般都是256
// 颜色索引
$rgb_quad = '';
if ($bit <= 8)//仅针对1,4,8位BMP位图建立颜色索引
{ for ($i = 0; $i < $colors_num; $i ++)
{
$colors = imagecolorsforindex($im, $i);//每一幅图的索引表都是不一样的!
$rgb_quad .= chr($colors['blue']) . chr($colors['green']) . chr($colors['red']) . "\0";
} // 位图数据
$bmp_data = ''; // 非压缩,即对RGB值不进行压缩,读取时顺序为BGR,高位在前,低位在后
if ($compression == 0 || $bit < 8)
{
$compression = 0; // 每行字节数必须为4的倍数,补齐。
$extra = '';
$padding = 32 - ( $width * $bit ) % 32 ;
if ($padding % 32 != 0)
{
$extra = str_repeat("\0", $padding);
} for ($j = $height - 1; $j >= 0; $j --)
{
$i = 0;
while ($i < $width)
{
$bin = 0;
$limit = $width - $i < 8 / $bit ? (8 / $bit - $width + $i) * $bit : 0; for ($k = 8 - $bit; $k >= $limit; $k -= $bit)
{
$index = imagecolorat($im, $i, $j);
$bin |= $index << $k;
$i ++;
} $bmp_data .= chr($bin);
} $bmp_data .= $extra;
}
}
// RLE8 压缩
else if ($compression == 1 && $bit == 8)
{
for ($j = $height - 1; $j >= 0; $j --)
{
$last_index = "\0";
$same_num = 0;
for ($i = 0; $i <= $width; $i ++)
{
$index = imagecolorat($im, $i, $j);
if ($index !== $last_index || $same_num > 255)
{
if ($same_num != 0)
{
$bmp_data .= chr($same_num) . chr($last_index);
} $last_index = $index;
$same_num = 1;
}
else
{
$same_num ++;
}
} $bmp_data .= "\0\0";
} $bmp_data .= "\0\1";
}
$size_quad = strlen($rgb_quad);
$size_data = strlen($bmp_data);
}
else
{
// 每行字节数必须为4的倍数,补齐。
$extra = '';
$padding = 4 - ($width * ($bit / 8)) % 4;
if ($padding % 4 != 0)
{
$extra = str_repeat("\0", $padding);
}
// 位图数据
$bmp_data = '';
for ($j = $height - 1; $j >= 0; $j --)
{
for ($i = 0; $i < $width; $i ++)
{
$index = imagecolorat($im, $i, $j);
$colors = imagecolorsforindex($im, $index);
if ($bit == 16)
{
$bin = 0 << $bit; $bin |= ($colors['red'] >> 3) << 10;
$bin |= ($colors['green'] >> 3) << 5;
$bin |= $colors['blue'] >> 3; $bmp_data .= pack("v", $bin);
}
else
{
$bmp_data .= pack("c*", $colors['blue'], $colors['green'], $colors['red']);
} // todo: 32bit;
}
$bmp_data .= $extra;
}
$size_quad = 0;
$size_data = strlen($bmp_data);
$colors_num = 0;
} // 位图文件头
$file_header = "BM" . pack("V3", 54 + $size_quad + $size_data, 0, 54 + $size_quad); // 位图信息头
$info_header = pack("V3v2V*", 0x28, $width, $height, 1, $bit, $compression, $size_data, 0, 0, $colors_num, 0); // 写入文件
if ($filename != '')
{
$fp = fopen($filename, "wb");
fwrite($fp, $file_header);
fwrite($fp, $info_header);
if($rgb_quad!=='')
fwrite($fp, $rgb_quad);
fwrite($fp, $bmp_data);
fclose($fp);
return true;
} // 浏览器输出
header("Content-Type: image/bmp");
echo $file_header . $info_header;
echo $rgb_quad;
echo $bmp_data;
return true;
}
/**
* 類似GD庫打開圖片, 打開bmp格式圖片
* $file : 圖片路徑
* 注意事项:不能读取1位和4位BMP图像,16位BMP图像显示有问题
*/ function imagecreatefrombmp($file)
{
static $CurrentBit;
static $echoMode=false;//仅针对4位的图片
$f = fopen($file,"r");
$Header = fread($f,2);
if($Header == "BM")
{
$Size = freaddword($f);// 位图文件的大小,以字节为单位(3-6字节)
$Reserved1 = freadword($f);// 位图文件保留字,必须为0(7-8字节)
$Reserved2 = freadword($f);// 位图文件保留字,必须为0(9-10字节)
$FirstByteOfImage = freaddword($f);// 位图数据的起始位置,以相对于位图
//(11-14字节)文件头的偏移量表示,以字节为单位
$SizeBITMAPINFOHEADER = freaddword($f);// 本结构所占用字节数(15-18字节)
$Width = freaddword($f);// 位图的宽度,以像素为单位(19-22字节)
$Height = freaddword($f);// 位图的高度,以像素为单位(23-26字节)
$biPlanes = freadword($f);// 目标设备的级别,必须为1(27-28字节)
$biBitCount = freadword($f);// 每个像素所需的位数,必须是1(双色),(29-30字节)
// 4(16色),8(256色)16(高彩色)或24(真彩色)之一
$RLECompression = freaddword($f);// 位图压缩类型,必须是 0(不压缩),(31-34字节)
// 1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一
$WidthxHeight = freaddword($f);// 位图的大小,以字节为单位(35-38字节)
$biXPelsPerMeter = freaddword($f);// 位图水平分辨率,每米像素数(39-42字节)
$biYPelsPerMeter = freaddword($f);// 位图垂直分辨率,每米像素数(43-46字节)
$NumberOfPalettesUsed = freaddword($f);// 位图实际使用的颜色表中的颜色数(47-50字节)
$NumberOfImportantColors = freaddword($f);// 位图显示过程中重要的颜色数(51-54字节)
/*
* 位图数据记录了位图的每一个像素值,记录顺序是在扫描行内是从左到右,扫描行之间是从下到上。位图的一个像素值所占的字节数:
* 当biBitCount=1时,8个像素占1个字节;
* 当biBitCount=4时,2个像素占1个字节;
* 当biBitCount=8时,1个像素占1个字节;
* 当biBitCount=24时,1个像素占3个字节;
* Windows规定一个扫描行所占的字节数必须是
* 4的倍数(即以long为单位),不足的以0填充,
*/
if($biBitCount < 16)//1位和4位图像仍不能读取。
{
$img = imagecreate($Width, $Height);
//读取颜色索引表,这个只在24位以下BMP图片中才需要
$Colors = pow(2, $biBitCount);
for($p=0; $p<$Colors; $p++)
{//$p代表索引值
$B = freadbyte($f);
$G = freadbyte($f);
$R = freadbyte($f);
$Reserved = freadbyte($f);
$Palette[] = imagecolorallocate($img, $R, $G, $B);
}
if($RLECompression == 0)
{
$Zbytek = (4-ceil(($Width/(8/$biBitCount)))%4)%4;
for($y=$Height-1; $y>=0; $y--)
{
$CurrentBit = 0;
for($x=0; $x<$Width; $x++)
{
$C = freadbits($f, $biBitCount,$echoMode);
//从索引表里找到相应的RGB颜色代码去填充图像该点的颜色
imagesetpixel($img, $x, $y, $Palette[$C]);
}
for($g=0; $g<$Zbytek; $g++)
{
freadbyte($f);
}
}
}
}
if($RLECompression == 1) //$BI_RLE8
{
$y = $Height;
$pocetb = 0;
while(true)
{
$y--;
$prefix = freadbyte($f);
$suffix = freadbyte($f);
$pocetb += 2;
$echoit = false;
if($echoit)
{
echo "Prefix: $prefix Suffix: $suffix<BR>";
}
if(($prefix == 0) && ($suffix == 1))
{
break;
}
if(feof($f))
{
break;
}
while(!(($prefix == 0) && ($suffix == 0)))
{
if($prefix==0)
{
$pocet = $suffix;
$Data .= fread($f,$pocet);
$pocetb += $pocet;
if($pocetb%2 == 1)
{
freadbyte($f);
$pocetb++;
}
}
if($prefix > 0)
{
$pocet = $prefix;
for($r=0; $r<$pocet; $r++)
{
$Data.=chr($suffix);
}
}
$prefix = freadbyte($f);
$suffix = freadbyte($f);
$pocetb += 2;
if($echoit)
{
echo "Prefix: $prefix Suffix: $suffix<BR>";
}
}
for($x=0; $x<strlen($Data); $x++)
{
imagesetpixel($img,$x,$y,$Palette[ord($Data[$x])]);
}
$Data="";
}
}
if($RLECompression == 2)
{
$y = $Height;
$pocetb = 0;
while(true)
{
$y--;
$prefix = freadbyte($f);
$suffix = freadbyte($f);
$pocetb += 2;
$echoit = false;
if($echoit)
{
echo "Prefix: $prefix Suffix: $suffix<BR>";
}
if(($prefix == 0) && ($suffix==1))
{
break;
}
if(feof($f))
{
break;
}
while(!(($prefix == 0) && ($suffix == 0)))
{
if($prefix == 0)
{
$pocet = $suffix;
$CurrentBit = 0;
for($h=0; $h<$pocet; $h++)
{
$Data.=chr(freadbits($f,4,$echoMode));
}
if($CurrentBit != 0)
{
freadbits($f,4,$echoMode);//不知道这里的代码对不对
}
$pocetb += ceil(($pocet/2));
if($pocetb%2 == 1)
{
freadbyte($f);
$pocetb++;
}
}
if($prefix > 0)
{
$pocet = $prefix;
$i = 0;
for($r=0; $r<$pocet; $r++)
{
if($i%2 == 0)
{
$Data .= chr($suffix%16);
}
else
{
$Data .= chr(floor($suffix/16));
}
$i++;
}
}
$prefix = freadbyte($f);
$suffix = freadbyte($f);
$pocetb += 2;
if($echoit)
{
echo "Prefix: $prefix Suffix: $suffix<BR>";
}
}
for($x=0; $x<strlen($Data); $x++)
{
imagesetpixel($img, $x, $y, $Palette[ord($Data[$x])]);
}
$Data="";
}
}
if($biBitCount >= 16)
{
$img = imagecreatetruecolor($Width, $Height);
$Zbytek = $Width%4;//微软规定每行必须是4的倍数,所以不足4的字节全部用0填充。
for($y=$Height-1; $y>=0; $y--)
{
for($x=0; $x<$Width; $x++)
{
if($biBitCount == 16){
$t1 = freadbyte($f);
$t2 = freadbyte($f);//freadbyte($f);freadbyte($f);
$B = $t1 >> 3 & 0x1f;
$G = ((( $t1 & 0x07 ) << 3)&0xff | (( $t2 & 0xe0 ) >> 5)&0xff) & 0xff;
$R = $t2 & 0x3e;
}else{
$B = freadbyte($f);
$G = freadbyte($f);
$R = freadbyte($f);
}
if($biBitCount == 32)freadbyte($f);
$color = imagecolorexact($img, $R, $G, $B);
if($color==-1)
{
$color=imagecolorallocate($img, $R, $G, $B);
}
imagesetpixel($img, $x, $y, $color);
}
for($z=0; $z<$Zbytek; $z++)
{
freadbyte($f);
}
}
}
return $img;
}
fclose($f);
} function freadbyte($f)
{
return ord(fread($f, 1));
}
function freadbits($f, $biBitCount,&$echoMode='')
{ $str='';
if($biBitCount==4){
if($echoMode==false){
$echoMode=true;
$str = freadbyte($f);
fseek($f, -1,SEEK_CUR);
$str = $str >> 4;
return $str & 0x0f;
}else{
$str = freadbyte($f);
$echoMode = false;
return $str & 0x0f;
}
}
if($biBitCount == 8){
$str=freadbyte($f);
return $str & 0xff;
}
} function freadword($f)
{
$b1 = freadbyte($f);
$b2 = freadbyte($f);
return $b2*256 + $b1;
} function freaddword($f)
{
$b1 = freadword($f);
$b2 = freadword($f);
return $b2*65536 + $b1;
}

imagecreatefrombmp、imagebmp php处理bmp文件的更多相关文章

  1. Matlab 读取文件夹中所有的bmp文件

    将srcimg文件下的bmp文件转为jpg图像,存放在dstimg文件夹下 str = 'srcimg'; dst = 'dstimg'; file=dir([str,'\*.bmp']); :len ...

  2. 图像预处理第9步:存为.bmp文件

    //图像预处理第9步:将最终标准化后的字符图像分为单个单个的HDIB保存,并存为.bmp文件 void CChildView::OnImgprcToDibAndSave() { unsigned ch ...

  3. Bitmap文件格式+生成一个BMP文件

    Bitmap的文件格式: #define UINT16 unsigned short #define DWORD unsigned int #define WORD short #define LON ...

  4. [代码片段]读取BMP文件

    文件名mybmp.c //实现了读取24位BMP文件,还有一些测试信息 //定义mybmp.c里面用到的函数和相关头文件.常量 //参考了网友sailinghz在CSDN论坛的帖子:http://bb ...

  5. 将HDC保存为BMP文件

    HDC在MSDN中的全称为:The handle of device context.通常,我们都是用来做相应的显示操作.        熟悉WIN32的朋友对于其应该不会陌生,经常采用GetDC,G ...

  6. 数据恢复培训资料:BMP文件详解

    BMP是一种与硬件设备无关的图像文件格式,使用非常广.它采用位映射存储格式,除了图像深度可选以外,不采用其他任何压缩,因此,BblP文件所占用的空间很大.BMP文件的图像深度可选lbit.4bit.8 ...

  7. BMP文件解析

    目录 BMP文件简介 BMP文件格式 位图头 位图信息 调色板 位图数据 C语言代码 获取文件大小 获取文件尺寸 获取文件偏移量 读取文件数据示例 一个问题 完整程序 BMP文件简介 BMP(全称Bi ...

  8. oracle windows 新建用户授权 导出导入bmp文件

    1.新建用户 sqlplus /no log conn /as sysdba create user tkjj identified by tkjj2017 grant dba to tkjj con ...

  9. BMP文件组成

    BMP文件组成 BMP文件由文件头.位图信息头.颜色信息和图形数据四部分组成. 如图: 位图文件头BITMAPFILEHEADER 位图信息头BITMAPINFOHEADER 调色板Palette 实 ...

随机推荐

  1. 分布式系列十二: Redis高级主题

    持久化 Redis 支持持久化, 其持久化数据有两种方式. 两种可以同时使用. 如果同时使用, Reids 在重启时将使用 AOF 方式来还原数据. RDB 按照一定策略定时同步内存的数据到磁盘.文件 ...

  2. JustSoso笔记

    当时想了大半天,想着到底要怎么绕过MD5呢,结果还是没做出来,即使问了学长,自己还是漏了一个步骤,file=hint.php,特此笔记,又学到了个引用变量的知识 学习自 https://www.ctf ...

  3. #20175204 张湲祯 2018-2019-2《Java程序设计》第六周学习总结

    20175204 张湲祯 2018-2019-2<Java程序设计>第六周学习总结 教材学习内容总结 -第七章内部类与异常类要点: 一.内部类: Java支持在一个类中定义另一个类,这样的 ...

  4. 3D Slicer中文教程(八)—导出STL文件

    一.STL文件简介 STL(立体平版印刷术的缩写)是由3D Systems创建的立体平版印刷CAD软件原生的文件格式STL有“标准三角语言”和“标准镶嵌语言”等几个事后回溯.这种文件格式是由许多其他软 ...

  5. crontab和at任务

    crontab周期任务 名称解释: cron来源于希腊语 chronos(χρόνος),原意是时间.(引用自维基百科) tab全称是table,表 常用参数: -e 编辑crontab文件 -l 显 ...

  6. spring boot 集成axis1.4 java.lang.NoClassDefFoundError: Could not initialize class org.apache.axis.client.AxisClient

    pom配置: <dependencies> <dependency> <groupId>org.springframework.boot</groupId&g ...

  7. C#如何实现DataGridView单元格拖拽

    参考: http://www.cnblogs.com/michaelxu/archive/2009/09/27/1574905.html

  8. Netty开发redis客户端,Netty发送redis命令,netty解析redis消息

    关键字:Netty开发redis客户端,Netty发送redis命令,netty解析redis消息, netty redis ,redis RESP协议.redis客户端,netty redis协议 ...

  9. Yii2 console执行定时脚本

    为什么要做crontab脚本 我们的项目使用YII2开发,并不是很大的一个电商平台,pv.IP访问量并不是很高,但客户的数据是日积月累已经产生100万条数据了,之前更新订单等数据使用定时脚本直接访问内 ...

  10. 重磅推出:AutoProject Studio 自动化项目生成器

    AutoProject Studio 自动化项目生成器 核心架构图 AutoProject Studio 自动化项目生成器是一款基于C#.Net Framework 4.0为平台自主研发.专为软件设计 ...