PHP代码总结:时间处理相关的辅助类

zkbhj 发表了文章 • 0 个评论 • 1095 次浏览 • 2019-05-30 14:49 • 来自相关话题

class TimeHelper
{
/**
* 得到当前时间的毫秒时间戳
* @return float 13位毫秒时间戳
*/
public static function getCurTimeMsec()
{
list($t1, $t2) = explode(' ', microtime());
return (float)sprintf('%.0f',(floatval($t1)+floatval($t2))*1000);
}

/**
* 获取当前时间17位的毫秒时间格式时间点(2017 07 29 21 44 43 129)
* @return integer
*/
public static function getCurTimeMsecFormat()
{
list($u_sec, $sec) = explode(' ', microtime());
return intval(date('YmdHis', intval($sec)) . str_pad(round(floatval($u_sec) * 1000), 3, 0, STR_PAD_LEFT));
}

/**
* 将20200629000000格式的日期转换成0000-00-00 00:00:00格式
*/
public static function dateNormalization($date)
{
if (strlen($date) == 14) {
return preg_replace('/(\d{4})(\d{2})(\d{2})(\d{2})(\d{2})(\d{2})/', "$1-$2-$3 $4:$5:$6", $date);
}
return $date;
}

/**
* 将17位的毫秒时间格式(2017 07 29 21 44 43 129)转换成毫秒时间戳格式(15XXX)
*/
public static function dateToMescNormalization($date)
{
if (strlen($date) == 17) {
$secDate = preg_replace('/(\d{4})(\d{2})(\d{2})(\d{2})(\d{2})(\d{2})/', "$1-$2-$3 $4:$5:$6", substr($date,0,14));
return strtotime($secDate).substr($date,14,3);
}
return $date;
}
} 查看全部
class TimeHelper
{
/**
* 得到当前时间的毫秒时间戳
* @return float 13位毫秒时间戳
*/
public static function getCurTimeMsec()
{
list($t1, $t2) = explode(' ', microtime());
return (float)sprintf('%.0f',(floatval($t1)+floatval($t2))*1000);
}

/**
* 获取当前时间17位的毫秒时间格式时间点(2017 07 29 21 44 43 129)
* @return integer
*/
public static function getCurTimeMsecFormat()
{
list($u_sec, $sec) = explode(' ', microtime());
return intval(date('YmdHis', intval($sec)) . str_pad(round(floatval($u_sec) * 1000), 3, 0, STR_PAD_LEFT));
}

/**
* 将20200629000000格式的日期转换成0000-00-00 00:00:00格式
*/
public static function dateNormalization($date)
{
if (strlen($date) == 14) {
return preg_replace('/(\d{4})(\d{2})(\d{2})(\d{2})(\d{2})(\d{2})/', "$1-$2-$3 $4:$5:$6", $date);
}
return $date;
}

/**
* 将17位的毫秒时间格式(2017 07 29 21 44 43 129)转换成毫秒时间戳格式(15XXX)
*/
public static function dateToMescNormalization($date)
{
if (strlen($date) == 17) {
$secDate = preg_replace('/(\d{4})(\d{2})(\d{2})(\d{2})(\d{2})(\d{2})/', "$1-$2-$3 $4:$5:$6", substr($date,0,14));
return strtotime($secDate).substr($date,14,3);
}
return $date;
}
}

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函数的参数里,前面三个点的形式...$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 发表了文章 • 0 个评论 • 1080 次浏览 • 2018-09-04 21:11 • 来自相关话题

原理:

对图片中选定区域的每一像素,增加若干宽度及高度,生成矩型。而每一像素的矩型重叠在一起,就形成了马赛克效果。

本例使用GD库的imagecolorat获取像素颜色,使用imagefilledrectangle画矩型。<?php
/** 图片局部打马赛克
* @param String $source 原图
* @param Stirng $dest 生成的图片
* @param int $x1 起点横坐标
* @param int $y1 起点纵坐标
* @param int $x2 终点横坐标
* @param int $y2 终点纵坐标
* @param int $deep 深度,数字越大越模糊
* @return boolean
*/
function imageMosaics($source, $dest, $x1, $y1, $x2, $y2, $deep){

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

// 获取原图信息
list($owidth, $oheight, $otype) = getimagesize($source);

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

switch($otype){
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($otype){
case 1: imagegif($source_img, $dest); break;
case 2: imagejpeg($source_img, $dest); break;
case 3: imagepng($source_img, $dest); break;
}

return is_file($dest)? true : false;

}

$source = 'source.jpg';
$dest = 'dest.jpg';

$flag = imageMosaics($source, $dest, 176, 98, 273, 197, 4);
echo '<img src="'.$source.'">';
echo '<img src="'.$dest.'">';

?> 查看全部
原理:

对图片中选定区域的每一像素,增加若干宽度及高度,生成矩型。而每一像素的矩型重叠在一起,就形成了马赛克效果。

本例使用GD库的imagecolorat获取像素颜色,使用imagefilledrectangle画矩型。
<?php  
/** 图片局部打马赛克
* @param String $source 原图
* @param Stirng $dest 生成的图片
* @param int $x1 起点横坐标
* @param int $y1 起点纵坐标
* @param int $x2 终点横坐标
* @param int $y2 终点纵坐标
* @param int $deep 深度,数字越大越模糊
* @return boolean
*/
function imageMosaics($source, $dest, $x1, $y1, $x2, $y2, $deep){

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

// 获取原图信息
list($owidth, $oheight, $otype) = getimagesize($source);

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

switch($otype){
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($otype){
case 1: imagegif($source_img, $dest); break;
case 2: imagejpeg($source_img, $dest); break;
case 3: imagepng($source_img, $dest); break;
}

return is_file($dest)? true : false;

}

$source = 'source.jpg';
$dest = 'dest.jpg';

$flag = imageMosaics($source, $dest, 176, 98, 273, 197, 4);
echo '<img src="'.$source.'">';
echo '<img src="'.$dest.'">';

?>

PHP中new self()和new static()有什么区别?

回复

zkbhj 发起了问题 • 1 人关注 • 0 个回复 • 1895 次浏览 • 2018-08-21 09:56 • 来自相关话题

PHP基础面试题(校招版)(凯冰科技出品)

回复

zkbhj 发起了问题 • 1 人关注 • 0 个回复 • 1344 次浏览 • 2018-08-16 19:55 • 来自相关话题

#日常积累# 将一个二维数组按元素类别(大类+小类)进行打散重新排序

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

需求大概如下:
 
有一个N个元素的数组要按下面的要求进行排序,这N个元素分别是一个个手机,每个手机都有自己的品牌属性,假设有两大品牌,华为A和小米B,实体手机会有:
华为手机(A1),
小米又分为
小米8(B2)、
小米6(B3)、
小米MIX(B4)。

然后要实现以下排序规则:

1、当某一类数据(AB两大类或B大类中的小类)连续超过3条时,按顺序将后面另一类型数据插入到第2条之后。优先进行AB类数据之间的相互插入,当A类数据不足时,进行B类中各小类数据的相互插入。
2、特殊规则:前2条数据为B类时,将后面第1条A类数据插入到第一条B类数据之后,后面数据的插入规则沿用2中描述的规则。

eg:
原顺序:B2,B2,A1,B2,B3,B4,A1,A1,A1,A1,B3,B3,B3,B3,B2
新顺序:B2,A1,B2,B2,A1,B3,B4,A1,A1,B3,A1,B3,B3,B2,B3

这个算法可以怎么实现?
 
按这个需求,想了很久并没有想到合适的办法。后来更改了下规则,将AB大类进行穿插打散,然后如果B类数据偏多,再将多余的B类数据,按照B小类进行穿插打散。实现代码如下:
 
/*
*1 华为手机
*3 小米手机5
*4 小米手机6
*5 小米手机8
*8 小米MIX
*9 小米MIX2S
*
*/
function phoneBreakUp($phone_list,$sort_by = 'phone_type')
{

//定义规则
$rule = [
'huawei' => [1],
'xiaomi' => [3,4,5,8,9]
];

//循环每类房源的数据,将华为和小米区分开
foreach ($phone_list as $key => $value) {

if(in_array($value[$sort_by], $rule['huawei'])){
$temp_phone_list['huawei'][] = $value;

}else{
$temp_phone_list['xiaomi'][] = $value;
}
}


//将华为数据和小米数据进行穿插处理
if(count($temp_phone_list['xiaomi']) >= count($temp_phone_list['huawei'])){

//首先把能够穿插的华为数据穿插处理
$break_phone_list = alternateArray($temp_phone_list['xiaomi'],$temp_phone_list['huawei']);

//如果小米类的数量比华为的多,还需要在剩余的小米数据里,进行穿插处理
for($i=2 * count($temp_phone_list['huawei']) + 1 ; $i < count($break_phone_list); $i++ ){
$whole_rent_list[$break_phone_list[$i][$sort_by]][] = $break_phone_list[$i];

}

//将获得的多余的小米数据进行穿插处理
$new_rent_list = alternateArrayMore($whole_rent_list);

//将穿插后的小米数据,替换掉大数组中的后面部分
array_splice($break_phone_list,2 * count($temp_phone_list['huawei']) + 1,count($break_phone_list) - 2 * count($temp_phone_list['huawei']) - 1,$new_rent_list);


}else{
$break_phone_list = alternateArray($temp_phone_list['huawei'],$temp_phone_list['xiaomi']);
}



return $break_phone_list;
}


/**
*
* 将华为和小米数据进行穿插(大类之间穿插)
*
*/
function alternateArray($big_array, $small_array)
{

$total_counts = count($big_array) + count($small_array);
$big_counts = $small_counts = 0;

for($i=0 ; $i< $total_counts; $i++){

if($i%2 == 0 && $small_counts < count($small_array)){
$break_up_list[$i] = $big_array[$big_counts];
$big_counts++;
}elseif($i%2 != 0 && $small_counts < count($small_array)){
$break_up_list[$i] = $small_array[$small_counts];
$small_counts++;
}else{
$break_up_list[$i] = $big_array[$big_counts];
$big_counts++;
}
}

return $break_up_list;

}


/**
*
* 将多余小米数据进行穿插(小类之间穿插)
*
*/
function alternateArrayMore($more_array)
{
//将二维数组依据二维元素的个数进行排序
foreach ($more_array as $key => $value) {
$more_array[$key]['counts'] = count($value);
}
array_multisort($more_array,SORT_DESC);

foreach ($more_array as $key => $value) {
unset($more_array[$key]['counts']);
}



//一直循环,直到取出二维数组中的所有元素
while (!empty($more_array)) {
foreach ($more_array as $key => $value) {
if(!empty($value)){
$break_up_list[] = array_pop($more_array[$key]);
}else{
//如果某个二维数组元素已经取尽,销毁掉
unset($more_array[$key]);
}
}
}

return $break_up_list;
} 查看全部
需求大概如下:
 
有一个N个元素的数组要按下面的要求进行排序,这N个元素分别是一个个手机,每个手机都有自己的品牌属性,假设有两大品牌,华为A和小米B,实体手机会有:
华为手机(A1),
小米又分为
小米8(B2)、
小米6(B3)、
小米MIX(B4)。

然后要实现以下排序规则:

1、当某一类数据(AB两大类或B大类中的小类)连续超过3条时,按顺序将后面另一类型数据插入到第2条之后。优先进行AB类数据之间的相互插入,当A类数据不足时,进行B类中各小类数据的相互插入。
2、特殊规则:前2条数据为B类时,将后面第1条A类数据插入到第一条B类数据之后,后面数据的插入规则沿用2中描述的规则。

eg:
原顺序:B2,B2,A1,B2,B3,B4,A1,A1,A1,A1,B3,B3,B3,B3,B2
新顺序:B2,A1,B2,B2,A1,B3,B4,A1,A1,B3,A1,B3,B3,B2,B3

这个算法可以怎么实现?
 
按这个需求,想了很久并没有想到合适的办法。后来更改了下规则,将AB大类进行穿插打散,然后如果B类数据偏多,再将多余的B类数据,按照B小类进行穿插打散。实现代码如下:
 
/*
*1 华为手机
*3 小米手机5
*4 小米手机6
*5 小米手机8
*8 小米MIX
*9 小米MIX2S
*
*/
function phoneBreakUp($phone_list,$sort_by = 'phone_type')
{

//定义规则
$rule = [
'huawei' => [1],
'xiaomi' => [3,4,5,8,9]
];

//循环每类房源的数据,将华为和小米区分开
foreach ($phone_list as $key => $value) {

if(in_array($value[$sort_by], $rule['huawei'])){
$temp_phone_list['huawei'][] = $value;

}else{
$temp_phone_list['xiaomi'][] = $value;
}
}


//将华为数据和小米数据进行穿插处理
if(count($temp_phone_list['xiaomi']) >= count($temp_phone_list['huawei'])){

//首先把能够穿插的华为数据穿插处理
$break_phone_list = alternateArray($temp_phone_list['xiaomi'],$temp_phone_list['huawei']);

//如果小米类的数量比华为的多,还需要在剩余的小米数据里,进行穿插处理
for($i=2 * count($temp_phone_list['huawei']) + 1 ; $i < count($break_phone_list); $i++ ){
$whole_rent_list[$break_phone_list[$i][$sort_by]][] = $break_phone_list[$i];

}

//将获得的多余的小米数据进行穿插处理
$new_rent_list = alternateArrayMore($whole_rent_list);

//将穿插后的小米数据,替换掉大数组中的后面部分
array_splice($break_phone_list,2 * count($temp_phone_list['huawei']) + 1,count($break_phone_list) - 2 * count($temp_phone_list['huawei']) - 1,$new_rent_list);


}else{
$break_phone_list = alternateArray($temp_phone_list['huawei'],$temp_phone_list['xiaomi']);
}



return $break_phone_list;
}


/**
*
* 将华为和小米数据进行穿插(大类之间穿插)
*
*/
function alternateArray($big_array, $small_array)
{

$total_counts = count($big_array) + count($small_array);
$big_counts = $small_counts = 0;

for($i=0 ; $i< $total_counts; $i++){

if($i%2 == 0 && $small_counts < count($small_array)){
$break_up_list[$i] = $big_array[$big_counts];
$big_counts++;
}elseif($i%2 != 0 && $small_counts < count($small_array)){
$break_up_list[$i] = $small_array[$small_counts];
$small_counts++;
}else{
$break_up_list[$i] = $big_array[$big_counts];
$big_counts++;
}
}

return $break_up_list;

}


/**
*
* 将多余小米数据进行穿插(小类之间穿插)
*
*/
function alternateArrayMore($more_array)
{
//将二维数组依据二维元素的个数进行排序
foreach ($more_array as $key => $value) {
$more_array[$key]['counts'] = count($value);
}
array_multisort($more_array,SORT_DESC);

foreach ($more_array as $key => $value) {
unset($more_array[$key]['counts']);
}



//一直循环,直到取出二维数组中的所有元素
while (!empty($more_array)) {
foreach ($more_array as $key => $value) {
if(!empty($value)){
$break_up_list[] = array_pop($more_array[$key]);
}else{
//如果某个二维数组元素已经取尽,销毁掉
unset($more_array[$key]);
}
}
}

return $break_up_list;
}