>  기사  >  백엔드 개발  >  PHP 확장 개발 노트(10) libpng 라이브러리의 IO 함수를 사용자 정의하여 메모리에 그림을 씁니다.

PHP 확장 개발 노트(10) libpng 라이브러리의 IO 함수를 사용자 정의하여 메모리에 그림을 씁니다.

WBOY
WBOY원래의
2016-08-08 09:21:341385검색

본 QR코드 확장자 dcode를 개발할 때, 생성된 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_write_to_png은 QRcode 데이터를 png에 쓰는 것입니다

이 부분을 주로 보시면 됩니다

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

여기서는 자기를 정의된 쓰기 함수 dcode_png_writerstate 구조에 데이터를 씁니다. 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 함수는 dcode_png_writer을 통해 사용자 정의 쓰기 함수를 설정합니다. 상태와 유사하며 동적으로 메모리를 할당합니다.

png_set_write_fn의 정의는 위에서 언급한 PNG 문서를 참조할 수 있습니다. 사용자 정의 함수도 오류 처리 및 기타 기능을 사용자 정의할 수 있으므로 실제 상황에 따라 error handler을 대신 사용할 수 있습니다. 내부적으로 종료하는 것입니다. 더 많은 관련 코드는 DCode 확장자

를 참조하세요. QRCode 생성 속도는 여전히 매우 빠릅니다. for ($i = 0; $i < 10000; $i ++)$i을 매개변수로 사용하면 3초 안에 10,000개의 QRCode를 생성할 수 있습니다.

저작권 안내: 이 글은 해당 블로거의 원본 글이므로 블로거의 허락 없이 복제할 수 없습니다.

위는 PHP 확장 개발 노트를 소개합니다. (10) libpng 라이브러리에서 IO 함수를 사용자 정의하고 내용을 포함하여 메모리에 그림을 씁니다. PHP 튜토리얼에 관심이 있는 친구들에게 도움이 되기를 바랍니다.

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
이전 기사:nginx-코드-499다음 기사:nginx-코드-499