|
@@ -0,0 +1,216 @@
|
|
|
+package com.gxzc.zen.common.util.io
|
|
|
+
|
|
|
+import java.io.File
|
|
|
+import java.io.IOException
|
|
|
+import java.io.RandomAccessFile
|
|
|
+
|
|
|
+/**
|
|
|
+ * 加入缓冲机制的 <br>
|
|
|
+ * 随机文件访问 <br>
|
|
|
+ * 效率速度为 RandomAccessFile 的 5倍左右,大文件更高
|
|
|
+ * 占用空间 <= 30MB (4096)
|
|
|
+ * @author NorthLan
|
|
|
+ * @date 2018/5/24
|
|
|
+ * @url https://noahlan.com
|
|
|
+ */
|
|
|
+open class BufferedRandomAccessFile(name: String, mode: String = "r", bufbitlen: Int = DEFAULT_BUFFER_BIT_LEN) : RandomAccessFile(name, mode) {
|
|
|
+ companion object {
|
|
|
+ private val DEFAULT_BUFFER_BIT_LEN = 10
|
|
|
+ private val DEFAULT_BUFFER_SIZE = 1 shl DEFAULT_BUFFER_BIT_LEN
|
|
|
+ }
|
|
|
+
|
|
|
+ private var buf: ByteArray
|
|
|
+ private var bufbitlen: Int = 0
|
|
|
+ private var bufsize: Int = 0
|
|
|
+ private var bufmask: Long = 0
|
|
|
+ private var bufdirty: Boolean = false
|
|
|
+ private var bufusedsize: Int = 0
|
|
|
+ private var curpos: Long = 0
|
|
|
+
|
|
|
+ private var bufstartpos: Long = 0
|
|
|
+ private var bufendpos: Long = 0
|
|
|
+ private var fileendpos: Long = 0
|
|
|
+
|
|
|
+ private var append: Boolean = false
|
|
|
+ var filename: String
|
|
|
+ var initfilelen: Long = 0
|
|
|
+
|
|
|
+ constructor(file: File) : this(file.path, "r", DEFAULT_BUFFER_BIT_LEN)
|
|
|
+ constructor(name: String, bufbitlen: Int) : this(name, "r", bufbitlen)
|
|
|
+ constructor(file: File, bufbitlen: Int) : this(file.path, "r", bufbitlen)
|
|
|
+ constructor(file: File, mode: String) : this(file.path, mode, DEFAULT_BUFFER_BIT_LEN)
|
|
|
+ constructor(file: File, mode: String, bufbitlen: Int) : this(file.path, mode, bufbitlen)
|
|
|
+
|
|
|
+ init {
|
|
|
+ this.append = mode == "r" != true
|
|
|
+
|
|
|
+ this.filename = name
|
|
|
+ this.initfilelen = super.length()
|
|
|
+ this.fileendpos = this.initfilelen - 1
|
|
|
+ this.curpos = super.getFilePointer()
|
|
|
+
|
|
|
+ if (bufbitlen < 0) {
|
|
|
+ throw IllegalArgumentException("bufbitlen size must >= 0")
|
|
|
+ }
|
|
|
+
|
|
|
+ this.bufbitlen = bufbitlen
|
|
|
+ this.bufsize = 1 shl bufbitlen
|
|
|
+ this.buf = ByteArray(this.bufsize)
|
|
|
+ this.bufmask = (this.bufsize.toLong() - 1L).inv()
|
|
|
+ this.bufdirty = false
|
|
|
+ this.bufusedsize = 0
|
|
|
+ this.bufstartpos = -1
|
|
|
+ this.bufendpos = -1
|
|
|
+ }
|
|
|
+
|
|
|
+ private fun flushbuf() {
|
|
|
+ if (this.bufdirty) {
|
|
|
+ if (super.getFilePointer() != this.bufstartpos) {
|
|
|
+ super.seek(this.bufstartpos)
|
|
|
+ }
|
|
|
+ super.write(this.buf, 0, this.bufusedsize)
|
|
|
+ this.bufdirty = false
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ private fun fillbuf(): Int {
|
|
|
+ super.seek(this.bufstartpos)
|
|
|
+ this.bufdirty = false
|
|
|
+ return super.read(this.buf)
|
|
|
+ }
|
|
|
+
|
|
|
+ fun read(pos: Long): Byte {
|
|
|
+ if (pos < this.bufstartpos || pos > this.bufendpos) {
|
|
|
+ this.flushbuf()
|
|
|
+ this.seek(pos)
|
|
|
+
|
|
|
+ if (pos < this.bufstartpos || pos > this.bufendpos) {
|
|
|
+ throw IOException()
|
|
|
+ }
|
|
|
+ }
|
|
|
+ this.curpos = pos
|
|
|
+ return this.buf[(pos - this.bufstartpos).toInt()]
|
|
|
+ }
|
|
|
+
|
|
|
+ fun write(bw: Byte): Boolean {
|
|
|
+ return this.write(bw, this.curpos)
|
|
|
+ }
|
|
|
+
|
|
|
+ fun append(bw: Byte): Boolean {
|
|
|
+ return this.write(bw, this.fileendpos + 1)
|
|
|
+ }
|
|
|
+
|
|
|
+ fun write(bw: Byte, pos: Long): Boolean {
|
|
|
+ if (pos >= this.bufstartpos && pos <= this.bufendpos) { // write pos in buf
|
|
|
+ this.buf[(pos - this.bufstartpos).toInt()] = bw
|
|
|
+ this.bufdirty = true
|
|
|
+
|
|
|
+ if (pos == this.fileendpos + 1) { // write pos is append pos
|
|
|
+ this.fileendpos++
|
|
|
+ this.bufusedsize++
|
|
|
+ }
|
|
|
+ } else { // write pos not in buf
|
|
|
+ this.seek(pos)
|
|
|
+
|
|
|
+ if (pos >= 0 && pos <= this.fileendpos && this.fileendpos != 0L) { // write pos is modify file
|
|
|
+ this.buf[(pos - this.bufstartpos).toInt()] = bw
|
|
|
+
|
|
|
+ } else if (pos == 0L && this.fileendpos == 0L || pos == this.fileendpos + 1) { // write pos is append pos
|
|
|
+ this.buf[0] = bw
|
|
|
+ this.fileendpos++
|
|
|
+ this.bufusedsize = 1
|
|
|
+ } else {
|
|
|
+ throw IndexOutOfBoundsException()
|
|
|
+ }
|
|
|
+ this.bufdirty = true
|
|
|
+ }
|
|
|
+ this.curpos = pos
|
|
|
+ return true
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun write(b: ByteArray, off: Int, len: Int) {
|
|
|
+ val writeendpos = this.curpos + len - 1
|
|
|
+
|
|
|
+ if (writeendpos <= this.bufendpos) { // b[] in cur buf
|
|
|
+ System.arraycopy(b, off, this.buf, (this.curpos - this.bufstartpos).toInt(), len)
|
|
|
+ this.bufdirty = true
|
|
|
+ this.bufusedsize = (writeendpos - this.bufstartpos + 1).toInt() //(int)(this.curpos - this.bufstartpos + len - 1);
|
|
|
+
|
|
|
+ } else { // b[] not in cur buf
|
|
|
+ super.seek(this.curpos)
|
|
|
+ super.write(b, off, len)
|
|
|
+ }
|
|
|
+
|
|
|
+ if (writeendpos > this.fileendpos)
|
|
|
+ this.fileendpos = writeendpos
|
|
|
+
|
|
|
+ this.seek(writeendpos + 1)
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun read(b: ByteArray, off: Int, len: Int): Int {
|
|
|
+ var ret = len
|
|
|
+ var readendpos = this.curpos + ret - 1
|
|
|
+ if (readendpos <= this.bufendpos && readendpos <= this.fileendpos) { // read in buf
|
|
|
+ System.arraycopy(this.buf, (this.curpos - this.bufstartpos).toInt(), b, off, len)
|
|
|
+ } else { // read b[] size > buf[]
|
|
|
+ if (readendpos > this.fileendpos) { // read b[] part in file
|
|
|
+ ret = (this.length() - this.curpos + 1).toInt()
|
|
|
+ }
|
|
|
+ super.seek(this.curpos)
|
|
|
+ ret = super.read(b, off, ret)
|
|
|
+ readendpos = this.curpos + ret - 1
|
|
|
+ }
|
|
|
+ this.seek(readendpos + 1)
|
|
|
+ return ret
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun write(b: ByteArray) {
|
|
|
+ this.write(b, 0, b.size)
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun read(b: ByteArray): Int {
|
|
|
+ return this.read(b, 0, b.size)
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun seek(pos: Long) {
|
|
|
+ if (pos < this.bufstartpos || pos > this.bufendpos) { // seek pos not in buf
|
|
|
+ this.flushbuf()
|
|
|
+
|
|
|
+ if (pos >= 0 && pos <= this.fileendpos && this.fileendpos != 0L) { // seek pos in file (file length > 0)
|
|
|
+ this.bufstartpos = pos and this.bufmask
|
|
|
+ this.bufusedsize = this.fillbuf()
|
|
|
+ } else if (pos == 0L && this.fileendpos == 0L || pos == this.fileendpos + 1) { // seek pos is append pos
|
|
|
+ this.bufstartpos = pos
|
|
|
+ this.bufusedsize = 0
|
|
|
+ }
|
|
|
+ this.bufendpos = this.bufstartpos + this.bufsize - 1
|
|
|
+ }
|
|
|
+ this.curpos = pos
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun length(): Long {
|
|
|
+ return this.max(this.fileendpos + 1, this.initfilelen)
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun setLength(newLength: Long) {
|
|
|
+ if (newLength > 0) {
|
|
|
+ this.fileendpos = newLength - 1
|
|
|
+ } else {
|
|
|
+ this.fileendpos = 0
|
|
|
+ }
|
|
|
+ super.setLength(newLength)
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun getFilePointer(): Long {
|
|
|
+ return this.curpos
|
|
|
+ }
|
|
|
+
|
|
|
+ private fun max(a: Long, b: Long): Long {
|
|
|
+ return if (a > b) a else b
|
|
|
+ }
|
|
|
+
|
|
|
+ override fun close() {
|
|
|
+ this.flushbuf()
|
|
|
+ super.close()
|
|
|
+ }
|
|
|
+}
|