ホームページ  >  記事  >  バックエンド開発  >  PHPで画像を最適化して幅と高さを取得する方法

PHPで画像を最適化して幅と高さを取得する方法

醉折花枝作酒筹
醉折花枝作酒筹転載
2021-06-18 17:26:182316ブラウズ

この記事ではPHPで画像の幅と高さを最適化する方法を紹介します。一定の参考値があるので、困っている友達が参考になれば幸いです。

PHPで画像を最適化して幅と高さを取得する方法

php 画像取得の幅と高さの最適化について

要件

フロントエンド要件への対応: 記事の詳細を入力するとき、すべての画像をプレースホルダーに置き換える必要があり、プレースホルダーは画像情報に対応する必要があります (主に幅と高さを知る必要があります)

目的:画像をクリックしたときにフローティング ウィンドウ効果を作成するには

実装計画

最適化前

定期的に画像を照合し、ループして各画像の幅と高さを取得します

問題: 記事に含まれる画像の数が少ない場合、上記の操作は大きな問題ではありません。ただし、画像が多すぎると効率が非常に低くなります。

コードは次のとおりです。

        preg_match_all(&#39;/<img.*? src="(.*?)".*?>/is&#39;, $str, $matchs);
       
        if(!empty($matchs[0])){
            $pics = [];
            $i = 0;
            foreach ($matchs[0] as $key => $m) {
                $fileInfo = file_get_contents($matchs[1][$key] . &#39;?x-oss-process=image/info&#39;);
                $fileInfo = json_decode($fileInfo, true);
                $data[&#39;Width&#39;] = $fileInfo[&#39;ImageWidth&#39;][&#39;value&#39;];
                $data[&#39;Height&#39;] = $fileInfo[&#39;ImageHeight&#39;][&#39;value&#39;];
                    
                $imgs[$i][&#39;ref&#39;] = &#39;<!--IMG#&#39; . $key . &#39;-->&#39;;
                $imgs[$i][&#39;pixel&#39;] = $data[&#39;Width&#39;] . &#39;*&#39; . $data[&#39;Height&#39;];
                preg_match(&#39;/alt="(.*?)"/i&#39;, $matchs[0][$key], $mt);
                $imgs[$i][&#39;alt&#39;] = isset($mt[1]) ? $mt[1] : &#39;&#39;;   //图片alt
                $imgs[$i][&#39;src&#39;] = $matchs[1][$key];                //图片地址
                $str = str_replace($m, &#39;<!--IMG#&#39; . $key . &#39;-->&#39;, $str);
                $i++;

            }
        }

最適化のアイデア

画像を取得する方法があるかどうか疑問に思っています。素早く?インターネット上で、基本的に画像全体をダウンロード/読み取るのではなく、画像のファイル情報の一部を読み取るという情報をいくつか見つけました。クラス ライブラリ: [https://github.com/tommoor/fastimage](https://github.com/tommoor/fastimage) を見つけて試してみました。以前のアイデア (写真の完全なダウンロード) と比較すると、確かにパフォーマンスが向上しています。興味のある友達は試してみるのも良いですが、1枚の写真から情報を得たい場合には、やはりこちらがおすすめです。しかし、バッチ実装では目標を達成できないようです。

上記の操作を分析すると、実際には、画像リソースを取得するために遅いプロセスがまだループ内に残っているはずです。では、考え方を変えて、写真をまとめて取得しても大丈夫でしょうか?コードを上に進みます

preg_match_all(&#39;/<img.*? src="(.*?)".*?>/is&#39;, $str, $matchs);

if(!empty($matchs[0])){
    //$time = microtime(true);
    //echo  &#39;  ---- start &#39; . PHP_EOL;

    foreach ($matchs[0] as $key => $m) {
        $urls[] = $matchs[1][$key] . &#39;?x-oss-process=image/info&#39;;
    }
    $imageInfos = batchCurl($urls);

    $i = 0;
    foreach ($matchs[0] as $key => $m) {
        $image = json_decode($imageInfos[$key], true);
        $_img[&#39;Width&#39;] = $width= $image[&#39;ImageWidth&#39;][&#39;value&#39;];
        $_img[&#39;Height&#39;] = $height = $image[&#39;ImageHeight&#39;][&#39;value&#39;];

        $imgs[$i][&#39;ref&#39;] = &#39;<!--IMG#&#39; . $key . &#39;-->&#39;;
        $imgs[$i][&#39;pixel&#39;] = $_img[&#39;Width&#39;] . &#39;*&#39; . $_img[&#39;Height&#39;];
        preg_match(&#39;/alt="(.*?)"/i&#39;, $matchs[0][$key], $mt);
        $imgs[$i][&#39;alt&#39;] = isset($mt[1]) ? $mt[1] : &#39;&#39;;   //图片alt
        $imgs[$i][&#39;src&#39;] = $matchs[1][$key];                //图片地址
        $str = str_replace($m, &#39;<!--IMG#&#39; . $key . &#39;-->&#39;, $str);

        $i++;
    }
    //echo  " ---- end  px in " . (microtime(true)-$time) . " seconds \n";
    //exit;
}
        
function batchCurl($urls)
{
    $res = $conn = [];

    // 创建批处理cURL句柄
    $mh = curl_multi_init();

    foreach ($urls as $i => $url) {
        // 创建一对cURL资源
        $conn[$i] = curl_init();
        // 设置URL和相应的选项
        curl_setopt($conn[$i], CURLOPT_URL, $url);
        curl_setopt($conn[$i], CURLOPT_HEADER, 0);
        curl_setopt($conn[$i], CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($conn[$i], CURLOPT_TIMEOUT, 10);
        // 302跳转
        curl_setopt($conn[$i], CURLOPT_FOLLOWLOCATION, 1);
        // 增加句柄
        curl_multi_add_handle($mh, $conn[$i]);
    }
    $active = null;
    //防卡死写法:执行批处理句柄
    do {
        $mrc = curl_multi_exec($mh, $active);
    } while ($mrc == CURLM_CALL_MULTI_PERFORM);

    while ($active && $mrc == CURLM_OK) {
        if (curl_multi_select($mh) != -1) {
            do {
                $mrc = curl_multi_exec($mh, $active);
            } while ($mrc == CURLM_CALL_MULTI_PERFORM);
        }
    }
    foreach ($urls as $i => $url) {
        //获取当前解析的cURL的相关传输信息
        $info = curl_multi_info_read($mh);
        //获取请求头信息
        $heards = curl_getinfo($conn[$i]);
        //获取输出的文本流
        $res[$i] = curl_multi_getcontent($conn[$i]);
        // 移除curl批处理句柄资源中的某个句柄资源
        curl_multi_remove_handle($mh, $conn[$i]);
        //关闭cURL会话
        curl_close($conn[$i]);
    }
    //关闭全部句柄
    curl_multi_close($mh);

    return $res;
}

3。パフォーマンスをテストすると、20枚の写真の効率はほぼ第2レベルに達します

![image](/img/bVcKCF2)

推奨学習:

php ビデオ チュートリアル

以上がPHPで画像を最適化して幅と高さを取得する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はsegmentfault.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。