ホームページ  >  記事  >  バックエンド開発  >  PHP拡張開発ノート(10) libpngライブラリのIO関数をカスタマイズしてメモリに画像を書き込む

PHP拡張開発ノート(10) libpngライブラリのIO関数をカスタマイズしてメモリに画像を書き込む

WBOY
WBOYオリジナル
2016-08-08 09:21:341385ブラウズ

このQRコード拡張子コードを開発する場合、ファイルを直接生成するのではなく、生成されたQRコードのpng画像を文字列の形で呼び出し元に返す必要があります。これは、ファイルを操作する必要がないため、より便利です。ファイルの操作は完全にユーザーに任されています。

画像の生成には libpng ライブラリが使用されます。libpng に関するドキュメントについては、こちらの png ドキュメントにアクセスしてください。このライブラリを使用して Ubuntu14.04 で拡張機能をコンパイルしたときも、小さな問題が発生しました。ubuntu 14 の行 0 にある Unknown の png_create_write_struct です。インターネットで検索したところ、これは依然として非常に一般的です。

コードは以下に簡単にリストされています:

<code><span>/** {{{ dcode_png_writer()
 * function is custom png_write callback function
 * Return void */</span><span>static</span><span>void</span> dcode_png_writer(png_structp png_ptr, png_bytep data, png_size_t length)
{
    png_mem_encode* p = (png_mem_encode*) png_get_io_ptr(png_ptr);
    size_t nsize = p->size + length;

    <span>if</span> (p->buffer)
        p->buffer = erealloc(p->buffer, nsize);
    <span>else</span>
        p->buffer = emalloc(nsize);

    <span>if</span> (!p->buffer)
    {
        png_error(png_ptr, <span>"PNG allocate memory error"</span>);
        <span>exit</span>(FAILURE);
    }

    <span>memcpy</span>(p->buffer + p->size, data, length);
    p->size += length;
}
<span>/* }}} */</span></code>
<code><span>/** {{{ dcode_write_to_png()
 * write qrcode struct to memory
 * Return char* */</span><span>static</span><span>char</span>* dcode_write_to_png(QRcode *qrcode, <span>int</span> size, <span>int</span> margin, <span>int</span> *pp_len)
{

    png_structp png_ptr;
    png_infop info_ptr;

    <span>unsigned</span><span>char</span> *row, *p, *q;
    <span>int</span> x, y, xx, yy, bit;
    <span>int</span> realwidth;

    realwidth = (qrcode->width + margin * <span>2</span>) * size;
    <span>int</span> row_fill_len = (realwidth + <span>7</span>) / <span>8</span>;

    png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
    <span>if</span> (png_ptr == NULL)
    {
        php_error(E_ERROR, <span>"Failed to initialize PNG writer"</span>);
        <span>return</span> NULL;
    }

    info_ptr = png_create_info_struct(png_ptr);
    <span>if</span> (info_ptr == NULL)
    {
        php_error(E_ERROR, <span>"Failed to initialize PNG info"</span>);
        <span>return</span> NULL;
    }

    <span>if</span> (setjmp(png_jmpbuf(png_ptr)))
    {
        png_destroy_write_struct(&png_ptr, &info_ptr);
        php_error(E_ERROR, <span>"Failed to set PNG jmpbuf"</span>);
        <span>return</span> NULL;
    }

    row = (<span>unsigned</span><span>char</span> *) emalloc(row_fill_len);
    <span>if</span> (row == NULL)
    {
        png_destroy_write_struct(&png_ptr, &info_ptr);
        php_error(E_ERROR, <span>"Failed to allocate memory"</span>);
        <span>return</span> NULL;
    }

    png_mem_encode state = {NULL, <span>0</span>};
    png_set_write_fn(png_ptr, &state, &dcode_png_writer, NULL);

    png_set_IHDR(png_ptr,
                info_ptr,
                realwidth,
                realwidth,
                <span>1</span>,
                PNG_COLOR_TYPE_GRAY,
                PNG_INTERLACE_NONE,
                PNG_COMPRESSION_TYPE_DEFAULT,
                PNG_FILTER_TYPE_DEFAULT);

    png_write_info(png_ptr, info_ptr);
    <span>memset</span>(row, <span>0xff</span>, (realwidth + <span>7</span>) / <span>8</span>);
    <span>for</span>(y = <span>0</span>; y < margin * size; y ++) {
        png_write_row(png_ptr, row);
    }

    p = qrcode->data;
    <span>for</span>(y = <span>0</span>; y < qrcode->width; y ++) {
        bit = <span>7</span>;
        <span>memset</span>(row, <span>0xff</span>, (realwidth + <span>7</span>) / <span>8</span>);
        q = row;
        q += margin * size / <span>8</span>;
        bit = <span>7</span> - (margin * size % <span>8</span>);
        <span>for</span>(x = <span>0</span>; x < qrcode->width; x ++) {
            <span>for</span>(xx = <span>0</span>; xx <size; xx ++) {
                *q ^= (*p & <span>1</span>) << bit;
                bit--;
                <span>if</span>(bit < <span>0</span>) {
                    q++;
                    bit = <span>7</span>;
                }
            }
            p++;
        }
        <span>for</span>(yy = <span>0</span>; yy < size; yy ++ ) {
            png_write_row(png_ptr, row);
        }
    }

    <span>memset</span>(row, <span>0xff</span>, (realwidth + <span>7</span>) / <span>8</span>);
    <span>for</span>(y = <span>0</span>; y < margin * size; y ++) {
        png_write_row(png_ptr, row);
    }

    png_write_end(png_ptr, info_ptr);
    png_destroy_write_struct(&png_ptr, &info_ptr);

    efree(row);

    <span>char</span> *bin_data = NULL;
    <span>if</span> (state.buffer) {
        bin_data = estrndup(state.buffer, state.size);
        *pp_len = state.size;
        efree(state.buffer);
    }

    <span>return</span> bin_data;
}
<span>/** }}} */</span></code>
  1. 最初の関数 dcode_png_writer は、png データを書き込むためのカスタム コールバック関数です。
  2. dcode_png_writer 是自定义的写 png 数据的 callback 函数。
  3. 第二个函数 dcode_write_to_png 是将 QRcode 数据写入 png

主要可以看下这个部分

<code>png_set_write_fn(png_ptr, &state, &dcode_png_writer, NULL);</code>

这个地方就是调用了自定义的 write 函数 dcode_png_writer,将数据写到了 state 这个结构体里,state 结构体如下

<code><span>typedef</span><span>struct</span> _png_mem_encode {
    <span>char</span> *buffer;
    size_t size;
} png_mem_encode ;</code>

png_set_write_fn 函数设置了自定义的 write 函数,通过 dcode_png_writer 来像 state 写入数据,动态的来分配内存。

关于 png_set_write_fn 的定义,可以参看上面提到的 PNG 文档,自定义函数还可以自定义错误处理等功能,这样可以根据实际情况来接管 error handler 而不是让其在内部退出。更多的相关代码请看 DCode 扩展

生成 QRCode 的速度还是很快的,如果用 for ($i = 0; $i < 10000; $i ++)$i 2 番目の関数 dcode_write_to_png は QRcode データを png に書き込むことです

主にこの部分を見ることができますrrreee
この場所はカスタム関数を呼び出します。関数 dcode_png_writerstate 構造体にデータを書き込みます。 state 構造体は次のとおりです

rrreee

png_set_write_fn 関数は次のように設定します。カスタム書き込み関数は、dcode_png_writer を通じてデータをステートに書き込み、メモリを動的に割り当てます。

png_set_write_fn の定義については、上記の PNG ドキュメントを参照してください。カスタム関数は、error handler をカスタマイズすることもできます。 > 内部的に終了させるのではなく、実際の状況に応じて引き継がれます。関連コードの詳細については、DCode 拡張機能を参照してください

for ($i = 0; $i > を使用すると、QRCode の生成速度は依然として非常に高速です。 $i をパラメータとして指定すると、3 秒で 10,000 個を生成できます。

🎜🎜🎜著作権声明: この記事はブロガーによるオリジナルの記事であり、ブロガーの許可なく複製することはできません。 🎜🎜 🎜 以上、PHP拡張開発ノート(10)libpngライブラリのIO関数をカスタマイズしてメモリに画像を書き込む方法を紹介しましたが、PHPチュートリアルに興味のある友人の参考になれば幸いです。 🎜 🎜 🎜
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
前の記事:nginxコード-499次の記事:nginxコード-499