본 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>
dcode_png_writer
는 png 데이터를 쓰기 위한 사용자 정의 콜백 함수입니다. dcode_write_to_png
은 QRcode 데이터를 png에 쓰는 것입니다이 부분을 주로 보시면 됩니다
<code>png_set_write_fn(png_ptr, &state, &dcode_png_writer, NULL);</code>
여기서는 자기를 정의된 쓰기 함수 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
함수는 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 튜토리얼에 관심이 있는 친구들에게 도움이 되기를 바랍니다.