美文网首页
java.nio.Buffer 类详解

java.nio.Buffer 类详解

作者: 孙广明 | 来源:发表于2018-12-26 10:35 被阅读0次

    1. 属性


    属性
    • capacity(容量)
    • limit (限制)
    • position (位置)
    • mark (标记)

    mark <= position <= limit <= capacity

    源码
     // Invariants: mark <= position <= limit <= capacity
        private int mark = -1;
        private int position = 0;
        private int limit;
        private int capacity
    

    说明 :

    1. capacity 代表包含元素的数量,即容量
    2. limit 代表缓冲区中的限制第一个不应该读取或者写入元素的index
    3. position 代表下一个要读取或者要写入元素的index
    4. mark 设置标记,标记当前position的位置,配合reset() 方法重置上次标记的position位置。

    2.非抽象方法


    构造函数
        // Creates a new buffer with the given mark, position, limit, and capacity,
        // after checking invariants.
        //
        Buffer(int mark, int pos, int lim, int cap) {       // package-private
            if (cap < 0)
                throw new IllegalArgumentException("Negative capacity: " + cap);
            this.capacity = cap;
            limit(lim);
            position(pos);
            if (mark >= 0) {
                if (mark > pos)
                    throw new IllegalArgumentException("mark > position: ("
                                                       + mark + " > " + pos + ")");
                this.mark = mark;
            }
        }
    
    • 创建出一个带有mark, position, limit, and capacity属性的Buffer,该方法包内可见。
    • 判断capacity 属性是否小于0,若小于 0 则抛出非法参数异常。
    获取容量(capacity)
        /**
         * Returns this buffer's capacity.
         *
         * @return  The capacity of this buffer
         */
        public final int capacity() {
            return capacity;
        }
    
    • 返回 capacity 值。
    获取位置(position)
        /**
         * Returns this buffer's position.
         *
         * @return  The position of this buffer
         */
        public final int position() {
            return position;
        }
    
    • 返回 position 值。
    设置位置(position)
        /**
         * Sets this buffer's position.  If the mark is defined and larger than the
         * new position then it is discarded.
         *
         * @param  newPosition
         *         The new position value; must be non-negative
         *         and no larger than the current limit
         *
         * @return  This buffer
         *
         * @throws  IllegalArgumentException
         *          If the preconditions on <tt>newPosition</tt> do not hold
         */
        public final Buffer position(int newPosition) {
            if ((newPosition > limit) || (newPosition < 0))
                throw new IllegalArgumentException();
            position = newPosition;
            if (mark > position) mark = -1;
            return this;
        }
    
    • 设置 position 值。
    • 传入的新 position 值若大于limit(限制) 或小于 0,抛出非法参数异常。
    • 若 mark (标记) 大于赋值后的 position 值,则弃用标志(mark = -1)。
    获取限制(limit)
        /**
         * Returns this buffer's limit.
         *
         * @return  The limit of this buffer
         */
        public final int limit() {
            return limit;
        }
    
    • 返回 limit 值。
    设置限制(limit)
        /**
         * Sets this buffer's limit.  If the position is larger than the new limit
         * then it is set to the new limit.  If the mark is defined and larger than
         * the new limit then it is discarded.
         *
         * @param  newLimit
         *         The new limit value; must be non-negative
         *         and no larger than this buffer's capacity
         *
         * @return  This buffer
         *
         * @throws  IllegalArgumentException
         *          If the preconditions on <tt>newLimit</tt> do not hold
         */
        public final Buffer limit(int newLimit) {
            if ((newLimit > capacity) || (newLimit < 0))
                throw new IllegalArgumentException();
            limit = newLimit;
            if (position > limit) position = limit;
            if (mark > limit) mark = -1;
            return this;
        }
    
    • 设置 limit 值。
    • 传入的新 limit 值若大于 capacity 或者 小于 0 ,则抛出非法参数异常
    • 若 position 大于 limit,则 position 值设置为 limit的值
    • 若 mark 大于 limit,则废除 mark
    设置标记(mark)
         /**
         * Sets this buffer's mark at its position.
         *
         * @return  This buffer
         */
        public final Buffer mark() {
            mark = position;
            return this;
        }
    
    • 设置标记的值为 position,相当于记录position的值,在使用reset() 方法时重置 position值。
    reset() 重置方法
        /**
         * Resets this buffer's position to the previously-marked position.
         *
         * <p> Invoking this method neither changes nor discards the mark's
         * value. </p>
         *
         * @return  This buffer
         *
         * @throws  InvalidMarkException
         *          If the mark has not been set
         */
        public final Buffer reset() {
            int m = mark;
            if (m < 0)
                throw new InvalidMarkException();
            position = m;
            return this;
        }
    
    • 此方法只有在标记有效的情况下才能使用,即 mrak 不小于 0
    • 重置 position 的值,即将 mark 的值赋值给 position
    clear() 清除方法
        /**
         * Clears this buffer.  The position is set to zero, the limit is set to
         * the capacity, and the mark is discarded.
         *
         * <p> Invoke this method before using a sequence of channel-read or
         * <i>put</i> operations to fill this buffer.  For example:
         *
         * <blockquote><pre>
         * buf.clear();     // Prepare buffer for reading
         * in.read(buf);    // Read data</pre></blockquote>
         *
         * <p> This method does not actually erase the data in the buffer, but it
         * is named as if it did because it will most often be used in situations
         * in which that might as well be the case. </p>
         *
         * @return  This buffer
         */
        public final Buffer clear() {
            position = 0;
            limit = capacity;
            mark = -1;
            return this;
        }
    
    • 清除Buffer的属性值,即 position = 0,limit = capacity,mark = -1
    flip() 翻转方法
       /**
         * Flips this buffer.  The limit is set to the current position and then
         * the position is set to zero.  If the mark is defined then it is
         * discarded.
         *
         * <p> After a sequence of channel-read or <i>put</i> operations, invoke
         * this method to prepare for a sequence of channel-write or relative
         * <i>get</i> operations.  For example:
         *
         * <blockquote><pre>
         * buf.put(magic);    // Prepend header
         * in.read(buf);      // Read data into rest of buffer
         * buf.flip();        // Flip buffer
         * out.write(buf);    // Write header + data to channel</pre></blockquote>
         *
         * <p> This method is often used in conjunction with the {@link
         * java.nio.ByteBuffer#compact compact} method when transferring data from
         * one place to another.  </p>
         *
         * @return  This buffer
         */
        public final Buffer flip() {
            limit = position;
            position = 0;
            mark = -1;
            return this;
        }
    
    • 反转Buffer,反向操作Buffer。
    • 将 position 值赋值给 limit,在设置 position 值为 0 ,废除标记。
    rewind() 回退方法
        /**
         * Rewinds this buffer.  The position is set to zero and the mark is
         * discarded.
         *
         * <p> Invoke this method before a sequence of channel-write or <i>get</i>
         * operations, assuming that the limit has already been set
         * appropriately.  For example:
         *
         * <blockquote><pre>
         * out.write(buf);    // Write remaining data
         * buf.rewind();      // Rewind buffer
         * buf.get(array);    // Copy data into array</pre></blockquote>
         *
         * @return  This buffer
         */
        public final Buffer rewind() {
            position = 0;
            mark = -1;
            return this;
        }
    
    • 回退Buffer,重新操作Buffer。
    • 设置 position 值为 0 ,废除标记。
    remaining() 剩余
        /**
         * Returns the number of elements between the current position and the
         * limit.
         *
         * @return  The number of elements remaining in this buffer
         */
        public final int remaining() {
            return limit - position;
        }
    
    • 返回剩余元素个数:limit - position
    hasRemaining() 是否有剩余元素
        /**
         * Tells whether there are any elements between the current position and
         * the limit.
         *
         * @return  <tt>true</tt> if, and only if, there is at least one element
         *          remaining in this buffer
         */
        public final boolean hasRemaining() {
            return position < limit;
        }
    
    • 判断Buffer是否有剩余元素

    3.抽象方法


    isReadOnly() 是否只读
        /**
         * Tells whether or not this buffer is read-only.
         *
         * @return  <tt>true</tt> if, and only if, this buffer is read-only
         */
        public abstract boolean isReadOnly();
    
    hasArray() 是否包含数组
        /**
         * Tells whether or not this buffer is read-only.
         *
         * @return  <tt>true</tt> if, and only if, this buffer is read-only
         */
        public abstract boolean isReadOnly();
    
    array() 获取数组
        /**
         * Returns the array that backs this
         * buffer&nbsp;&nbsp;<i>(optional operation)</i>.
         *
         * <p> This method is intended to allow array-backed buffers to be
         * passed to native code more efficiently. Concrete subclasses
         * provide more strongly-typed return values for this method.
         *
         * <p> Modifications to this buffer's content will cause the returned
         * array's content to be modified, and vice versa.
         *
         * <p> Invoke the {@link #hasArray hasArray} method before invoking this
         * method in order to ensure that this buffer has an accessible backing
         * array.  </p>
         *
         * @return  The array that backs this buffer
         *
         * @throws  ReadOnlyBufferException
         *          If this buffer is backed by an array but is read-only
         *
         * @throws  UnsupportedOperationException
         *          If this buffer is not backed by an accessible array
         *
         * @since 1.6
         */
        public abstract Object array();
    
    arrayOffset() 获取数组偏移量
        /**
         * Returns the offset within this buffer's backing array of the first
         * element of the buffer&nbsp;&nbsp;<i>(optional operation)</i>.
         *
         * <p> If this buffer is backed by an array then buffer position <i>p</i>
         * corresponds to array index <i>p</i>&nbsp;+&nbsp;<tt>arrayOffset()</tt>.
         *
         * <p> Invoke the {@link #hasArray hasArray} method before invoking this
         * method in order to ensure that this buffer has an accessible backing
         * array.  </p>
         *
         * @return  The offset within this buffer's array
         *          of the first element of the buffer
         *
         * @throws  ReadOnlyBufferException
         *          If this buffer is backed by an array but is read-only
         *
         * @throws  UnsupportedOperationException
         *          If this buffer is not backed by an accessible array
         *
         * @since 1.6
         */
        public abstract int arrayOffset();
    
    isDirect() 是否直接缓冲区
        /**
         * Tells whether or not this buffer is
         * <a href="ByteBuffer.html#direct"><i>direct</i></a>.
         *
         * @return  <tt>true</tt> if, and only if, this buffer is direct
         *
         * @since 1.6
         */
        public abstract boolean isDirect();
    

    推荐书籍:
    • [高洪岩]《Java多线程编程核心技术》
    • [高洪岩]《Java并发编程:核心方法与框架》
    • [高洪岩]《NIO与Socket编程技术指南》

    相关文章

      网友评论

          本文标题:java.nio.Buffer 类详解

          本文链接:https://www.haomeiwen.com/subject/lzvflqtx.html