也许我会以错误的方式解决这个问题,但我在 ByteBuffer上缺少一些绝对的put方法.

如果你看看ByteBuffer,你会发现大多数put方法都有绝对变量和相对变量.

除了:

>将字节数组的部分写入ByteBuffer.
>将ByteBuffer写入ByteBuffer.

..而且我需要那些.

要明确ByteBuffer有方法:

put(byte[] src,int offset,int length)
 put(ByteBuffer src)

但缺乏:

put(int index,byte[] src,int length)
 put(int index,ByteBuffer src)

我有理由为什么我不想移动缓冲区的位置指针,因此我只想使用绝对put方法.

知道为什么这些方法被遗漏了吗?

我当然可以在不移动缓冲区位置指针的情况下模仿丢失的方法,但这将涉及在源字节上循环. Javadoc明确指出,这些方法(可能)比移动更有效,而不是循环并逐个移动字节.我相信Javadoc,因为我的测试表明相同.我需要从实施中尽可能多地挤出速度,因此我当然倾向于利用我可以获得的任何批量方法……如果它们存在的话.

即使ByteBuffer也缺少部分字节数组移动的绝对get方法.但我目前实际上并不需要这样的方法.但又奇怪的是它不存在.

解决方法

获得所需方法的一种方法是让第二个ByteBuffer共享相同的内存,这样您就可以在不改变原始位置的情况下更改其位置.

不幸的是,slice方法也没有采用位置参数;相反,它使用原始缓冲区的当前位置.所以你做不到:

dstBuffer.slice(100).put(srcBuffer);

这里有一些想法,没有特别的顺序,除了它是我想到的顺序:

>如果它符合您使用缓冲区的方式,您可以使用slice()准备缓冲区的副本,并在需要将数据放在与原始位置无关的位置时保留它.
>如果要绝对放置的位置始终大于或等于原始缓冲区的位置指针,则可以执行以下操作:

dstBuffer.slice().position(desiredPosition - dstBuffer.position()).put(srcBuffer);

不幸的是,由于切片上的位置不允许为负,因此无法放在较早的位置.编辑:没关系,我忘记了duplicate方法.请参阅@ Borisbrodski的好答案.

>如果您没有使用直接字节缓冲区,System.arraycopy简单快捷:

System.arraycopy(
    srcBuffer.array(),srcBuffer.arrayOffset() + srcBuffer.position(),dstBuffer.array(),dstBuffer.arrayOffset() + desiredPosition,srcBuffer.remaining()
);

>如果不需要并发访问,则可以在需要执行绝对放置时暂时更改缓冲区的位置,然后将其放回.如果您需要并发访问但线程争用率较低,则可以同步对缓冲区的所有访问(可能很明显,但为了完整性而包括在内):

synchronize (lock) {
    int originalPosition = dstBuffer.position();
    dstBuffer.position(desiredPosition);
    dstBuffer.put(srcBuffer);
    dstBuffer.position(originalPosition);
}

>如果其他想法都不适合你,你可以破解缓冲区.这很麻烦,但这是一个例子:

private static final sun.misc.Unsafe UNSAFE;
static {
    Object result = null;
    try {
        Class<?> klass = Class.forName("sun.misc.Unsafe");
        for (Field field : klass.getDeclaredFields()) {
            if (field.getType() == klass &&
                (field.getModifiers() & (Modifier.FINAL | Modifier.STATIC)) ==
                    (Modifier.FINAL | Modifier.STATIC)) {
                field.setAccessible(true);
                result = field.get(null);
                break;
            }
        }
    } catch (Throwable t) {}
    UNSAFE = result == null ? null : (sun.misc.Unsafe)result;
}

private static final Field ADDRESS_FIELD;
static {
    Field f;
    try {
        f = Buffer.class.getDeclaredField("address");
        f.setAccessible(true);
    } catch (NoSuchFieldException | SecurityException e) {
        f = null;
    }
    ADDRESS_FIELD = f;
}


public static void absolutePut(ByteBuffer dstBuffer,int dstPosition,ByteBuffer srcBuffer) {
    if (!srcBuffer.isDirect()) {
        absolutePut(dstBuffer,dstPosition,srcBuffer.array(),srcBuffer.remaining());
        return;
    }

    if (UNSAFE != null && ADDRESS_FIELD != null && dstBuffer.isDirect()) {
        try {
            long dstAddress = (long)ADDRESS_FIELD.get(dstBuffer) + dstPosition;
            long srcAddress = (long)ADDRESS_FIELD.get(srcBuffer) + srcBuffer.position();
            UNSAFE.copyMemory(srcAddress,dstAddress,srcBuffer.remaining());
        } catch (illegalaccessexception e) {
            throw new RuntimeException(e);
        }
    } else {
        // fallback to basic loop
        for (int i = srcBuffer.position(); i < srcBuffer.limit(); i++) {
            dstBuffer.put(dstPosition + i,srcBuffer.get(i));
        }
    }
}

public static void absolutePut(ByteBuffer dstBuffer,int srcOffset,int length) {
    if (UNSAFE != null && ADDRESS_FIELD != null && dstBuffer.isDirect()) {
        try {
            long dstAddress = (long)ADDRESS_FIELD.get(dstBuffer) + dstPosition;
            UNSAFE.copyMemory(
                src,UNSAFE.arrayBaSEOffset(byte[].class) + srcOffset,null,length);
        } catch (illegalaccessexception e) {
            throw new RuntimeException(e);
        }
    } else {
        // fallback to System.arraycopy
        System.arraycopy(
            src,srcOffset,dstBuffer.arrayOffset() + dstPosition,length);
    }
}

我给这个代码一些最小的测试,混合了直接和非直接缓冲区,看起来没问题.如果反射技术失败(例如,因为您在applet安全沙箱中或Java实现不兼容),它可以回退到更简洁的方法.

java – 缺少ByteBuffer上的一些绝对方法的更多相关文章

  1. android – 缓冲SurfaceCodec的Surface输入

    >来自Android的任何人都可以评论相机和MediaCodec之间是否协调ByteBuffer格式是否在路线图上?

  2. android usb requestWait不返回

    解决方法你的代码中的requestWait()将等待request.queue的结果这里有一些可能的原因,它不返回:>您的请求队列位于与USB设备实际发送数据的不同端点上.仔细检查您是否在正确的界面和端点上进行侦听.>在USB设备固件方面,不要在公共接口类上创建中断端点.创建一个新的供应商特定的接口,否则内核可能会抓住接口,如果它是HID,请求队列可能会失败,这是我发生了什么.

  3. android – NinePatchDrawable没有从块中获取填充

    但它没有.结果:TextView不显示9-Patch图像的内容区域中的文本.每个坐标中的paddings设置为0.谢谢阅读.解决方法最后,我做到了Android没有正确解释块数据.可能有bug所以你必须自己反序列化块以获取填充数据.开始了:使用上面的类如下:它会工作完美!

  4. java – 从文件中读取long []的最快方法?

    我有一个包含大约200,000个长值的文件,我希望尽快将其读入long[].它适用于Android应用;函数调用很慢我需要加载才能快速.我可以用什么?

  5. 使用Java将大于2GB的文件读入内存

    当前代码:对于性能调优,变量in不得来自磁盘.我有很多RAM.解决方法使用NIO将文件读入一个巨大的ByteBuffer,然后创建一个读取ByteBuffer的流类.在开源中有几个这样的浮动.

  6. Java:DataInputStream替换为字节顺序

    下面是我的代码替换DataInputStream来包装一个InputStream,但是除了读取大尾数类型的普通方法之外,还提供了额外的方法来读取小端数据类型.如果你愿意,随意使用它.我有几点保留如下.注意不改变功能的方法.没有办法我可以实现DataInputStream作为基类,并使用它的方法,如read(),readInt(),readChar()等等?我的班级层次似乎有点奇怪.这是否合适?像readUTF()或readLine()这样的其他类型的其他类型需要一个小的endian版本吗?还是主观具体方案

  7. java – 我如何子类化ByteBuffer?

    因此,JavaNIO架构师没有创建一个ByteBuffer接口,而是一个ByteBufferclass,它不是一个最终的类,但它没有包公共构造函数,因此它不能在其包之外被子类化.阿福.:P我有一个程序,使用内存映射文件字节缓冲区(通过FileChannel.map()获得)在一堆的地方,我正在尝试追踪一个讨厌的错误,在该文件的问题是保持打开,因为有至少一个不被释放到垃圾回收的ByteBuffer.

  8. 使用sun.misc.Unsafe获取Java数组项的地址?

    我正在努力理解sun.misc.Unsafe的文档–我想因为它不适合一般用途,没有人真的为让它可读而烦恼–但实际上我真的需要一种方法来找到一个元素的地址一个数组.有没有人有任何工作代码这样做?

  9. java – 比较直接和非直接ByteBuffer的get / put操作

    从直接bytebuffer获取/放置比直接bytebuffer的get/put更快?我会讨厌看到比那更好.;)听起来很复杂通常最简单的解决方案更好,更快.您可以使用此代码自行测试.版画我确定JNI通话需要的时间超过1.2ns.为了证明它不是“JNI”的呼叫,而是围绕它引起延迟的问题.您可以直接使用Unsafe编写相同的循环.版画因此,您可以看到本地电话比您期望的JNI呼叫快得多.这种延迟的主要原因可能是L2缓存速度.总结以上是DEVMAX为你收集整理的java–比较直接和非直接ByteBuffer的get

  10. Java InputStream到ByteBuffer

    我正在读取dds纹理,但是由于一旦构建了jar,我无法通过url和file访问这些纹理,并且必须使用InputStream.所以我需要知道如何从java.io.InputStream获取一个java.nio.ByteBuffer.Ps:无论通过第三方库,我只需要它的工作解决方法对我而言,在这种情况下最好的是Apachecommons-io处理这个和类似的任务.IoUtils类型具有静态方法来读取I

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部