ByteArrayOutputStream.avt

Переключить прокрутку окна
Загрузить этот исходный код

/*
    Исходный код среды исполнения ПВТ-ОО.

    Этот исходный код является частью проекта ПВТ-ОО.

    Copyright © 2021 Малик Разработчик

    Это свободная программа: вы можете перераспространять её и/или
    изменять её на условиях Меньшей Стандартной общественной лицензии GNU в том виде,
    в каком она была опубликована Фондом свободного программного обеспечения;
    либо версии 3 лицензии, либо (по вашему выбору) любой более поздней версии.

    Эта программа распространяется в надежде, что она может быть полезна,
    но БЕЗО ВСЯКИХ ГАРАНТИЙ; даже без неявной гарантии ТОВАРНОГО ВИДА
    или ПРИГОДНОСТИ ДЛЯ ОПРЕДЕЛЁННЫХ ЦЕЛЕЙ. Подробнее см. в Меньшей Стандартной
    общественной лицензии GNU.

    Вы должны были получить копию Меньшей Стандартной общественной лицензии GNU
    вместе с этой программой. Если это не так, см.
    <http://www.gnu.org/licenses/>.
*/

package avt.io;

import avt.io.extension.*;

public class ByteArrayOutputStream(Object, Closeable, Extension, MarkExtension, ByteWriter, TruncateExtension, LimitedSizeExtension, SeekExtension, Extendable)
{
    package int fldLength;
    package int fldPosition;
    package byte[] fldArray;
    package final ByteArrayInputStream fldSource;
    private int fldMarkedPosition;

    public () {
        fldArray = new byte[127];
    }

    public (byte[] array) {
        int slen = fldLength = fldPosition = array == null ? 0 : array.length;
        Array.copy(array, 0, fldArray = new byte[slen >= Int.MAX_VALUE - 127 ? Int.MAX_VALUE : slen + 127], 0, slen);
    }

    public (byte[] array, int length) {
        int alen = array == null ? 0 : array.length;
        int slen = fldLength = fldPosition = length < 0 ? 0 : length > alen ? alen : length;
        Array.copy(array, 0, fldArray = new byte[slen >= Int.MAX_VALUE - 127 ? Int.MAX_VALUE : slen + 127], 0, slen);
    }

    public (byte[] array, int length, int position) {
        int alen = array == null ? 0 : array.length;
        int slen = fldLength = length < 0 ? 0 : length > alen ? alen : length;
        fldPosition = position < 0 ? 0 : position > slen ? slen : position;
        Array.copy(array, 0, fldArray = new byte[slen >= Int.MAX_VALUE - 127 ? Int.MAX_VALUE : slen + 127], 0, slen);
    }

    protected (ByteArrayInputStream source, int position) {
        if(source == null)
        {
            fldArray = new byte[127];
            return;
        }
        int slen = fldLength = source.fldLength;
        fldPosition = position < 0 ? 0 : position > slen ? slen : position;
        fldArray = source.fldArray;
        fldSource = source;
    }

    public String toString() { return new String(fldArray, 0, fldLength); }

    public void close() {  }

    public void reset() { fldPosition = fldMarkedPosition; }

    public void mark(int readLimit) { fldMarkedPosition = fldPosition; }

    public void write(int byteData) {
        int curlen = fldLength;
        int curpos = fldPosition;
        int newpos = curpos + 1;
        byte[] array = newpos >= 0 && newpos <= curlen ? fldArray : madeLarger(curlen, newpos);
        array[curpos] = (byte) byteData;
        fldPosition = newpos;
    }

    public void write(byte[] src) { write(src, 0, src == null ? 0 : src.length); }

    public void write(byte[] src, int offset, int length) {
        if(src == null)
        {
            throw new NullPointerException("аргумент src равен нулевой ссылке");
        }
        Array.checkBounds(src, offset, length);
        if(length <= 0) return;
        int curlen = fldLength;
        int curpos = fldPosition;
        int newpos = curpos + length;
        byte[] array = newpos >= 0 && newpos <= curlen ? fldArray : madeLarger(curlen, newpos);
        Array.copy(src, offset, array, curpos, length);
        fldPosition = newpos;
    }

    public void flush() {  }

    public void truncate() {
        int length = fldLength = fldPosition;
        if(fldMarkedPosition > length) fldMarkedPosition = length;
        ByteArrayInputStream source = fldSource;
        if(source != null)
        {
            source.fldLength = length;
            source.notifyLengthChanged(length);
        }
    }

    public long available() { return (long) (fldLength - fldPosition); }

    public long position() { return (long) fldPosition; }

    public long size() { return (long) fldLength; }

    public long seek(long offset, int from) {
        long pos;
        long length = (long) fldLength;
        switch(from)
        {
        case BEGIN:
            pos = offset;
            break;
        case END:
            pos = offset + length;
            break;
        case CURRENT:
            pos = offset + (long) fldPosition;
            break;
        default:
            throw new IllegalArgumentException("аргумент from имеет недопустимое значение");
        }
        if(pos < 0L) pos = 0L;
        if(pos > length) pos = length;
        return (long) (fldPosition = (int) pos);
    }

    public Extension[] getExtensions() { return new Extension[] { this }; }

    public byte[] toByteArray() {
        int length = fldLength;
        byte[] result = new byte[length];
        Array.copy(fldArray, 0, result, 0, length);
        return result;
    }

    package final void notifyLengthChanged(int length) {
        if(fldPosition > length) fldPosition = length;
        if(fldMarkedPosition > length) fldMarkedPosition = length;
    }

    private byte[] madeLarger(int curLength, int newLength) {
        if(newLength < 0)
        {
            throw new BufferTooLargeError("объём буфера очень велик");
        }
        ByteArrayInputStream source = fldSource;
        byte[] array = fldArray;
        if(array == null)
        {
            fldArray = array = new byte[newLength >= Int.MAX_VALUE - 127 ? Int.MAX_VALUE : newLength + 127];
            if(source != null) source.fldArray = array;
        } else
        {
            int curCapacity = array.length;
            if(newLength > curCapacity)
            {
                int newCapacity = (curCapacity << 1) + 1;
                if(newCapacity < 0) newCapacity = Int.MAX_VALUE;
                if(newCapacity < newLength) newCapacity = newLength >= Int.MAX_VALUE - 127 ? Int.MAX_VALUE : newLength + 127;
                Array.copy(array, 0, fldArray = array = new byte[newCapacity], 0, curLength);
                if(source != null) source.fldArray = array;
            }
        }
        fldLength = newLength;
        if(source != null) source.fldLength = newLength;
        return array;
    }
}