Maison  >  Article  >  développement back-end  >  Principe et mise en œuvre de la technologie de reconnaissance d'images PHP

Principe et mise en œuvre de la technologie de reconnaissance d'images PHP

墨辰丷
墨辰丷original
2018-05-31 10:04:072842parcourir

Cet article présente principalement le principe et la mise en œuvre de la technologie de reconnaissance d'images PHP. Cette vérification de mot de passe qui est habituellement effectuée a le même objectif. Si vous en avez besoin, vous pouvez en savoir plus.

En fait, la technologie de reconnaissance d'image n'est pas différente de la vérification du mot de passe que nous effectuons habituellement. Les données à vérifier sont stockées à l'avance dans la base de données, puis les données saisies (reconnues) sont comparées aux données contenues. la base de données lorsqu'elle est utilisée. Les données sont comparées, mais la technologie de reconnaissance d'image a une certaine tolérance aux pannes et notre vérification de mot de passe habituelle doit correspondre à 100 %.

Il y a quelques jours, un ami a parlé de créer un jeu pour cliquer sur la loterie et identifier le texte dans l'image. À cette époque, ce qui m'est immédiatement venu à l'esprit était le contrôle js ou le flash comme calque de masque. Je pense que cette méthode est la plus pratique, la plus rapide et la plus efficace. Et elle économise les ressources du serveur, mais la demande est de reconnaître le texte dans l'image via PHP.

Il se trouve que les nouvelles de ces deux jours comprenaient : 1. Le paiement par reconnaissance faciale de Jack Ma ; 2. 12306 a utilisé un nouveau code de vérification, indiquant qu'aucun logiciel de saisie de billets national ne peut être utilisé maintenant, et il a été interdit. dans la journée suivant sa sortie. Puis il m’est arrivé de lire un article sur la technologie de reconnaissance d’images Java ce matin-là. J'ai donc pensé à jeter un œil à la technologie de reconnaissance d'image de PHP.

En fait, la soi-disant reconnaissance d'images n'est plus une nouvelle technologie. Au moins, les informations que j'ai trouvées datent d'il y a longtemps. C'est juste que je n'ai jamais été impliqué dans cet aspect du travail, donc je ne l'ai jamais vu.

Parlons d'abord des exigences de cette expérience : il y a une image avec trois nombres dans trois positions, et il faut extraire la valeur du nombre à la position correspondante. (Les étudiants aux yeux perçants peuvent voir que le code ci-dessous est quelque chose que j'ai pris à quelqu'un d'autre. Oui, je l'ai copié directement de quelqu'un d'autre et je l'ai supprimé. Après tout, je n'ai fait qu'effleurer la surface et je publierai éventuellement le code initial de l'auteur original. .)

class gjPhone
{

  protected $imgPath; // 图片路径
  protected $imgSize; // 图片大小
  protected $hecData; // 分离后数组
  protected $horData; // 横向整理的数据
  protected $verData; // 纵向整理的数据
  function __construct ($path)
  {
    $this->imgPath = $path;
  }

  public function getHec ()
  {
    $size = getimagesize($this->imgPath);
    $res = imagecreatefrompng($this->imgPath);
    for ($i = 0; $i < $size[1]; ++ $i) {
      for ($j = 0; $j < $size[0]; ++ $j) {
        $rgb = imagecolorat($res, $j, $i);
        $rgbarray = imagecolorsforindex($res, $rgb);
        if ($rgbarray[&#39;red&#39;] < 125 || $rgbarray[&#39;green&#39;] < 125 ||
             $rgbarray[&#39;blue&#39;] < 125) {
          $data[$i][$j] = 1;
        } else {
          $data[$i][$j] = 0;
        }
      }
    }
    $this->imgSize = $size;
    $this->hecData = $data;
  }

  public function magHorData ()
  {
    $data = $this->hecData;
    $size = $this->imgSize;
    $z = 0;
    for ($i = 0; $i < $size[1]; ++ $i) {
      if (in_array(&#39;1&#39;, $data[$i])) {
        $z ++;
        for ($j = 0; $j < $size[0]; ++ $j) {
          if ($data[$i][$j] == &#39;1&#39;) {
            $newdata[$z][$j] = 1;
          } else {
            $newdata[$z][$j] = 0;
          }
        }
      }
    }
    return $this->horData = $newdata;
  }

  public function showPhone ($ndatas)
  {
    error_reporting(0);
    $phone = null;
    $d = 0;
    foreach ($ndatas as $key => $val) {
      if (in_array(1, $val)) {
        foreach ($val as $k => $v) {
          $ndArr[$d] .= $v;
        }
      }
      if (! in_array(1, $val)) {
        $d ++;
      }
    }
    foreach ($ndArr as $key01 => $val01) {
      $phone .= $this->initData($val01);
    }
    return $phone;
  }

  /**
   * 初始数据
   */
  public function initData ($numStr)
  {
    $result = null;
    $data = array(
        &#39;1&#39; => &#39;00000000111000000000000001110000000001001000100000000010100011000000000011000110000000000110000100000000010110011000000&#39;,
        &#39;5&#39; => &#39;00000000001000000000000000010000000000100100100000000000101001110000000000100000110000000011000000100000001101000010000&#39;,
        &#39;10&#39; => &#39;00000011100011100000000011001100100100100010010001000110000100100010001100001001000100011000010010001001001001100010100&#39;
    );
    foreach ($data as $key => $val) {
      similar_text($numStr, $val, $pre);
      if ($pre > 95) { // 相似度95%以上
        $result = $key;
        break;
      }
    }
    return $result;
  }
}

$imgurl = &#39;jd.png&#39;;
list ($width, $heght, $type, $attr) = getimagesize($imgurl);
$new_w = 17;
$new_h = 11;
$thisimage = imagecreatetruecolor($new_w, $new_h); // $new_w, $new_h 为裁剪后的图片宽高
$background = imagecolorallocate($thisimage, 255, 255, 255);
imagefilledrectangle($thisimage, 0, 0, $new_w, $new_h, $background);
$oldimg = imagecreatefrompng($imgurl); // 载入原始图片
                    
// 首先定位要取图的位置(这里可以通过前端js或者其他手段定位,由于我这是测试,所以就ps定位并写死了)
$weizhi = array(
    &#39;1&#39; => 165,
    &#39;5&#39; => 308,
    &#39;10&#39; => 456
);

foreach ($weizhi as $wwzz) {
  $src_y = 108;
  imagecopy($thisimage, $oldimg, 0, 0, $wwzz, $src_y, $new_w, $new_h); // $src_y,$new_w为原图中裁剪区域的左上角坐标拷贝图像的一部分将src_im图像中坐标从src_x,src_y开始,宽度为src_w,高度为src_h的一部分拷贝到dst_im图像中坐标为dst_x和dst_y的位置上。
  $tem_png = &#39;tem_1.png&#39;;
  imagepng($thisimage, __DIR__ . &#39;/&#39; . $tem_png); // 通过定位从原图中copy出想要识别的位置并生成新的缓存图,用以后面的图像识别类使用。
  
  $gjPhone = new gjPhone($tem_png); // 实例化类
  $gjPhone->getHec(); // 进行图像像素分离
  $horData = $gjPhone->magHorData(); // 将分离出是数据转成01表示的图像、这里可以根据自己喜好定
  $phone = $gjPhone->showPhone($horData); // 将转换好的01表示的数据与库中的数据进行匹配,匹配度95以上就算成功,库这里由于是做测试就直接写了数组
  echo &#39;| &#39; . $phone . &#39; | &#39;;
}

De ce point de vue, il est effectivement excusable que le code de vérification 12306 a été fissuré. Il n'est pas nécessaire d'être aussi violent verbalement. Continuez simplement à récupérer les images du code de vérification et convertissez-les en données lisibles par votre propre programme et stockez-les dans la base de données, puis faites-les correspondre lors de la vérification. Le principe de paiement par reconnaissance faciale d’Alibaba peut alors être considéré comme compris, mais ils peuvent être très sophistiqués dans ce qu’ils font.

Au début, j'ai vu un formulaire de code de vérification d'Alibaba Cloud. Au début, je pensais que cela pourrait être mieux, mais maintenant, il semble que tant que vous le souhaitez, il peut réellement être craqué.

D'accord, voici le code original.

/**
 * 电话号码识别.
 * @author by zsc for 2010.03.24
 */
class gjPhone
{

  protected $imgPath; // 图片路径
  protected $imgSize; // 图片大小
  protected $hecData; // 分离后数组
  protected $horData; // 横向整理的数据
  protected $verData; // 纵向整理的数据
  function __construct ($path)
  {
    $this->imgPath = $path;
  }

  /**
   * 颜色分离转换...
   *
   * @param unknown_type $path      
   * @return unknown
   */
  public function getHec ()
  {
    $size = getimagesize($this->imgPath);
    $res = imagecreatefrompng($this->imgPath);
    for ($i = 0; $i < $size[1]; ++ $i) {
      for ($j = 0; $j < $size[0]; ++ $j) {
        $rgb = imagecolorat($res, $j, $i);
        $rgbarray = imagecolorsforindex($res, $rgb);
        if ($rgbarray[&#39;red&#39;] < 125 || $rgbarray[&#39;green&#39;] < 125 ||
             $rgbarray[&#39;blue&#39;] < 125) {
          $data[$i][$j] = 1;
        } else {
          $data[$i][$j] = 0;
        }
      }
    }
    $this->imgSize = $size;
    $this->hecData = $data;
  }

  /**
   * 颜色分离后的数据横向整理...
   *
   * @return unknown
   */
  public function magHorData ()
  {
    $data = $this->hecData;
    $size = $this->imgSize;
    $z = 0;
    for ($i = 0; $i < $size[1]; ++ $i) {
      if (in_array(&#39;1&#39;, $data[$i])) {
        $z ++;
        for ($j = 0; $j < $size[0]; ++ $j) {
          if ($data[$i][$j] == &#39;1&#39;) {
            $newdata[$z][$j] = 1;
          } else {
            $newdata[$z][$j] = 0;
          }
        }
      }
    }
    return $this->horData = $newdata;
  }

  /**
   * 整理纵向数据...
   *
   * @return unknown
   */
  public function magVerData ($newdata)
  {
    for ($i = 0; $i < 132; ++ $i) {
      for ($j = 1; $j < 13; ++ $j) {
        $ndata[$i][$j] = $newdata[$j][$i];
      }
    }
    
    $sum = count($ndata);
    $c = 0;
    for ($a = 0; $a < $sum; $a ++) {
      $value = $ndata[$a];
      if (in_array(1, $value)) {
        $ndatas[$c] = $value;
        $c ++;
      } elseif (is_array($ndatas)) {
        $b = $c - 1;
        if (in_array(1, $ndatas[$b])) {
          $ndatas[$c] = $value;
          $c ++;
        }
      }
    }
    
    return $this->verData = $ndatas;
  }

  /**
   * 显示电话号码...
   *
   * @return unknown
   */
  public function showPhone ($ndatas)
  {
    $phone = null;
    $d = 0;
    foreach ($ndatas as $key => $val) {
      if (in_array(1, $val)) {
        foreach ($val as $k => $v) {
          $ndArr[$d] .= $v;
        }
      }
      if (! in_array(1, $val)) {
        $d ++;
      }
    }
    foreach ($ndArr as $key01 => $val01) {
      $phone .= $this->initData($val01);
    }
    return $phone;
  }

  /**
   * 分离显示...
   *
   * @param unknown_type $dataArr      
   */
  function drawWH ($dataArr)
  {
    if (is_array($dataArr)) {
      foreach ($dataArr as $key => $val) {
        foreach ($val as $k => $v) {
          if ($v == 0) {
            $c .= "<font color=&#39;#FFFFFF&#39;>" . $v . "</font>";
          } else {
            $c .= $v;
          }
        }
        $c .= "<br/>";
      }
    }
    echo $c;
  }

  /**
   * 初始数据...
   *
   * @param unknown_type $numStr      
   * @return unknown
   */
  public function initData ($numStr)
  {
    $result = null;
    $data = array(
        0 => &#39;000011111000001111111110011000000011110000000001110000000001110000000001110000000001011000000011011100000111000111111100000001110000&#39;,
        1 => &#39;011000000000011000000000111111111111111111111111&#39;,
        2 => &#39;001000000011011000000111110000001101110000011001110000011001110000110001111001100001011111100001000110000001&#39;,
        3 => &#39;001000000010011000000011110000000001110000000001110000110001110000110001011001110011011111011111000110001100&#39;,
        4 => &#39;000000001100000000111100000001111100000011101100000111001100001100001100011000001100111111111111111111111111000000001100000000000100&#39;,
        5 => &#39;111111000001111111000001110001000001110001000001110001100001110001100001110000110011110000111111000000001100&#39;,
        6 => &#39;000011111000001111111110011000110011110001100001110001100001110001100001110001100001010001110011010000111111000000001100&#39;,
        7 => &#39;110000000000110000000111110000111111110001110000110111000000111100000000111000000000111000000000&#39;,
        8 => &#39;000100011110011111111111110011100001110001100001110001100001110001100001110011100001011111111111000100011110&#39;,
        9 => &#39;001111000000011111100001110000110001110000110001110000110001110000110001011000100001011111100111000111111110000001110000&#39;
    );
    foreach ($data as $key => $val) {
      similar_text($numStr, $val, $pre);
      if ($pre > 95) { // 相似度95%以上
        $result = $key;
        break;
      }
    }
    return $result;
  }
}

$imgPath = "http://bj.ganji.com/tel/5463013757650d6c5e31093e563c51315b6c5c6c5237.png";
$gjPhone = new gjPhone($imgPath);
// 进行颜色分离
$gjPhone->getHec();
// 画出横向数据
$horData = $gjPhone->magHorData();
echo "===============横向数据==============<br/><br/><br/>";
$gjPhone->drawWH($horData);
// 画出纵向数据
$verData = $gjPhone->magVerData($horData);
echo "<br/><br/><br/>===============纵向数据==============< br/><br/><br/>";
$gjPhone->drawWH($verData);

// 输出电话
$phone = $gjPhone->showPhone($verData);
echo "<br/><br/><br/>===============电话==============<br /><br/><br/>" . $phone;

Ce qui précède représente l'intégralité du contenu de cet article, j'espère qu'il sera utile à l'étude de chacun.


Recommandations associées :

phpAnalyse d'instance de téléchargement asynchrone d'iframe de fichier de formulaire

Analyse détaillée de la mise en œuvre de PHP SFTP des fonctions de téléchargement et de téléchargement

Définition et utilisation du modèle de stratégie PHP (explication détaillée)

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn