PHP 日本大会 2024

imageconvolution

(PHP 5 >= 5.1.0, PHP 7, PHP 8)

imageconvolution应用 3x3 卷积矩阵,使用系数和偏移量

描述

imageconvolution(
    GdImage $image,
    array $matrix,
    float $divisor,
    float $offset
): bool

使用给定的系数和偏移量在图像上应用卷积矩阵。

参数

image

一个 GdImage 对象,由其中一个图像创建函数返回,例如 imagecreatetruecolor()

matrix

一个 3x3 矩阵:一个包含三个包含三个浮点数的数组的数组。

divisor

卷积结果的除数,用于归一化。

offset

颜色偏移。

返回值

成功时返回 true,失败时返回 false

变更日志

版本 描述
8.0.0 image 现在需要一个 GdImage 实例;以前,需要一个有效的 gd resource

范例

示例 #1 浮雕 PHP.net 徽标

<?php
$image
= imagecreatefromgif('https://php.net/images/php.gif');

$emboss = array(array(2, 0, 0), array(0, -1, 0), array(0, 0, -1));
imageconvolution($image, $emboss, 1, 127);

header('Content-Type: image/png');
imagepng($image, null, 9);
?>

以上示例将输出

Output of example : Embossing the PHP.net logo

示例 #2 高斯模糊

<?php
$image
= imagecreatetruecolor(180,40);

// 写入文本并在图像上应用高斯模糊
imagestring($image, 5, 10, 8, 'Gaussian Blur Text', 0x00ff00);
$gaussian = array(array(1.0, 2.0, 1.0), array(2.0, 4.0, 2.0), array(1.0, 2.0, 1.0));
imageconvolution($image, $gaussian, 16, 0);

// 重新写入文本以进行比较
imagestring($image, 5, 10, 18, 'Gaussian Blur Text', 0x00ff00);

header('Content-Type: image/png');
imagepng($image, null, 9);
?>

以上示例将输出

Output of example : Gaussian blur

参见

添加注释

用户贡献的注释 10 条注释

magilvia
13 年前
这是一个用于缩略图创建的很好的锐化设置

<?php
$sharpenMatrix
= array
(
array(-
1.2, -1, -1.2),
array(-
1, 20, -1),
array(-
1.2, -1, -1.2)
);

// 计算锐化除数
$divisor = array_sum(array_map('array_sum', $sharpenMatrix));

$offset = 0;

// 应用矩阵
imageconvolution($img, $sharpenMatrix, $divisor, $offset);
?>
[email protected]
14年前
需要进行归一化处理以保持图像平衡(否则任何滤镜都可能很快使图像几乎完全变成黑色或白色)。

这是一个简短易用的类,用于自动处理归一化并简化3x3矩阵的输入。

此代码遵循`imageconvolution()`函数使用的“三个数组的数组”语法,并自动计算必要的归一化除数。

<?php

class ConvolutionFilter {
public
$matrix;
public
$div;

public function
computeDiv() {
$this->div = array_sum ($this->matrix[0]) + array_sum ($this->matrix[1]) + array_sum ($this->matrix[2]);
}

function
__construct() {
$matrix = func_get_args();
$this->matrix = array( array($matrix[0], $matrix[1], $matrix[2]),
array(
$matrix[3], $matrix[4], $matrix[5]),
array(
$matrix[6], $matrix[7], $matrix[8])
);
$this->computeDiv();
}
}

?>

示例用法

<?php

$gaussianFilter
= new ConvolutionFilter( 1.0, 2.0, 1.0,
2.0, 3.0, 2.0,
1.0, 2.0, 1.0 );
imageconvolution($image, $gaussianFilter->matrix, $gaussianFilter->div, 0);

?>

一些常用的滤镜

<?php

$identityFilter
= new ConvolutionFilter( 0.0, 0.0, 0.0,
0.0, 1.0, 0.0,
0.0, 0.0, 0.0 );
$sharpenFilter = new ConvolutionFilter( 0.0, -1.0, 0.0,
-
1.0, 5.0, -1.0,
0.0, -1.0, 0.0 );
$edgeFilter = new ConvolutionFilter( 0.0, 1.0, 0.0,
1.0, -4.0, 1.0,
0.0, 1.0, 0.0 );
$findEdgesFilter = new ConvolutionFilter( -1.0, -1.0, -1.0,
-
2.0, 8.0, -1.0,
-
1.0, -1.0, -1.0 );

?>

请记住,您可以使用`imagefilter()`来满足此类基本需求,但是当您想要创建自己的滤镜时,上面的类会让您更容易操作。
匿名
19年前
下面的例子没有提供一个3x3矩阵。正确地说,它是一个多维数组。

<?php
$matrix
= array( array( -1, -1, -1 ),
array( -
1, 16, -1 ),
array( -
1, -1, -1 ) );
?>
[email protected]
18年前
我见过很多人想出各种方法在矩形(例如图片)后面创建投影。但我还没有找到一个既快速、又兼容PHP 4,而且看起来不错的方案。这是我昨晚想出的一个方案。它接收一个图像,用背景填充它,并在指定的坐标处使用指定的颜色和距离偏移量创建模糊的投影。效果很棒!!!

function blurRect(&$image, $distance, $rectX1, $rectY1, $rectX2, $rectY2, $shadowR, $shadowG, $shadowB, $backR, $backG, $backB) {

$potentialOverlap = ($distance * 2) * ($distance * 2);

$backgroundColor = imagecolorallocate($image, $backR, $backG, $backB);
$shadowColor = imagecolorallocate($image, $shadowR, $shadowG, $shadowB);

$imageWidth = imagesx($image);
$imageHeight = imagesy($image);

imageFilledRectangle($image, 0, 0, $imageWidth - 1, $imageHeight - 1, $backgroundColor);
imageFilledRectangle($image, $rectX1, $rectY1, $rectX2, $rectY2, $shadowColor);

for ( $pointX = $rectX1 - $distance; $pointX < $imageWidth; $pointX++ ) {
for ( $pointY = $rectY1 - $distance; $pointY < $imageHeight; $pointY++ ) {

if ( $pointX > $rectX1 + $distance &&
$pointX < $rectX2 - $distance &&
$pointY > $rectY1 + $distance &&
$pointY < $rectY2 - $distance ) {
$pointY = $rectY2 - $distance;
}

$boxX1 = $pointX - $distance;
$boxY1 = $pointY - $distance;
$boxX2 = $pointX + $distance;
$boxY2 = $pointY + $distance;

$xOverlap = max(0, min($boxX2, $rectX2) - max($boxX1, $rectX1));
$yOverlap = max(0, min($boxY2, $rectY2) - max($boxY1, $rectY1));

$totalOverlap = $xOverlap * $yOverlap;
$shadowPcnt = $totalOverlap / $potentialOverlap;
$backPcnt = 1.0 - $shadowPcnt;

$newR = $shadowR * $shadowPcnt + $backR * $backPcnt;
$newG = $shadowG * $shadowPcnt + $backG * $backPcnt;
$newB = $shadowB * $shadowPcnt + $backB * $backPcnt;

$newcol = imagecolorallocate($image, $newR, $newG, $newB);
imagesetpixel($image, $pointX, $pointY, $newcol);
}
}
}
[email protected]
19年前
矩阵可用于锐化、模糊、边缘检测等,类似于Photoshop。

一个锐化示例

<?php

$sharpenMatrix
= array(-1,-1,-1,-1,16,-1,-1,-1,-1);
$divisor = 8;
$offset = 0;

imageconvolution($myImage, $sharpenMatrix, $divisor, $offset);

?>

以下是构建不同类型矩阵的一些信息。(如果你有Photoshop(或PSP、GIMP),你可以在PHP中应用矩阵之前测试你的矩阵)

http://loriweb.pair.com/8udf-basics.html (涵盖模糊)
http://loriweb.pair.com/8udf-sharpen.html
http://loriweb.pair.com/8udf-edges.html
http://loriweb.pair.com/8udf-emboss.html
mgcclx at gmail dot com
17年前
对于非捆绑的GD库,PHP中似乎没有`imageconvolution()`函数。这种情况比较少见,但仍然会发生。这就是我用PHP编写`imageconvolution()`函数副本的原因。与下面的文章相比,这个版本使用了偏移量,速度提高了30%。
因为它是用PHP编写的,所以它比捆绑版本慢50倍。
实际上,这是GD库中`gdimageconvolutaion()`函数的复制版本,它不支持`imageconvolution()`函数具有的数据验证功能。但我猜使用此函数的人知道他们在做什么。

脚本
<?php
// 只要需要使用imageconvolution...就包含此文件。
// 你可以在你的项目中使用它,但请保留下面的注释 :)
// 非常适合任何图像处理库
// 由Chao Xu(Mgccl)于2007年2月28日编写
// www.webdevlogs.com
// V 1.0
if(!function_exists('imageconvolution')){
function
imageconvolution($src, $filter, $filter_div, $offset){
if (
$src==NULL) {
return
0;
}

$sx = imagesx($src);
$sy = imagesy($src);
$srcback = ImageCreateTrueColor ($sx, $sy);
ImageCopy($srcback, $src,0,0,0,0,$sx,$sy);

if(
$srcback==NULL){
return
0;
}

for (
$y=0; $y<$sy; ++$y){
for(
$x=0; $x<$sx; ++$x){
$new_r = $new_g = $new_b = 0;
$alpha = imagecolorat($srcback, $pxl[0], $pxl[1]);
$new_a = $alpha >> 24;

for (
$j=0; $j<3; ++$j) {
$yv = min(max($y - 1 + $j, 0), $sy - 1);
for (
$i=0; $i<3; ++$i) {
$pxl = array(min(max($x - 1 + $i, 0), $sx - 1), $yv);
$rgb = imagecolorat($srcback, $pxl[0], $pxl[1]);
$new_r += (($rgb >> 16) & 0xFF) * $filter[$j][$i];
$new_g += (($rgb >> 8) & 0xFF) * $filter[$j][$i];
$new_b += ($rgb & 0xFF) * $filter[$j][$i];
}
}

$new_r = ($new_r/$filter_div)+$offset;
$new_g = ($new_g/$filter_div)+$offset;
$new_b = ($new_b/$filter_div)+$offset;

$new_r = ($new_r > 255)? 255 : (($new_r < 0)? 0:$new_r);
$new_g = ($new_g > 255)? 255 : (($new_g < 0)? 0:$new_g);
$new_b = ($new_b > 255)? 255 : (($new_b < 0)? 0:$new_b);

$new_pxl = ImageColorAllocateAlpha($src, (int)$new_r, (int)$new_g, (int)$new_b, $new_a);
if (
$new_pxl == -1) {
$new_pxl = ImageColorClosestAlpha($src, (int)$new_r, (int)$new_g, (int)$new_b, $new_a);
}
if ((
$y >= 0) && ($y < $sy)) {
imagesetpixel($src, $x, $y, $new_pxl);
}
}
}
imagedestroy($srcback);
return
1;
}
}
?>
phunction.sf.net
14年前
你不需要任何自定义函数来计算$matrix的$divisor,使用`array_map()`和`array_sum()`就可以做到。

<?php

$matrix
= array
(
array(-
1, -1, -1),
array(-
1, 16, -1),
array(-
1, -1, -1),
);

$divisor = array_sum(array_map('array_sum', $matrix)); // 8

?>
Jase
17年前
下面的注释是一个非常好的解决方法

但是,当我将错误报告设置为E_ALL时,php向我抛出了许多警告。

这可以通过一行代码避免,并且(据我所见)不会影响函数的其余部分。

<?php
// 只要需要使用 imageconvolution 函数,就包含此文件…
// 你可以在你的项目中使用它,但请保留下面的注释 :)
// 非常适合任何图像处理库
// 作者:徐超 (Mgccl) 2007年2月28日
// www.webdevlogs.com
// V 1.0
if(!function_exists('imageconvolution')){
function
imageconvolution($src, $filter, $filter_div, $offset){
if (
$src==NULL) {
return
0;
}

$sx = imagesx($src);
$sy = imagesy($src);
$srcback = ImageCreateTrueColor ($sx, $sy);
ImageCopy($srcback, $src,0,0,0,0,$sx,$sy);

if(
$srcback==NULL){
return
0;
}

# 此处修复
# $pxl 数组是问题所在,因此只需使用非常小的值对其进行设置
$pxl = array(1,1);
# 这个小修复对我有用,因为未定义的数组会抛出错误

for ($y=0; $y<$sy; ++$y){
for(
$x=0; $x<$sx; ++$x){
$new_r = $new_g = $new_b = 0;
$alpha = imagecolorat($srcback, $pxl[0], $pxl[1]);
$new_a = $alpha >> 24;

for (
$j=0; $j<3; ++$j) {
$yv = min(max($y - 1 + $j, 0), $sy - 1);
for (
$i=0; $i<3; ++$i) {
$pxl = array(min(max($x - 1 + $i, 0), $sx - 1), $yv);
$rgb = imagecolorat($srcback, $pxl[0], $pxl[1]);
$new_r += (($rgb >> 16) & 0xFF) * $filter[$j][$i];
$new_g += (($rgb >> 8) & 0xFF) * $filter[$j][$i];
$new_b += ($rgb & 0xFF) * $filter[$j][$i];
}
}

$new_r = ($new_r/$filter_div)+$offset;
$new_g = ($new_g/$filter_div)+$offset;
$new_b = ($new_b/$filter_div)+$offset;

$new_r = ($new_r > 255)? 255 : (($new_r < 0)? 0:$new_r);
$new_g = ($new_g > 255)? 255 : (($new_g < 0)? 0:$new_g);
$new_b = ($new_b > 255)? 255 : (($new_b < 0)? 0:$new_b);

$new_pxl = ImageColorAllocateAlpha($src, (int)$new_r, (int)$new_g, (int)$new_b, $new_a);
if (
$new_pxl == -1) {
$new_pxl = ImageColorClosestAlpha($src, (int)$new_r, (int)$new_g, (int)$new_b, $new_a);
}
if ((
$y >= 0) && ($y < $sy)) {
imagesetpixel($src, $x, $y, $new_pxl);
}
}
}
imagedestroy($srcback);
return
1;
}
}
?>
[email protected]
18年前
这是针对PHP 5.1 以下版本的函数实现。
<?php
if(!function_exists("imageconvolution"))
{
function
imageconvolution(&$img,$mat,$div,$off)
{
if(!
imageistruecolor($img) || !is_array($mat) || count($mat)!=3 || count($mat[0])!=3 || count($mat[1])!=3 || count($mat[2])!=3) return FALSE;
unset(
$bojainfo);
for(
$nx=0;$nx<imagesx($img)-1;$nx++)
{
for(
$ny=0;$ny<imagesy($img)-1;$ny++)
{
$rgb=imagecolorat($img,$nx,$ny);
$bojainfo[$nx][$ny][r]=($rgb>>16)&0xFF;
$bojainfo[$nx][$ny][g]=($rgb>>8)&0xFF;
$bojainfo[$nx][$ny][b]=$rgb&0xFF;
}
}
for(
$nx=1;$nx<imagesx($img)-1;$nx++)
{
for(
$ny=1;$ny<imagesy($img)-1;$ny++)
{
$nr=$mat[0][0]*$bojainfo[$nx-1][$ny-1][r] + $mat[0][1]*$bojainfo[$nx][$ny-1][r] + $mat[0][2]*$bojainfo[$nx+1][$ny-1][r] + $mat[1][0]*$bojainfo[$nx-1][$ny][r] + $mat[1][1]*$bojainfo[$nx][$ny][r] + $mat[1][2]*$bojainfo[$nx+1][$ny][r] + $mat[2][0]*$bojainfo[$nx-1][$ny+1][r] + $mat[2][1]*$bojainfo[$nx][$ny+1][r] + $mat[2][2]*$bojainfo[$nx+1][$ny+1][r];
$nr=intval(round($nr/$div));
if(
$nr<0) $nr=0;
elseif(
$nr>255) $nr=255;
$ng=$mat[0][0]*$bojainfo[$nx-1][$ny-1][g] + $mat[0][1]*$bojainfo[$nx][$ny-1][g] + $mat[0][2]*$bojainfo[$nx+1][$ny-1][g] + $mat[1][0]*$bojainfo[$nx-1][$ny][g] + $mat[1][1]*$bojainfo[$nx][$ny][g] + $mat[1][2]*$bojainfo[$nx+1][$ny][g] + $mat[2][0]*$bojainfo[$nx-1][$ny+1][g] + $mat[2][1]*$bojainfo[$nx][$ny+1][g] + $mat[2][2]*$bojainfo[$nx+1][$ny+1][g];
$ng=intval(round($ng/$div));
if(
$ng<0) $ng=0;
elseif(
$ng>255) $ng=255;
$nb=$mat[0][0]*$bojainfo[$nx-1][$ny-1][b] + $mat[0][1]*$bojainfo[$nx][$ny-1][b] + $mat[0][2]*$bojainfo[$nx+1][$ny-1][b] + $mat[1][0]*$bojainfo[$nx-1][$ny][b] + $mat[1][1]*$bojainfo[$nx][$ny][b] + $mat[1][2]*$bojainfo[$nx+1][$ny][b] + $mat[2][0]*$bojainfo[$nx-1][$ny+1][b] + $mat[2][1]*$bojainfo[$nx][$ny+1][b] + $mat[2][2]*$bojainfo[$nx+1][$ny+1][b];
$nb=intval(round($nb/$div));
if(
$nb<0) $nb=0;
elseif(
$nb>255) $nb=255;
$nrgb=($nr<<16)+($ng<<8)+$nb;
if(!
imagesetpixel($img,$nx,$ny,$nrgb)) return FALSE;
}
}
return
TRUE;
}
}
?>

它运行速度有点慢,所以我不建议处理大型图像。此外,偏移量 (offset) 没有实现(不知道它应该做什么)。
[email protected]
18年前
花了我一段时间,但多亏了PHP文档页面上array_values函数的一些用户注释,我才想出了动态计算除数的方法。

我在Win32系统上使用PHP 5.1.0b2和捆绑的GD库。当我尝试使用imageconvolution函数(无论是正常使用还是通过下面的函数)时,生成的图像(我只尝试了JPEG和GIF)总是显得太亮,即使除数使矩阵和等于1。唯一能降低亮度的方法是使offset参数大到离谱。所以,我不确定这是否会影响其他人。

以下是一些函数以及示例。

<?php
$im
= imagecreatefromjpeg('path/to/pic.jpg');
$matrix = array( array(5,5,5),
array(
5,15,5),
array(
5,5,5) );
makeFilter($im, $matrix);

header ( 'Content-Type: image/jpeg' );
imagejpeg($im);
imagedestroy($im);

/**
* 函数
*/
// 将3X3数组矩阵展平,以便获得所有值的总和
function array_flatten($array) {
(array)
$tempArray = array();

foreach (
$array as $value ) {
if (
is_array($value) ) {
$tempArray = array_merge($tempArray, array_flatten($value));
} else {
$tempArray[] = $value;
}
}

return
$tempArray;
}

// 动态创建除数,并传递偏移量
function makeFilter($resource, $matrix, $offset=1.0) {
global $
$resource;
(float)
$divisor = array_sum(array_flatten($matrix));
return
imageconvolution($resource, $matrix, $divisor, $offset) ? true : false;
}
?>
To Top