PHP如何在数组特定位置处插入新元素?

回复

zkbhj 回复了问题 • 1 人关注 • 1 个回复 • 1766 次浏览 • 2018-12-19 17:43 • 来自相关话题

PHP如何运行时设置错误报告输出到页面上?

回复

zkbhj 回复了问题 • 1 人关注 • 1 个回复 • 1795 次浏览 • 2018-11-21 14:35 • 来自相关话题

PHP如何求两个数组的交叉并补集?

回复

zkbhj 回复了问题 • 1 人关注 • 1 个回复 • 1920 次浏览 • 2018-11-09 11:52 • 来自相关话题

PHP升级7.2之后需要注意的“坑”

zkbhj 发表了文章 • 0 个评论 • 1761 次浏览 • 2018-09-12 20:25 • 来自相关话题

最近升级了PHP版本,从7.1升级到7.2,升级前版本:
PHP 7.1.14 (cli) (built: Feb 2 2018 08:42:59) ( NTS )
Copyright (c) 1997-2018 The PHP Group
Zend Engine v3.1.0, Copyright (c) 1998-2018 Zend Technologies
with Zend OPcache v7.1.14, Copyright (c) 1999-2018, by Zend Technologies
with Xdebug v2.6.0, Copyright (c) 2002-2018, by Derick Rethans升级后版本:
PHP 7.2.2 (cli) (built: Feb 24 2018 17:51:12) ( ZTS DEBUG )
Copyright (c) 1997-2018 The PHP Group
Zend Engine v3.2.0, Copyright (c) 1998-2018 Zend Technologies
with Zend OPcache v7.2.2, Copyright (c) 1999-2018, by Zend Technologies升级完成之后发现有几个框架在使用时都出现了问题,主要原因集中在7.2之后废弃了一些功能,下面列出几个常见的问题:

1、each函数已被废弃:

之前版本写法:<?php
$array = array();
each($array);

// Deprecated: The each() function is deprecated. This message will be suppressed on further calls在7.2版本中会提示过时,可以使用foreach替代each方法,也可以自己修改each方法替代:
<?php
function func_new_each(&$array){
$res = array();
$key = key($array);
if($key !== null){
next($array);
$res[1] = $res['value'] = $array[$key];
$res[0] = $res['key'] = $key;
}else{
$res = false;
}
return $res;
}
2、当传递一个无效参数时,count()函数将抛出warning警告:

之前版本写法<?php
count('');

// Warning: count(): Parameter must be an array or an object that implements Countable在7.2版本中将严格执行类型区分,参数类型不正确,将会出现警告,所以需要在使用count方法时注意参数的值,不过也可以通过自己修改方法来替代(不建议):
<?php
function func_new_count($array_or_countable,$mode = COUNT_NORMAL){
if(is_array($array_or_countable) || is_object($array_or_countable)){
return count($array_or_countable, $mode);
}else{
return 0;
}
}3、create_function被废弃,可以用匿名函数来代替:

之前版本写法:<?php
$newfunc = create_function('$a,$b', 'return "ln($a) + ln($b) = " . log($a * $b);');
echo "New anonymous function: $newfunc\n";
echo $newfunc(2, M_E) . "\n";
// outputs
// New anonymous function: lambda_1
// ln(2) + ln(2.718281828459) = 1.6931471805599

// Warning This function has been DEPRECATED as of PHP 7.2.0. Relying on this function is highly discouraged.在7.2版本中会有警告提示,可修改为匿名函数来替代:
<?php
$newfunc = function ($a,$b){
return "ln($a) + ln($b) = " . log($a * $b);
};
echo $newfunc(2, M_E) . "\n";以上就是升级之后暂时遇到的几个问题,其它相关修改可详看链家产品技术团队做的翻译及整理:PHP7.2 版本指南
https://mp.weixin.qq.com/s/60pohj2n7Pxba3G9vY92yg 查看全部
最近升级了PHP版本,从7.1升级到7.2,升级前版本:
PHP 7.1.14 (cli) (built: Feb  2 2018 08:42:59) ( NTS )
Copyright (c) 1997-2018 The PHP Group
Zend Engine v3.1.0, Copyright (c) 1998-2018 Zend Technologies
with Zend OPcache v7.1.14, Copyright (c) 1999-2018, by Zend Technologies
with Xdebug v2.6.0, Copyright (c) 2002-2018, by Derick Rethans
升级后版本:
PHP 7.2.2 (cli) (built: Feb 24 2018 17:51:12) ( ZTS DEBUG )
Copyright (c) 1997-2018 The PHP Group
Zend Engine v3.2.0, Copyright (c) 1998-2018 Zend Technologies
with Zend OPcache v7.2.2, Copyright (c) 1999-2018, by Zend Technologies
升级完成之后发现有几个框架在使用时都出现了问题,主要原因集中在7.2之后废弃了一些功能,下面列出几个常见的问题:

1、each函数已被废弃:

之前版本写法:
<?php
$array = array();
each($array);

// Deprecated: The each() function is deprecated. This message will be suppressed on further calls
在7.2版本中会提示过时,可以使用foreach替代each方法,也可以自己修改each方法替代:
<?php
function func_new_each(&$array){
$res = array();
$key = key($array);
if($key !== null){
next($array);
$res[1] = $res['value'] = $array[$key];
$res[0] = $res['key'] = $key;
}else{
$res = false;
}
return $res;
}

2、当传递一个无效参数时,count()函数将抛出warning警告:

之前版本写法
<?php
count('');

// Warning: count(): Parameter must be an array or an object that implements Countable
在7.2版本中将严格执行类型区分,参数类型不正确,将会出现警告,所以需要在使用count方法时注意参数的值,不过也可以通过自己修改方法来替代(不建议):
<?php
function func_new_count($array_or_countable,$mode = COUNT_NORMAL){
if(is_array($array_or_countable) || is_object($array_or_countable)){
return count($array_or_countable, $mode);
}else{
return 0;
}
}
3、create_function被废弃,可以用匿名函数来代替:

之前版本写法:
<?php
$newfunc = create_function('$a,$b', 'return "ln($a) + ln($b) = " . log($a * $b);');
echo "New anonymous function: $newfunc\n";
echo $newfunc(2, M_E) . "\n";
// outputs
// New anonymous function: lambda_1
// ln(2) + ln(2.718281828459) = 1.6931471805599

// Warning This function has been DEPRECATED as of PHP 7.2.0. Relying on this function is highly discouraged.
在7.2版本中会有警告提示,可修改为匿名函数来替代:
<?php
$newfunc = function ($a,$b){
return "ln($a) + ln($b) = " . log($a * $b);
};
echo $newfunc(2, M_E) . "\n";
以上就是升级之后暂时遇到的几个问题,其它相关修改可详看链家产品技术团队做的翻译及整理:PHP7.2 版本指南
https://mp.weixin.qq.com/s/60pohj2n7Pxba3G9vY92yg

PHP中如何识别图片是否翻转并矫正?

回复

zkbhj 回复了问题 • 1 人关注 • 1 个回复 • 1654 次浏览 • 2018-09-11 17:57 • 来自相关话题

PHP函数的参数里,前面三个点的形式...$args是什么含义和用法?

zkbhj 发表了文章 • 0 个评论 • 3228 次浏览 • 2018-09-07 13:30 • 来自相关话题

这是PHP5.6添加的功能(一种语法糖): 
可以通过...将函数参数存储在紧接的可遍历的变量中。
 
function add($a, $b, $c)
{
return $a + $b + $c;
}

$num=[2, 3];
echo add(1, ...$num); //6 
上面的第二个和第三个参数将会从$num中循环赋值($b为2,$c为3)。
 
手册地址:http://php.net/manual/zh/migra ... s.php 查看全部
这是PHP5.6添加的功能(一种语法糖): 
可以通过...将函数参数存储在紧接的可遍历的变量中。
 
function add($a, $b, $c)
{
return $a + $b + $c;
}

$num=[2, 3];
echo add(1, ...$num); //6
 
上面的第二个和第三个参数将会从$num中循环赋值($b为2,$c为3)。
 
手册地址:http://php.net/manual/zh/migra ... s.php

PHP获取远程文件的mime类型的方法

zkbhj 发表了文章 • 0 个评论 • 1918 次浏览 • 2018-09-06 19:45 • 来自相关话题

 /**
* 获取远程或本地文件信息
* @param string $strUrl 远程文件或本地文件地址
* @param integer $intType 调用方式(1:get_headers 2:fsocketopen 3:curl 4:本地文件)
* @param array $arrOptional
* @return array
* @author mengdj<mengdj#outlook.com>
*/
function remote_filesize($strUrl,$intType=1,$arrOptional=array()){
$arrRet=array(
"length"=>0, //大小,字节为单位
"mime"=>"", //mime类型
"filename"=>"", //文件名
"status"=>0 //状态码
);
switch($intType){
case 1:
//利用get_headers函数
if(($arrTmp=get_headers($strUrl,true))){
$arrRet=array("length"=>$arrTmp['Content-Length'],"mime"=>$arrTmp['Content-Type']);
if(preg_match('/filename=\"(.*)\"/si',$arrTmp['Content-Disposition'],$arr)){
$arrRet["filename"]=$arr[1];
}
if(preg_match('/\s(\d+)\s/',$arrTmp[0],$arr)){
$arrRet["status"]=$arr[1];
}
}
break;
case 2:
//利用fsocket
if(($arrUrl=parse_url($strUrl))){
if($fp=@fsockopen($arrUrl['host'],empty($arrUrl['port'])?80:$arrUrl['port'],$error)){
@fputs($fp,"GET ".(empty($arrUrl['path'])?'/':$arrUrl['path'])." HTTP/1.1\r\n");
@fputs($fp,"Host: $arrUrl[host]\r\n");
@fputs($fp,"Connection: Close\r\n\r\n");
while(!feof($fp)){
$tmp=fgets($fp);
if(trim($tmp)==''){
//此行代码只读到头信息即可
break;
}else{
(preg_match('/(HTTP.*)(\s\d{3}\s)/',$tmp,$arr))&&$arrRet['status']=trim($arr[2]);
(preg_match('/Content-Length:(.*)/si',$tmp,$arr))&&$arrRet['length']=trim($arr[1]);
(preg_match('/Content-Type:(.*)/si',$tmp,$arr))&&$arrRet['mime']=trim($arr[1]);
(preg_match('/filename=\"(.*)\"/si',$tmp,$arr))&&$arrRet['filename']=trim($arr[1]);
}
}
@fclose($fp);
}
}
break;
case 3:
//利用curl
if(($ch=curl_init($strUrl))){
curl_setopt($ch,CURLOPT_HEADER,1);
curl_setopt($ch,CURLOPT_NOBODY,1);
curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);
if(isset($arrOptional['user'])&&isset($arrOptional['password'])){
$headers=array('Authorization: Basic '.base64_encode($arrOptional['user'].':'.$arrOptional['password']));
curl_setopt($ch,CURLOPT_HTTPHEADER,$headers);
}
$tmp=curl_exec($ch);
curl_close($ch);
(preg_match('/Content-Length:\s([0-9].+?)\s/',$tmp,$arr))&&$arrRet['length']=trim($arr[1]);
(preg_match('/Content-Type:\s(.*)\s/',$tmp,$arr))&&$arrRet['mime']=trim($arr[1]);
(preg_match('/filename=\"(.*)\"/i',$tmp,$arr))&&$arrRet['filename']=trim($arr[1]);
(preg_match('/(HTTP.*)(\s\d{3}\s)/',$tmp,$arr))&&$arrRet['status']=trim($arr[2]);
}
break;
case 4:
//本地处理
if(file_exists($strUrl)) {
$arrRet=array(
"length"=>filesize($strUrl),
"mime" =>mime_content_type($strUrl),
"filename"=>basename($strUrl),
"status"=>200
);
}else{
$arrRet=array(
"length"=>0,
"mime" =>'',
"filename"=>basename($strUrl),
"status"=>404
);
}
break;
}
if(isset($arrOptional['getimagesize'])&&$arrRet['status']=='200'){
if(($arrTmp=@getimagesize($strUrl))){
$arrRet['width']=$arrTmp[0];
$arrRet['height']=$arrTmp[1];
$arrRet['type']=$arrTmp[2];
$arrRet['tag']=$arrTmp[3];
$arrRet['bits']=$arrTmp['bits'];
$arrRet['channels']=$arrTmp['channels'];
!isset($arrRet['mime'])&&$arrRet['mime']=$arrTmp['mime'];
}
}
return $arrRet;
} 查看全部
 
/**
* 获取远程或本地文件信息
* @param string $strUrl 远程文件或本地文件地址
* @param integer $intType 调用方式(1:get_headers 2:fsocketopen 3:curl 4:本地文件)
* @param array $arrOptional
* @return array
* @author mengdj<mengdj#outlook.com>
*/
function remote_filesize($strUrl,$intType=1,$arrOptional=array()){
$arrRet=array(
"length"=>0, //大小,字节为单位
"mime"=>"", //mime类型
"filename"=>"", //文件名
"status"=>0 //状态码
);
switch($intType){
case 1:
//利用get_headers函数
if(($arrTmp=get_headers($strUrl,true))){
$arrRet=array("length"=>$arrTmp['Content-Length'],"mime"=>$arrTmp['Content-Type']);
if(preg_match('/filename=\"(.*)\"/si',$arrTmp['Content-Disposition'],$arr)){
$arrRet["filename"]=$arr[1];
}
if(preg_match('/\s(\d+)\s/',$arrTmp[0],$arr)){
$arrRet["status"]=$arr[1];
}
}
break;
case 2:
//利用fsocket
if(($arrUrl=parse_url($strUrl))){
if($fp=@fsockopen($arrUrl['host'],empty($arrUrl['port'])?80:$arrUrl['port'],$error)){
@fputs($fp,"GET ".(empty($arrUrl['path'])?'/':$arrUrl['path'])." HTTP/1.1\r\n");
@fputs($fp,"Host: $arrUrl[host]\r\n");
@fputs($fp,"Connection: Close\r\n\r\n");
while(!feof($fp)){
$tmp=fgets($fp);
if(trim($tmp)==''){
//此行代码只读到头信息即可
break;
}else{
(preg_match('/(HTTP.*)(\s\d{3}\s)/',$tmp,$arr))&&$arrRet['status']=trim($arr[2]);
(preg_match('/Content-Length:(.*)/si',$tmp,$arr))&&$arrRet['length']=trim($arr[1]);
(preg_match('/Content-Type:(.*)/si',$tmp,$arr))&&$arrRet['mime']=trim($arr[1]);
(preg_match('/filename=\"(.*)\"/si',$tmp,$arr))&&$arrRet['filename']=trim($arr[1]);
}
}
@fclose($fp);
}
}
break;
case 3:
//利用curl
if(($ch=curl_init($strUrl))){
curl_setopt($ch,CURLOPT_HEADER,1);
curl_setopt($ch,CURLOPT_NOBODY,1);
curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);
if(isset($arrOptional['user'])&&isset($arrOptional['password'])){
$headers=array('Authorization: Basic '.base64_encode($arrOptional['user'].':'.$arrOptional['password']));
curl_setopt($ch,CURLOPT_HTTPHEADER,$headers);
}
$tmp=curl_exec($ch);
curl_close($ch);
(preg_match('/Content-Length:\s([0-9].+?)\s/',$tmp,$arr))&&$arrRet['length']=trim($arr[1]);
(preg_match('/Content-Type:\s(.*)\s/',$tmp,$arr))&&$arrRet['mime']=trim($arr[1]);
(preg_match('/filename=\"(.*)\"/i',$tmp,$arr))&&$arrRet['filename']=trim($arr[1]);
(preg_match('/(HTTP.*)(\s\d{3}\s)/',$tmp,$arr))&&$arrRet['status']=trim($arr[2]);
}
break;
case 4:
//本地处理
if(file_exists($strUrl)) {
$arrRet=array(
"length"=>filesize($strUrl),
"mime" =>mime_content_type($strUrl),
"filename"=>basename($strUrl),
"status"=>200
);
}else{
$arrRet=array(
"length"=>0,
"mime" =>'',
"filename"=>basename($strUrl),
"status"=>404
);
}
break;
}
if(isset($arrOptional['getimagesize'])&&$arrRet['status']=='200'){
if(($arrTmp=@getimagesize($strUrl))){
$arrRet['width']=$arrTmp[0];
$arrRet['height']=$arrTmp[1];
$arrRet['type']=$arrTmp[2];
$arrRet['tag']=$arrTmp[3];
$arrRet['bits']=$arrTmp['bits'];
$arrRet['channels']=$arrTmp['channels'];
!isset($arrRet['mime'])&&$arrRet['mime']=$arrTmp['mime'];
}
}
return $arrRet;
}

PHP如何实现对图片指定位置(可多个)进行打码处理

zkbhj 发表了文章 • 0 个评论 • 1793 次浏览 • 2018-09-05 16:43 • 来自相关话题

<?php
/**
* 图片通用处理逻辑类
*/
namespace common\models\logics;

use Yii;
use yii\helpers\Json;


class ImageLogic
{
/**
* PHP将网页上的图片攫取到本地存储
* @param $imgUrl 图片url地址
* @param string $saveDir 本地存储路径 默认存储在当前路径
* @param null $fileName 图片存储到本地的文件名
* @return mix
*/
public static function crabImage($imgUrl, $fileName=null, $saveDir='/tmp/remote/'){
if(empty($imgUrl)){
return false;
}

//获取图片信息大小
$imgSize = getImageSize($imgUrl);
if(!in_array($imgSize['mime'],array('image/jpg', 'image/gif', 'image/png', 'image/jpeg'),true)){
return false;
}

//获取后缀名
$_mime = explode('/', $imgSize['mime']);
$_ext = '.'.end($_mime);

if(empty($fileName)){ //生成唯一的文件名
$fileName = uniqid(time(),true).$_ext;
}else{
$fileName .= $_ext;
}

//判断你是否已经本地有该图片,如果有则直接返回
if(file_exists($saveDir.$fileName)){
return $saveDir.$fileName;
}

//开始攫取
ob_start();
readfile($imgUrl);
$imgInfo = ob_get_contents();
ob_end_clean();

if(!file_exists($saveDir)){
mkdir($saveDir,0777,true);
}
$fp = fopen($saveDir.$fileName, 'a');
$imgLen = strlen($imgInfo); //计算图片源码大小
$_inx = 204800; //每次写入200k
$_time = ceil($imgLen/$_inx);
for($i=0; $i<$_time; $i++){
fwrite($fp,substr($imgInfo, $i*$_inx, $_inx));
}
fclose($fp);

return $saveDir.$fileName;
}


/**
* 将百度AI返回的位置信息转化为大码方法可以是别的位置坐标信息
* 百度:位置数组(坐标0点为左上角)
* left:表示定位位置的长方形左上顶点的水平坐标
* top:表示定位位置的长方形左上顶点的垂直坐标
* width:表示定位位置的长方形的宽度
* height:表示定位位置的长方形的高度
*
* 大码要求的坐标信息(坐标0点为左上角)
* x1:起点横坐标
* y1:起点纵坐标
* x2:终点横坐标
* y2:终点纵坐标
*/
public static function getLocationReal($location = )
{
if(empty($location))
$position = ;

//内容周围padding
$padding = 5;

//循环计算每个位置的坐标数据
foreach($location as $key => $value){
$position[$key]['x1'] = $value['left'] - $padding;
$position[$key]['y1'] = $value['top'] - $padding;
$position[$key]['x2'] = $value['left'] + $value['width'] + $padding;
$position[$key]['y2'] = $value['top'] + $value['height'] + $padding;
}

return $position;

}


/** 图片局部打马赛克
* @param String $source 原图
* @param Stirng $target 生成的图片
* @param int $x1 起点横坐标
* @param int $y1 起点纵坐标
* @param int $x2 终点横坐标
* @param int $y2 终点纵坐标
* @param int $deep 深度,数字越大越模糊
* @return boolean
*/
public static function imageMosaics($source, $target, $x1, $y1, $x2, $y2, $deep = 6){

// 判断原图是否存在
if(!file_exists($source)){
return false;
}

// 获取原图信息
list($o_width, $o_height, $o_type) = getimagesize($source);

// 判断区域是否超出图片
if($x1>$o_width || $x1<0 || $x2>$o_width || $x2<0 || $y1>$o_height || $y1<0 || $y2>$o_height || $y2<0){
return false;
}

switch($o_type){
case 1: $source_img = imagecreatefromgif($source); break;
case 2: $source_img = imagecreatefromjpeg($source); break;
case 3: $source_img = imagecreatefrompng($source); break;
default:
return false;
}

// 打马赛克
for($x=$x1; $x<$x2; $x=$x+$deep){
for($y=$y1; $y<$y2; $y=$y+$deep){
$color = imagecolorat($source_img, $x+round($deep/2), $y+round($deep/2));
imagefilledrectangle($source_img, $x, $y, $x+$deep, $y+$deep, $color);
}
}

// 生成图片
switch($o_type){
case 1: imagegif($source_img, $target); break;
case 2: imagejpeg($source_img, $target); break;
case 3: imagepng($source_img, $target); break;
}

return is_file($target)? true : false;

}

/**
* 在图片固定位置(支持多个位置)打码
* @param String $source 原图
* @param Stirng $target 生成的图片
* @param array $position 大码位置信息,可能有多个需要打码的位置
* @param int $position->x1 起点横坐标
* @param int $position->y1 起点纵坐标
* @param int $position->x2 终点横坐标
* @param int $position->y2 终点纵坐标
* @param int $deep 深度,数字越大越模糊
* return string 返回处理后的本地图片地址,如 /tmp/convert/1a987608924ce098fc8ccd3deaac8d09.png
*/
public static function imageAddMosaics($source, $target, $position, $deep = 6)
{
//默认生成临时文件地址
$target = empty($target) ? '/tmp/
convert/'.md5(Json::encode($position)).'.png' : $target;

//将要打码的位置信息从百度模式转为本地模式
$position = self::getLocationReal($position);

//判断图片是否为远程图片
if(strpos($source,'http') !==false){
$source = self::crabImage($source,md5(Json::encode($position)));
}

//循环处理图片打码
foreach ($position as $key => $value) {

//如果是第一次进行打码操作
if($key == 0){
self::imageMosaics($source, $target, $value['x1'], $value['y1'], $value['x2'], $value['y2'], 6);
}else{
self::imageMosaics($target, $target, $value['x1'], $value['y1'], $value['x2'], $value['y2'], 6);
}
}

return $target;

}

}


//使用
use common\models\logics\ImageLogic;
$source = 'https://qiniu.zkbhj.com/common/images/jiagou/%E5%87%AF%E5%86%B0%E7%A7%91%E6%8A%80%E9%9D%99%E6%80%81%E8%B5%84%E6%BA%90%E6%9C%8D%E5%8A%A1%E6%9E%B6%E6%9E%84.png';
$target = '';
$position = [
0 => [
'width'=>656,
'top'=>452,
'left'=>110,
'height'=>35

],
1 => [
'width'=>748,
'top'=>1400,
'left'=>188,
'height'=>39

]
];

$img = ImageLogic::imageAddMosaics($source, $target, $position);




需要用到百度AI图片文本内容识别接口,参考:http://ai.baidu.com/docs#/OCR-API/0d9adafa

请求百度图片识别接口代码示例:
<?php

/**
* 发起http post请求(REST API), 并获取REST请求的结果
* @param string $url
* @param string $param
* @return - http response body if succeeds, else false.
*/
function request_post($url = '', $param = '')
{
if (empty($url) || empty($param)) {
return false;
}

$postUrl = $url;
$curlPost = $param;
// 初始化curl
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, $postUrl);
curl_setopt($curl, CURLOPT_HEADER, 0);
// 要求结果为字符串且输出到屏幕上
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
// post提交方式
curl_setopt($curl, CURLOPT_POST, 1);
curl_setopt($curl, CURLOPT_POSTFIELDS, $curlPost);
// 运行curl
$data = curl_exec($curl);
curl_close($curl);

return $data;
}
//token需要通过自己的appid和secretkey获取
$token = '24.6a792a76b5edaf7b8d48f65a8843c3c5.2592000.1538639572.282335-11769895';
$url = 'https://aip.baidubce.com/rest/2.0/ocr/v1/general?access_token=' . $token;
$img = file_get_contents('source.png');
$img = base64_encode($img);
$bodys = array(
"image" => $img
);
$res = request_post($url, $bodys);
echo $res;




  查看全部
<?php
/**
* 图片通用处理逻辑类
*/
namespace common\models\logics;

use Yii;
use yii\helpers\Json;


class ImageLogic
{
/**
* PHP将网页上的图片攫取到本地存储
* @param $imgUrl 图片url地址
* @param string $saveDir 本地存储路径 默认存储在当前路径
* @param null $fileName 图片存储到本地的文件名
* @return mix
*/
public static function crabImage($imgUrl, $fileName=null, $saveDir='/tmp/remote/'){
if(empty($imgUrl)){
return false;
}

//获取图片信息大小
$imgSize = getImageSize($imgUrl);
if(!in_array($imgSize['mime'],array('image/jpg', 'image/gif', 'image/png', 'image/jpeg'),true)){
return false;
}

//获取后缀名
$_mime = explode('/', $imgSize['mime']);
$_ext = '.'.end($_mime);

if(empty($fileName)){ //生成唯一的文件名
$fileName = uniqid(time(),true).$_ext;
}else{
$fileName .= $_ext;
}

//判断你是否已经本地有该图片,如果有则直接返回
if(file_exists($saveDir.$fileName)){
return $saveDir.$fileName;
}

//开始攫取
ob_start();
readfile($imgUrl);
$imgInfo = ob_get_contents();
ob_end_clean();

if(!file_exists($saveDir)){
mkdir($saveDir,0777,true);
}
$fp = fopen($saveDir.$fileName, 'a');
$imgLen = strlen($imgInfo); //计算图片源码大小
$_inx = 204800; //每次写入200k
$_time = ceil($imgLen/$_inx);
for($i=0; $i<$_time; $i++){
fwrite($fp,substr($imgInfo, $i*$_inx, $_inx));
}
fclose($fp);

return $saveDir.$fileName;
}


/**
* 将百度AI返回的位置信息转化为大码方法可以是别的位置坐标信息
* 百度:位置数组(坐标0点为左上角)
* left:表示定位位置的长方形左上顶点的水平坐标
* top:表示定位位置的长方形左上顶点的垂直坐标
* width:表示定位位置的长方形的宽度
* height:表示定位位置的长方形的高度
*
* 大码要求的坐标信息(坐标0点为左上角)
* x1:起点横坐标
* y1:起点纵坐标
* x2:终点横坐标
* y2:终点纵坐标
*/
public static function getLocationReal($location = )
{
if(empty($location))
$position = ;

//内容周围padding
$padding = 5;

//循环计算每个位置的坐标数据
foreach($location as $key => $value){
$position[$key]['x1'] = $value['left'] - $padding;
$position[$key]['y1'] = $value['top'] - $padding;
$position[$key]['x2'] = $value['left'] + $value['width'] + $padding;
$position[$key]['y2'] = $value['top'] + $value['height'] + $padding;
}

return $position;

}


/** 图片局部打马赛克
* @param String $source 原图
* @param Stirng $target 生成的图片
* @param int $x1 起点横坐标
* @param int $y1 起点纵坐标
* @param int $x2 终点横坐标
* @param int $y2 终点纵坐标
* @param int $deep 深度,数字越大越模糊
* @return boolean
*/
public static function imageMosaics($source, $target, $x1, $y1, $x2, $y2, $deep = 6){

// 判断原图是否存在
if(!file_exists($source)){
return false;
}

// 获取原图信息
list($o_width, $o_height, $o_type) = getimagesize($source);

// 判断区域是否超出图片
if($x1>$o_width || $x1<0 || $x2>$o_width || $x2<0 || $y1>$o_height || $y1<0 || $y2>$o_height || $y2<0){
return false;
}

switch($o_type){
case 1: $source_img = imagecreatefromgif($source); break;
case 2: $source_img = imagecreatefromjpeg($source); break;
case 3: $source_img = imagecreatefrompng($source); break;
default:
return false;
}

// 打马赛克
for($x=$x1; $x<$x2; $x=$x+$deep){
for($y=$y1; $y<$y2; $y=$y+$deep){
$color = imagecolorat($source_img, $x+round($deep/2), $y+round($deep/2));
imagefilledrectangle($source_img, $x, $y, $x+$deep, $y+$deep, $color);
}
}

// 生成图片
switch($o_type){
case 1: imagegif($source_img, $target); break;
case 2: imagejpeg($source_img, $target); break;
case 3: imagepng($source_img, $target); break;
}

return is_file($target)? true : false;

}

/**
* 在图片固定位置(支持多个位置)打码
* @param String $source 原图
* @param Stirng $target 生成的图片
* @param array $position 大码位置信息,可能有多个需要打码的位置
* @param int $position->x1 起点横坐标
* @param int $position->y1 起点纵坐标
* @param int $position->x2 终点横坐标
* @param int $position->y2 终点纵坐标
* @param int $deep 深度,数字越大越模糊
* return string 返回处理后的本地图片地址,如 /tmp/convert/1a987608924ce098fc8ccd3deaac8d09.png
*/
public static function imageAddMosaics($source, $target, $position, $deep = 6)
{
//默认生成临时文件地址
$target = empty($target) ? '/tmp/
convert/'.md5(Json::encode($position)).'.png' : $target;

//将要打码的位置信息从百度模式转为本地模式
$position = self::getLocationReal($position);

//判断图片是否为远程图片
if(strpos($source,'http') !==false){
$source = self::crabImage($source,md5(Json::encode($position)));
}

//循环处理图片打码
foreach ($position as $key => $value) {

//如果是第一次进行打码操作
if($key == 0){
self::imageMosaics($source, $target, $value['x1'], $value['y1'], $value['x2'], $value['y2'], 6);
}else{
self::imageMosaics($target, $target, $value['x1'], $value['y1'], $value['x2'], $value['y2'], 6);
}
}

return $target;

}

}


//使用
use common\models\logics\ImageLogic;
$source = 'https://qiniu.zkbhj.com/common/images/jiagou/%E5%87%AF%E5%86%B0%E7%A7%91%E6%8A%80%E9%9D%99%E6%80%81%E8%B5%84%E6%BA%90%E6%9C%8D%E5%8A%A1%E6%9E%B6%E6%9E%84.png';
$target = '';
$position = [
0 => [
'width'=>656,
'top'=>452,
'left'=>110,
'height'=>35

],
1 => [
'width'=>748,
'top'=>1400,
'left'=>188,
'height'=>39

]
];

$img = ImageLogic::imageAddMosaics($source, $target, $position);




需要用到百度AI图片文本内容识别接口,参考:http://ai.baidu.com/docs#/OCR-API/0d9adafa

请求百度图片识别接口代码示例:
<?php

/**
* 发起http post请求(REST API), 并获取REST请求的结果
* @param string $url
* @param string $param
* @return - http response body if succeeds, else false.
*/
function request_post($url = '', $param = '')
{
if (empty($url) || empty($param)) {
return false;
}

$postUrl = $url;
$curlPost = $param;
// 初始化curl
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, $postUrl);
curl_setopt($curl, CURLOPT_HEADER, 0);
// 要求结果为字符串且输出到屏幕上
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
// post提交方式
curl_setopt($curl, CURLOPT_POST, 1);
curl_setopt($curl, CURLOPT_POSTFIELDS, $curlPost);
// 运行curl
$data = curl_exec($curl);
curl_close($curl);

return $data;
}
//token需要通过自己的appid和secretkey获取
$token = '24.6a792a76b5edaf7b8d48f65a8843c3c5.2592000.1538639572.282335-11769895';
$url = 'https://aip.baidubce.com/rest/2.0/ocr/v1/general?access_token=' . $token;
$img = file_get_contents('source.png');
$img = base64_encode($img);
$bodys = array(
"image" => $img
);
$res = request_post($url, $bodys);
echo $res;

QQ截图20180905165443.jpg

 

PHP环境有问题调试记录

回复

zkbhj 发起了问题 • 1 人关注 • 0 个回复 • 1848 次浏览 • 2018-09-05 12:39 • 来自相关话题

PHP如何接受数据流保存为图片数据并且还带有多个其他参数?

回复

zkbhj 回复了问题 • 1 人关注 • 1 个回复 • 1295 次浏览 • 2018-09-04 22:38 • 来自相关话题