这个把source缓冲区中的数据写到当前缓冲区的方法是比较经典的:

   if (source == null) throw new IllegalArgumentException("source == null");
//这个判断是防止自己把数据写到自身这个Buffer中
if (source == this) throw new IllegalArgumentException("source == this");
//每个Segment中最大存储的自己是8k,因此写的时候需要进行校验
checkOffsetAndCount(source.size, 0, byteCount); //核心设计来了
每次读取source中的数据时,都是从head开始,如果当前的剩余的byteCount小于head中的有效数据
判断当前buffer中的尾Segment有没有操作权限并且可用空间能否放得下当前剩余的byteCount,如果可以放的下直接
调用source.head.writeTo(tail, (int) byteCount)方法把source中的数据写进tail中;结束
否则就把head分割成两部分。 此时source的head就是分割的后一个Segment,而前一个Segment则给到了当前的Segment了,成为了他的尾Segment。 while (byteCount > 0) {
// 说明当前剩余的长度包含在了source的头中,也意味着循环的结束
if (byteCount < (source.head.limit - source.head.pos)) {
//找到当前Segment的尾部
Segment tail = head != null ? head.prev : null;
//权限判断和可用空间的判断是否都符合写入条件
if (tail != null && tail.owner
&& (byteCount + tail.limit - (tail.shared ? 0 : tail.pos) <= Segment.SIZE)) {
//如果符合要求,那就直接把source中byteCount长度的字节移动到tail中
source.head.writeTo(tail, (int) byteCount);
//更新source的长度
source.size -= byteCount;
//更新当前缓冲区的长度
size += byteCount;
//结束
return;
} else { //不符合上述条件的,我没就通过split把head分割成2部分,前一部分就是我们需要写入的数据,此时也是source的头Segment
source.head = source.head.split((int) byteCount);
}
} //需要移动的就是source的头
Segment segmentToMove = source.head;
//计算移动的长度,更新会用到
long movedByteCount = segmentToMove.limit - segmentToMove.pos;
//通过pop方法位source指定新的head
source.head = segmentToMove.pop();
//如果当前的buffer时空的,那么segmentToMove直接作为buffer的头即可
if (head == null) {
head = segmentToMove;
head.next = head.prev = head;
} else {
//否则就把segmentToMove插入到尾部
Segment tail = head.prev;
tail = tail.push(segmentToMove);
//这里还进行了一次压缩:里面也会进行write操作
tail.compact();
}
//更新source的长度
source.size -= movedByteCount;
//更新当前缓冲区的长度
size += movedByteCount;
//更新剩余需要写入的长度
byteCount -= movedByteCount;
}

Buffer中的public void write(Buffer source, long byteCount)解析的更多相关文章

  1. Android开发之Source无法覆写public void onClick(View v)

    初学Android开发,在为一个按钮[该按钮继承OnClickListener()]写监听时,发现无法在Source中引入public void onClick(View v),当时非常纳闷,平常情况 ...

  2. 一个I/O线程可以并发处理N个客户端连接和读写操作 I/O复用模型 基于Buf操作NIO可以读取任意位置的数据 Channel中读取数据到Buffer中或将数据 Buffer 中写入到 Channel 事件驱动消息通知观察者模式

    Tomcat那些事儿 https://mp.weixin.qq.com/s?__biz=MzI3MTEwODc5Ng==&mid=2650860016&idx=2&sn=549 ...

  3. Java NIO Buffer中各种状态属性的含义

    关于NIO Buffer中的3个重要状态属性的含义: postion, limit与capacity. public class NioTest { public static void main(S ...

  4. java.nio.Buffer 中的 flip()方法

    在Java NIO编程中,对缓冲区操作常常需要使用  java.nio.Buffer中的 flip()方法. Buffer 中的 flip() 方法涉及到 Buffer 中的capacity.posi ...

  5. 从depth buffer中构建view-space position

    观察透视投影矩阵: 对于x和y,矩阵变换只是一个缩放系数,那么逆变换就是缩放系数的倒数,所以 设Xndc Yndc为NDC空间中的XY坐标,Xview Yview Zview为view space中的 ...

  6. 在java中public void与public static void有什么区别 ?

    public void 修饰是非静态方法,该类方法属于对象,在对象初始化(new Object())后才能被调用:public static void 修饰是静态方法,属于类,使用类名.方法名直接调用 ...

  7. linux中内存使用,swap,cache,buffer的含义总结

    首先介绍一下linux中内存是如何使用的.当有应用需要读写磁盘数据时,由系统把相关数据从磁盘读取到内存,如果物理内存不够,则把内存中的部分数据导入到磁盘,从而把磁盘的部分空间当作虚拟内存来使用,也称为 ...

  8. 在java中public void与public static void区别

    static 方法可以被main方法直接调用,而非static方法不可以.因为static方法是属于类的,是类方法.可以通过类名.方法名直接调用.而非static方法必须等对象被new出来以后才能使用 ...

  9. Linux系统中的Page cache和Buffer cache

    Linux系统中的Page cache和Buffer cache Linux中有两个很容易混淆的概念,pagecache和buffercache,首先简单将一些Linux系统下内存的分布,使用free ...

  10. (1)定义一个接口CanFly,描述会飞的方法public void fly(); (2)分别定义类飞机和鸟,实现CanFly接口。 (3)定义一个测试类,测试飞机和鸟,在main方法中创建飞机对象和鸟对象, 再定义一个makeFly()方法,其中让会飞的事物飞。并在main方法中调用该方法, 让飞机和鸟起飞。

    package b; public interface CanFly { public void fly(); } package b; public class FeiJi implements C ...

随机推荐

  1. Restful Fast Request 添加前置脚本,实现不同环境免设置token 直接请求

    idea安装Restful Fast Request插件后,进行如下设置,并打开 项目全局参数 对话框 进入前置脚本 tab 编写如下groovy脚本代码(插件脚本语言默认支持groovy,该语言被称 ...

  2. swoft-个基于 Swoole 原生协程的PHP 微服务框架

    刚才百度了一下swoft框架,官网打不开了,仓库也暂停了.不由感慨.曾经和同事踩了许多坑使用此极其小众的框架完成微服务项目.使用它的唯一目的就是提高程序性能(底层使用了协程),为此大家都学习了很多新知 ...

  3. python——numpy

    NumPy(Numerical Python) 是 Python 语言的一个扩展程序库,支持大量的维度数组与矩阵运算,此外也针对数组运算提供大量的数学函数库. import numpy a = num ...

  4. css 多行文本展开收起

    <template> <div class="content"> <div :class="[isOpen ? 'text' : 'text ...

  5. 导入maven包时,非常慢的解决方案

    在导入maven包时,很多时候导的很慢,导致找不到包,需要将maven包下载地址替换为aliyun的地址,以下为解决方案 找到使用的maven地址,在文件内添加以下节点然后重启ide即可 <mi ...

  6. 『教程』mariadb的主从复制

    一.MariaDB简介 MariaDB数据库的主从复制方案,是其自带的功能,并且主从复制并不是复制磁盘上的数据库文件,而是通过binlog日志复制到需要同步的从服务器上. MariaDB数据库支持单向 ...

  7. MySQL -my.cnf配置文件优化

    # [mysqld] datadir=/var/lib/mysql #socket=/var/lib/mysql/mysql.sock user=mysql ### 设置主从的时候的唯一ID 每台主机 ...

  8. mininet配置命令

    Mininet实验手册 一.安装 1.  直接使用带有完整软件的VM(略) 2.  源码安装 1)  下载 git clone git://github.com/mininet/mininet 2)  ...

  9. 织梦(DedeCMS)安全方案

    一.织梦(DedeCMS)介绍 织梦(DedeCMS)作为国内龙头CMS系统之一,因其系统结构科学合理,操作使用简单,拥有海量用户.和其他CMS一样,安全漏洞也是其无法避免的问题.虽然官方不断发布补丁 ...

  10. 使用vCenter对ESXi主机进行补丁升级

    使用vCenter 对ESXi 主机进行补丁升级 背景说明:公司内部有许多ESXi主机需要进行补丁升级,记录一下通过vCenter对ESXi主机进行补丁升级的过程,也可以使用esxcli命令行方式: ...