Android: сохранить растровое изображение в формате BMP

У меня есть растровое изображение в памяти, и мне нужно сохранить его в файл BMP (используя формат файла BMP).

Есть ли способ сделать это на Android?

(Я прочитал много постов, предлагающих использовать формат png - который без потерь - но это не то, что мне нужно: мне действительно нужен формат bmp).

У меня уже есть код, чтобы сохранить его в формате JPEG или PNG, используя метод Bitmap.compress:

/**
 * Save data to file using format.
 * When format is null : the bitmap will be saved in bmp format
 **/

public void writeBitmapToFile(Bitmap data, File file, Bitmap.CompressFormat format) {
    FileOutputStream os = null;
    try {
        os = new FileOutputStream(file);
        if(format==null){

            //TODO : write data to file using the bmp format

        }else{
            data.compress(format, 100, os); //ok for JPEG and PNG
        }
        os.flush();
    } catch (Exception e) {
        //irrelevant code
    } finally {
        //irrelevant code
    }
}

3 ответа

Решение

(Я отвечаю на свой вопрос)

Вот мое текущее решение. Он получен из этого источника: https://github.com/ultrakain/AndroidBitmapUtil (благодаря ultrakain и @Francescoverheye)

Я просто исправляю небольшую ошибку в вычислении фиктивных байтов, которые должны быть добавлены к каждой строке (чтобы длина каждой строки в байтах была кратна 4 (как того требуют спецификации формата bmp).

Я также внес некоторые изменения, чтобы улучшить выступления.

import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;

import android.graphics.Bitmap;
import android.util.Log;

public class AndroidBmpUtil {

    private static final int BMP_WIDTH_OF_TIMES = 4;
    private static final int BYTE_PER_PIXEL = 3;

    /**
     * Android Bitmap Object to Window's v3 24bit Bmp Format File
     * @param orgBitmap
     * @param filePath
     * @return file saved result
     */
    public static boolean save(Bitmap orgBitmap, String filePath) throws IOException {
        long start = System.currentTimeMillis();
        if(orgBitmap == null){
            return false;
        }

        if(filePath == null){
            return false;
        }

        boolean isSaveSuccess = true;

        //image size
        int width = orgBitmap.getWidth();
        int height = orgBitmap.getHeight();

        //image dummy data size
        //reason : the amount of bytes per image row must be a multiple of 4 (requirements of bmp format)
        byte[] dummyBytesPerRow = null;
        boolean hasDummy = false;
        int rowWidthInBytes = BYTE_PER_PIXEL * width; //source image width * number of bytes to encode one pixel.
        if(rowWidthInBytes%BMP_WIDTH_OF_TIMES>0){
            hasDummy=true;
            //the number of dummy bytes we need to add on each row
            dummyBytesPerRow = new byte[(BMP_WIDTH_OF_TIMES-(rowWidthInBytes%BMP_WIDTH_OF_TIMES))];
            //just fill an array with the dummy bytes we need to append at the end of each row
            for(int i = 0; i < dummyBytesPerRow.length; i++){
                dummyBytesPerRow[i] = (byte)0xFF;
            }
        }

        //an array to receive the pixels from the source image
        int[] pixels = new int[width * height];

        //the number of bytes used in the file to store raw image data (excluding file headers)
        int imageSize = (rowWidthInBytes+(hasDummy?dummyBytesPerRow.length:0)) * height;
        //file headers size
        int imageDataOffset = 0x36; 

        //final size of the file
        int fileSize = imageSize + imageDataOffset;

        //Android Bitmap Image Data
        orgBitmap.getPixels(pixels, 0, width, 0, 0, width, height);

        //ByteArrayOutputStream baos = new ByteArrayOutputStream(fileSize);
        ByteBuffer buffer = ByteBuffer.allocate(fileSize);

        /**
         * BITMAP FILE HEADER Write Start
         **/
        buffer.put((byte)0x42);
        buffer.put((byte)0x4D);

        //size
        buffer.put(writeInt(fileSize));

        //reserved
        buffer.put(writeShort((short)0));
        buffer.put(writeShort((short)0));

        //image data start offset
        buffer.put(writeInt(imageDataOffset));

        /** BITMAP FILE HEADER Write End */

        //*******************************************

        /** BITMAP INFO HEADER Write Start */
        //size
        buffer.put(writeInt(0x28));

        //width, height
        //if we add 3 dummy bytes per row : it means we add a pixel (and the image width is modified.
        buffer.put(writeInt(width+(hasDummy?(dummyBytesPerRow.length==3?1:0):0)));
        buffer.put(writeInt(height));

        //planes
        buffer.put(writeShort((short)1));

        //bit count
        buffer.put(writeShort((short)24));

        //bit compression
        buffer.put(writeInt(0));

        //image data size
        buffer.put(writeInt(imageSize));

        //horizontal resolution in pixels per meter
        buffer.put(writeInt(0));

        //vertical resolution in pixels per meter (unreliable)
        buffer.put(writeInt(0));

        buffer.put(writeInt(0));

        buffer.put(writeInt(0));

        /** BITMAP INFO HEADER Write End */

        int row = height;
        int col = width;
        int startPosition = (row - 1) * col;
        int endPosition = row * col;
        while( row > 0 ){
            for(int i = startPosition; i < endPosition; i++ ){
                buffer.put((byte)(pixels[i] & 0x000000FF));
                buffer.put((byte)((pixels[i] & 0x0000FF00) >> 8));
                buffer.put((byte)((pixels[i] & 0x00FF0000) >> 16));
            }
            if(hasDummy){
                buffer.put(dummyBytesPerRow);
            }
            row--;
            endPosition = startPosition;
            startPosition = startPosition - col;
        }

        FileOutputStream fos = new FileOutputStream(filePath);
        fos.write(buffer.array());
        fos.close();
        Log.v("AndroidBmpUtil" ,System.currentTimeMillis()-start+" ms");

        return isSaveSuccess;
    }

    /**
     * Write integer to little-endian
     * @param value
     * @return
     * @throws IOException
     */
    private static byte[] writeInt(int value) throws IOException {
        byte[] b = new byte[4];

        b[0] = (byte)(value & 0x000000FF);
        b[1] = (byte)((value & 0x0000FF00) >> 8);
        b[2] = (byte)((value & 0x00FF0000) >> 16);
        b[3] = (byte)((value & 0xFF000000) >> 24);

        return b;
    }

    /**
     * Write short to little-endian byte array
     * @param value
     * @return
     * @throws IOException
     */
    private static byte[] writeShort(short value) throws IOException {
        byte[] b = new byte[2];

        b[0] = (byte)(value & 0x00FF);
        b[1] = (byte)((value & 0xFF00) >> 8);

        return b;
    }
}

Код для преобразования растрового объекта Android в 8-битный BMP-файл. Он получен из этого проекта C#: https://www.codeproject.com/articles/70442/c-rgb-to-palette-based-bit-greyscale-bitmap-clas

import android.graphics.Bitmap;
import android.graphics.Color;


public class BitmapConvertor {

    private  byte[] Color_palette = new byte[1024]; //a palette containing 256 colors
   private   byte[] BMP_File_Header = new byte[14];
   private  byte[] DIB_header = new byte[40];
  private   byte[] Bitmap_Data = null;


    //returns a byte array of a grey scale bitmap image
    public    byte[] CreateGrayBitmapArray(Bitmap Image) {
        try {
            create_parts(Image);
            //Create the array
            byte[] bitmap_array = new byte[BMP_File_Header.length + DIB_header.length
                    + Color_palette.length + Bitmap_Data.length];
            Copy_to_Index(bitmap_array, BMP_File_Header, 0);
            Copy_to_Index(bitmap_array, DIB_header, BMP_File_Header.length);
            Copy_to_Index(bitmap_array, Color_palette, BMP_File_Header.length + DIB_header.length);
            Copy_to_Index(bitmap_array, Bitmap_Data, BMP_File_Header.length + DIB_header.length + Color_palette.length);

            return bitmap_array;
        } catch (Exception e) {
            return null; //return a null single byte array if fails
        }
    }


    //creates byte array of 256 color grayscale palette
    private   byte[] create_palette() {
        byte[] color_palette = new byte[1024];
        for (int i = 0; i < 256; i++) {
            color_palette[i * 4 + 0] = (byte) (i); //bule
            color_palette[i * 4 + 1] = (byte) (i); //green
            color_palette[i * 4 + 2] = (byte) (i); //red
            color_palette[i * 4 + 3] = (byte) 0; //padding
        }
        return color_palette;
    }


    //adds dtata of Source array to Destinition array at the Index
   private   boolean Copy_to_Index(byte[] destination, byte[] source, int index) {
        try {
            for (int i = 0; i < source.length; i++) {
                destination[i + index] = source[i];
            }
            return true;
        } catch (Exception e) {
            return false;
        }
    }


    //create different part of a bitmap file
  private    void create_parts(Bitmap img) {
        //Create Bitmap Data
        Bitmap_Data = ConvertToGrayscale(img);
        //Create Bitmap File Header (populate BMP_File_Header array)
        Copy_to_Index(BMP_File_Header, new byte[]{(byte) 'B', (byte) 'M'}, 0); //magic number
        Copy_to_Index(BMP_File_Header, writeInt(BMP_File_Header.length
                + DIB_header.length + Color_palette.length + Bitmap_Data.length), 2); //file size
        Copy_to_Index(BMP_File_Header, new byte[]{(byte) 'M', (byte) 'C', (byte) 'A', (byte) 'T'}, 6); //reserved for application generating the bitmap file (not imprtant)
        Copy_to_Index(BMP_File_Header, writeInt(BMP_File_Header.length
                + DIB_header.length + Color_palette.length), 10); //bitmap raw data offset
        //Create DIB Header (populate DIB_header array)
        Copy_to_Index(DIB_header, writeInt(DIB_header.length), 0); //DIB header length
        Copy_to_Index(DIB_header, writeInt(((Bitmap) img).getWidth()), 4); //image width
        Copy_to_Index(DIB_header, writeInt(((Bitmap) img).getHeight()), 8); //image height
        Copy_to_Index(DIB_header, new byte[]{(byte) 1, (byte) 0}, 12); //color planes. N.B. Must be set to 1
        Copy_to_Index(DIB_header, new byte[]{(byte) 8, (byte) 0}, 14); //bits per pixel
        Copy_to_Index(DIB_header, writeInt(0), 16); //compression method N.B. BI_RGB = 0
        Copy_to_Index(DIB_header, writeInt(Bitmap_Data.length), 20); //lenght of raw bitmap data
        Copy_to_Index(DIB_header, writeInt(1000), 24); //horizontal reselution N.B. not important
        Copy_to_Index(DIB_header, writeInt(1000), 28); //vertical reselution N.B. not important
        Copy_to_Index(DIB_header, writeInt(256), 32); //number of colors in the palette
        Copy_to_Index(DIB_header, writeInt(0), 36); //number of important colors used N.B. 0 = all colors are imprtant
        //Create Color palett
        Color_palette = create_palette();
    }


    //convert the color pixels of Source image into a grayscale bitmap (raw data)
  private    byte[] ConvertToGrayscale(Bitmap Source) {
        Bitmap source = (Bitmap) Source;
        int padding = (source.getWidth() % 4) != 0 ? 4 - (source.getWidth() % 4) : 0; //determine padding needed for bitmap file
        byte[] bytes = new byte[source.getWidth() * source.getHeight() + padding * source.getHeight()]; //create array to contain bitmap data with paddin
        for (int y = 0; y < source.getHeight(); y++) {
            for (int x = 0; x < source.getWidth(); x++) {
                int pixel = source.getPixel(x, y);
                int g = (int) (0.3 * Color.red(pixel) + 0.59 * Color.green(pixel) + 0.11 * Color.blue(pixel)); //grayscale shade corresponding to rgb
                bytes[(source.getHeight() - 1 - y) * source.getWidth() + (source.getHeight() - 1 - y) * padding + x] = (byte) g;
            }
            //add the padding
            for (int i = 0; i < padding; i++) {
                bytes[(source.getHeight() - y) * source.getWidth() + (source.getHeight() - 1 - y) * padding + i] = (byte) 0;
            }
        }
        return bytes;
    }


    /**
     * Write integer to little-endian
     *
     * @param value
     * @return
     * @throws IOException
     */
    private  byte[] writeInt(int value) {
        byte[] b = new byte[4];

        b[0] = (byte) (value & 0x000000FF);
        b[1] = (byte) ((value & 0x0000FF00) >> 8);
        b[2] = (byte) ((value & 0x00FF0000) >> 16);
        b[3] = (byte) ((value & 0xFF000000) >> 24);

        return b;
    }
}

К сожалению, в моем случае приведенные выше ответы не работали полностью для изображений в градациях серого. Не знаю почему.

Вот почему я написал это (я также использую ByteBuffer за спектакль):

import android.graphics.Bitmap;
import android.util.Log;

import java.io.FileOutputStream;
import java.io.OutputStream;
import java.nio.ByteBuffer;

public class BmpFile {
    // Private constants
    private final static int BITMAPFILEHEADER_SIZE = 14;
    private final static int BITMAPINFOHEADER_SIZE = 40;
    // Private variable declaration
    // Bitmap file header
    private byte bitmapFileHeader[] = new byte[14];
    private byte bfType[] = {'B', 'M'};
    private int bfSize = 0;
    private int bfReserved1 = 0;
    private int bfReserved2 = 0;
    private int bfOffBits = BITMAPFILEHEADER_SIZE + BITMAPINFOHEADER_SIZE;
    // Bitmap info header
    private byte bitmapInfoHeader[] = new byte[40];
    private int biSize = BITMAPINFOHEADER_SIZE;
    private int biWidth = 0;
    private int biHeight = 0;
    private int biPlanes = 1;
    private int biBitCount = 24;
    private int biCompression = 0;
    private int biSizeImage = 0x030000;
    private int biXPelsPerMeter = 0x0;
    private int biYPelsPerMeter = 0x0;
    private int biClrUsed = 0;
    private int biClrImportant = 0;
    // Bitmap raw data
    private int pixels[];
    // File section
    private ByteBuffer buffer = null;
    private OutputStream outputStream;

    // Default constructor
    public BmpFile() {
    }

    public void saveBitmap(
        String parFilename,
        Bitmap bitmap
    ) {
        try {
            outputStream = new FileOutputStream(parFilename);
            save(bitmap);
            outputStream.close();
        } catch (Exception e) {
            Log.e("Exception", e.getMessage());
        }
    }

    public void saveBitmap(
        Bitmap bitmap,
        OutputStream outputStream
    ) {
        this.outputStream = outputStream;
        save(bitmap);
    }

    /*
     *  The saveMethod is the main method of the process. This method
     *  will call the convertImage method to convert the memory image to
     *  a byte array; method writeBitmapFileHeader creates and writes
     *  the bitmap file header; writeBitmapInfoHeader creates the
     *  information header; and writeBitmap writes the image.
     */
    private void save(
        Bitmap bitmap
    ) {
        try {
            convertImage(bitmap);
            writeBitmapFileHeader();
            writeBitmapInfoHeader();
            writeBitmap();
            // write to output stream
            outputStream.write(buffer.array());
        } catch (Exception e) {
            Log.e("Exception", e.getMessage());
        }
    }

    /*
     * convertImage converts the memory image to the bitmap format (BRG).
     * It also computes some information for the bitmap info header.
     */
    private boolean convertImage(
        Bitmap bitmap
    ) {
        int pad;
        int width = bitmap.getWidth();
        int height = bitmap.getHeight();

        pixels = new int[width * height];

        bitmap.getPixels(
            pixels,
            0,
            width,
            0,
            0,
            width,
            height);

        pad = (4 - ((width * 3) % 4)) * height;
        biSizeImage = ((width * height) * 3) + pad;
        bfSize = biSizeImage + BITMAPFILEHEADER_SIZE +
            BITMAPINFOHEADER_SIZE;

        buffer = ByteBuffer.allocate(bfSize);
        biWidth = width;
        biHeight = height;
        return (true);
    }

    /*
     * writeBitmap converts the image returned from the pixel grabber to
     * the format required. Remember: scan lines are inverted in
     * a bitmap file!
     * Each scan line must be padded to an even 4-byte boundary.
     */
    private void writeBitmap() {
        int size;
        int value;
        int j;
        int i;
        int rowCount;
        int rowIndex;
        int lastRowIndex;
        int pad;
        int padCount;
        byte rgb[] = new byte[3];
        size = (biWidth * biHeight) - 1;
        pad = 4 - ((biWidth * 3) % 4);
        if (pad == 4)   // Bug correction
            pad = 0;    // Bug correction
        rowCount = 1;
        padCount = 0;
        rowIndex = size - biWidth;
        lastRowIndex = rowIndex;
        try {
            for (j = 0; j < size; j++) {
                value = pixels[rowIndex];
                rgb[0] = (byte) (value & 0xFF);
                rgb[1] = (byte) ((value >> 8) & 0xFF);
                rgb[2] = (byte) ((value >> 16) & 0xFF);
                buffer.put(rgb);
                if (rowCount == biWidth) {
                    padCount += pad;
                    for (i = 1; i <= pad; i++) {
                        buffer.put((byte) 0x00);
                    }
                    rowCount = 1;
                    rowIndex = lastRowIndex - biWidth;
                    lastRowIndex = rowIndex;
                } else
                    rowCount++;
                rowIndex++;
            }
            // Update the size of the file
            bfSize += padCount - pad;
            biSizeImage += padCount - pad;
        } catch (Exception e) {
            Log.e("Exception", e.getMessage());
        }
    }

    /*
     * writeBitmapFileHeader writes the bitmap file header to the file.
     */
    private void writeBitmapFileHeader() {
        try {
            buffer.put(bfType);
            buffer.put(intToDWord(bfSize));
            buffer.put(intToWord(bfReserved1));
            buffer.put(intToWord(bfReserved2));
            buffer.put(intToDWord(bfOffBits));
        } catch (Exception e) {
            Log.e("Exception", e.getMessage());
        }
    }

    /*
     * writeBitmapInfoHeader writes the bitmap information header
     * to the file.
     */
    private void writeBitmapInfoHeader() {
        try {
            buffer.put(intToDWord(biSize));
            buffer.put(intToDWord(biWidth));
            buffer.put(intToDWord(biHeight));
            buffer.put(intToWord(biPlanes));
            buffer.put(intToWord(biBitCount));
            buffer.put(intToDWord(biCompression));
            buffer.put(intToDWord(biSizeImage));
            buffer.put(intToDWord(biXPelsPerMeter));
            buffer.put(intToDWord(biYPelsPerMeter));
            buffer.put(intToDWord(biClrUsed));
            buffer.put(intToDWord(biClrImportant));
        } catch (Exception e) {
            Log.e("Exception", e.getMessage());
        }
    }

    /*
     * intToWord converts an int to a word, where the return
     * value is stored in a 2-byte array.
     */
    private byte[] intToWord(
        int parValue
    ) {
        byte retValue[] = new byte[2];
        retValue[0] = (byte) (parValue & 0x00FF);
        retValue[1] = (byte) ((parValue >> 8) & 0x00FF);
        return (retValue);
    }

    /*
     * intToDWord converts an int to a double word, where the return
     * value is stored in a 4-byte array.
     */
    private byte[] intToDWord(
        int parValue
    ) {
        byte retValue[] = new byte[4];
        retValue[0] = (byte) (parValue & 0x00FF);
        retValue[1] = (byte) ((parValue >> 8) & 0x000000FF);
        retValue[2] = (byte) ((parValue >> 16) & 0x000000FF);
        retValue[3] = (byte) ((parValue >> 24) & 0x000000FF);
        return (retValue);
    }
}

Обратите внимание, приведенный выше код очень медленный. Я обнаружил, что для оптимизации приведенного выше кода необходимо создать байтовый массив для хранения всех данных до того, как вы их поместите. Конечно, я работал в Xamarin C#, поэтому, может быть, поэтому. В любом случае, вот мой код Xamarin на случай, если у кого-то возникнет такая же проблема.

      public class AndroidBmpUtil
        {

            private static int BMP_WIDTH_OF_TIMES = 4;
            private static int BYTE_PER_PIXEL = 3;

            /**
             * Android Bitmap Object to Window's v3 24bit Bmp Format File
             * @param orgBitmap
             * @param filePath
             * @return file saved result
             */
            public static byte[] ConvertAndroidBitmapByteArray(Bitmap orgBitmap, String filePath)
            {
                if (orgBitmap == null)
                {
                    return null;
                }

                if (filePath == null)
                {
                    return null;
                }

                //image size
                int width = orgBitmap.Width;
                int height = orgBitmap.Height;

                //image dummy data size
                //reason : the amount of bytes per image row must be a multiple of 4 (requirements of bmp format)
                byte[] dummyBytesPerRow = null;
                bool hasDummy = false;
                int rowWidthInBytes = BYTE_PER_PIXEL * width; //source image width * number of bytes to encode one pixel.
                if (rowWidthInBytes % BMP_WIDTH_OF_TIMES > 0)
                {
                    hasDummy = true;
                    //the number of dummy bytes we need to add on each row
                    dummyBytesPerRow = new byte[(BMP_WIDTH_OF_TIMES - (rowWidthInBytes % BMP_WIDTH_OF_TIMES))];
                    //just fill an array with the dummy bytes we need to append at the end of each row
                    for (int i = 0; i < dummyBytesPerRow.Length; i++)
                    {
                        dummyBytesPerRow[i] = (byte)0xFF;
                    }
                }

                //an array to receive the pixels from the source image
                int[] pixels = new int[width * height];

                //the number of bytes used in the file to store raw image data (excluding file headers)
                int imageSize = (rowWidthInBytes + (hasDummy ? dummyBytesPerRow.Length : 0)) * height;
                //file headers size
                int imageDataOffset = 0x36;

                //final size of the file
                int fileSize = imageSize + imageDataOffset;

                //Android Bitmap Image Data
                orgBitmap.GetPixels(pixels, 0, width, 0, 0, width, height);

                //ByteArrayOutputStream baos = new ByteArrayOutputStream(fileSize);
                ByteBuffer buffer = ByteBuffer.Allocate(fileSize);

                /**
                 * BITMAP FILE HEADER Write Start
                 **/
                buffer.Put((sbyte)0x42);
                buffer.Put((sbyte)0x4D);

                //size
                buffer.Put(writeInt(fileSize));

                //reserved
                buffer.Put(writeShort((short)0));
                buffer.Put(writeShort((short)0));

                //image data start offset
                buffer.Put(writeInt(imageDataOffset));

                /** BITMAP FILE HEADER Write End */

                //*******************************************

                /** BITMAP INFO HEADER Write Start */
                //size
                buffer.Put(writeInt(0x28));

                //width, height
                //if we add 3 dummy bytes per row : it means we add a pixel (and the image width is modified.
                buffer.Put(writeInt(width + (hasDummy ? (dummyBytesPerRow.Length == 3 ? 1 : 0) : 0)));
                buffer.Put(writeInt(height));

                //planes
                buffer.Put(writeShort((short)1));

                //bit count
                buffer.Put(writeShort((short)24));

                //bit compression
                buffer.Put(writeInt(0));

                //image data size
                buffer.Put(writeInt(imageSize));

                //horizontal resolution in pixels per meter
                buffer.Put(writeInt(0));

                //vertical resolution in pixels per meter (unreliable)
                buffer.Put(writeInt(0));

                buffer.Put(writeInt(0));

                buffer.Put(writeInt(0));

                /** BITMAP INFO HEADER Write End */
                int row = height;
                int col = width;
                int startPosition = (row - 1) * col;
                int endPosition = row * col;

                // This while loop is a lengthy process
                // Puts take a while so only do one by creating a big array called final
                byte[] final = new byte[0];
                while (row > 0)
                {
                    // This array is also used to cut down on time of puts
                    byte[] b = new byte[(endPosition - startPosition)*3];
                    int counter = 0;
                    for (int i = startPosition; i < endPosition; i++)
                    {

                        b[counter] = (byte)((pixels[i] & 0x000000FF));
                        b[counter + 1] = (byte)((pixels[i] & 0x0000FF00) >> 8);
                        b[counter + 2] = (byte)((pixels[i] & 0x00FF0000) >> 16);
                        counter += 3;
                    }
                    int finalPriorLength = final.Length;
                    Array.Resize<byte>(ref final, finalPriorLength + b.Length);
                    Array.Copy(b, 0, final, finalPriorLength, b.Length);

                    if (hasDummy)
                    {
                        finalPriorLength = final.Length;
                        Array.Resize<byte>(ref final, finalPriorLength + dummyBytesPerRow.Length);
                        Array.Copy(dummyBytesPerRow, 0, final, finalPriorLength, dummyBytesPerRow.Length);
                    }
                    row--;
                    endPosition = startPosition;
                    startPosition = startPosition - col;
                }
                buffer.Put(final);
                buffer.Rewind();

                IntPtr classHandle = JNIEnv.FindClass("java/nio/ByteBuffer");
                IntPtr methodId = JNIEnv.GetMethodID(classHandle, "array", "()[B");
                IntPtr resultHandle = JNIEnv.CallObjectMethod(buffer.Handle, methodId);
                byte[] result = JNIEnv.GetArray<byte>(resultHandle);
                JNIEnv.DeleteLocalRef(resultHandle);

                return result;
            }

            /**
             * Write integer to little-endian
             * @param value
             * @return
             * @throws IOException
             */
            private static byte[] writeInt(int value)
            {
                byte[] b = new byte[4];

                b[0] = (byte)(value & 0x000000FF);
                b[1] = (byte)((value & 0x0000FF00) >> 8);
                b[2] = (byte)((value & 0x00FF0000) >> 16);
                b[3] = (byte)((value & 0xFF000000) >> 24);

                return b;
            }

            /**
             * Write short to little-endian byte array
             * @param value
             * @return
             * @throws IOException
             */
            private static byte[] writeShort(short value)
            {
                byte[] b = new byte[2];

                b[0] = (byte)(value & 0x00FF);
                b[1] = (byte)((value & 0xFF00) >> 8);

                return b;
            }
        }
    }

Чтобы использовать это вот код вызова:

using (FileStream outStream = new    FileStream(@yourFilePath, FileMode.Create))
{
    Bitmap Signature = Bitmap.CreateBitmap(user defined values...); 
    byte[] buffer = AndroidBmpUtil.ConvertAndroidBitmapByteArray(Signature, @yourFilePath); 

    // Actually create the file                                 
    outStream.Write(buffer, 0, buffer.Length);
}
Другие вопросы по тегам