RrunningW```
5148de21创建于 2月1日历史提交
/*
Copyright (c) 2025 WuJingrun(吴京润)

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
 */
package f_io

public class RotatableBuffer {
    public var offset = 0 // offset 是当前缓冲区中未读字节的索引
    private var buffer: (Array<Byte>, Array<Byte>)//缓冲区分两半,读完一半交换两个一半缓冲区
    private var sectionEnd = -1 // 每一个part是一个section,如果当前缓冲区有boundaryBytes,sectionEnd应该是boundaryBytes在buffer中的索引,否则应该是-1
    private var bufferEnd = -1 // 缓冲区结束位置,如果读到的字节数小于bufferSize,bufferEnd < bufferSize,否则bufferEnd == bufferSize
    private let bufferSize: Int64 // hashBufferSize * 2
    public RotatableBuffer(private let input: InputStream, private let boundaryBytes: Array<Byte>, private let halfBufferSize!: Int64 = 4096) {
        buffer = (Array<Byte>(halfBufferSize, repeat: 0), Array<Byte>(halfBufferSize, repeat: 0))
        bufferEnd = input.read(buffer[0])
        if (bufferEnd == halfBufferSize) {
            bufferEnd += input.read(buffer[1])
        }
        bufferSize = halfBufferSize * 2
        sectionEnd = indexOf(boundaryBytes)
    }

    private operator func [](index: Int64): Byte {
        if (index < 0 || index >= bufferSize) {
            throw IndexOutOfBoundsException('index: ${index}; bufferSize: ${bufferSize}')
        } else if (index < halfBufferSize) {
            buffer[0][index]
        } else {
            buffer[1][index - halfBufferSize]
        }
    }
    public func indexOf(bytes: Array<Byte>, from!: Int64 = this.offset): Int64 {
        if (bytes.size == 0) {
            return offset
        }
        let first = bytes[0]
        let max = bufferSize - bytes.size
        var i = from
        while (i <= max) {
            if (this[i] != first) {
                while (let _ <- i++ && i <= max && this[i] != first) {}
            }
            if (i <= max) {
                var j = i + 1
                let end = j + bytes.size - 1
                var k = 1
                while (j < end && this[j] == bytes[k]) {
                    j++
                    k++
                }
                if (j == end) {
                    return i
                }
            }
            i++
        }
        return -1
    }
    public func addOffset(off: Int64) {
        this.offset += off
    }
    public func read(bytes: Array<Byte>): (length: Int64, remainder: Bool, partEnd: Bool) {
        let size = min(bytes.size, if (sectionEnd > 0 && sectionEnd < bufferSize) {
            sectionEnd
        } else {
            bufferSize
        } - offset)
        for (i in 0..size) {
            bytes[i] = this[offset + i]
        }
        offset += size
        let partEnd = offset == sectionEnd
        if (offset == halfBufferSize) {
            buffer = (buffer[1], buffer[0])
            offset = 0
            if (bufferEnd < bufferSize) {
                bufferEnd -= halfBufferSize
                sectionEnd -= halfBufferSize
            } else {
                bufferEnd = halfBufferSize + input.read(buffer[1])
                sectionEnd = indexOf(boundaryBytes)
            }
        } else if (offset > halfBufferSize && offset < bufferSize) {
            buffer = (buffer[1], buffer[0])
            offset -= halfBufferSize
            if (bufferEnd < bufferSize) {
                bufferEnd -= halfBufferSize
                sectionEnd -= halfBufferSize
            } else {
                bufferEnd = halfBufferSize + input.read(buffer[1])
                sectionEnd = indexOf(boundaryBytes)
            }
        } else if (offset == bufferSize) {
            offset = 0
            bufferEnd = input.read(buffer[0])
            if (bufferEnd == halfBufferSize) {
                bufferEnd += input.read(buffer[1])
            }
            sectionEnd = indexOf(boundaryBytes)
        } else if (offset == sectionEnd && sectionEnd + boundaryBytes.size >= halfBufferSize) {
            buffer = (buffer[1], buffer[0])
            offset += boundaryBytes.size - halfBufferSize
            if (bufferEnd < bufferSize) {
                bufferEnd -= halfBufferSize
                sectionEnd -= halfBufferSize
            } else {
                bufferEnd = halfBufferSize + input.read(buffer[1])
                sectionEnd = indexOf(boundaryBytes)
            }
        } else if (offset == sectionEnd){
            offset = sectionEnd + boundaryBytes.size
            sectionEnd = indexOf(boundaryBytes)
        }
        (size, bufferEnd > halfBufferSize, partEnd)
    }
}