Java中ByteBuffer的allocate方法 和allocateDirect方法的區(qū)別和選用原則解析
背景
公司開發(fā)一個商用的支付應用。寫協(xié)議的時候需要用到byte類型數(shù)組來填充協(xié)議的報文數(shù)據(jù)。在調研了JDK各個類庫之后,最終選用Java類庫中的ByteBuffer。
在Java中,ByteBuffer是java.nio包中的一個類,用于處理字節(jié)數(shù)據(jù)。ByteBuffer提供了兩種方式來分配內存:allocate和allocateDirect。
/**
* Allocates a new direct byte buffer.
*
* <p> The new buffer's position will be zero, its limit will be its
* capacity, its mark will be undefined, and each of its elements will be
* initialized to zero. Whether or not it has a
* {@link #hasArray backing array} is unspecified.
*
* @param capacity
* The new buffer's capacity, in bytes
*
* @return The new byte buffer
*
* @throws IllegalArgumentException
* If the <tt>capacity</tt> is a negative integer
*/
public static ByteBuffer allocateDirect(int capacity) {
// Android-changed: Android's DirectByteBuffers carry a MemoryRef.
// return new DirectByteBuffer(capacity);
DirectByteBuffer.MemoryRef memoryRef = new DirectByteBuffer.MemoryRef(capacity);
return new DirectByteBuffer(capacity, memoryRef);
} /**
* Allocates a new byte buffer.
*
* <p> The new buffer's position will be zero, its limit will be its
* capacity, its mark will be undefined, and each of its elements will be
* initialized to zero. It will have a {@link #array backing array},
* and its {@link #arrayOffset array offset} will be zero.
*
* @param capacity
* The new buffer's capacity, in bytes
*
* @return The new byte buffer
*
* @throws IllegalArgumentException
* If the <tt>capacity</tt> is a negative integer
*/
public static ByteBuffer allocate(int capacity) {
if (capacity < 0)
throw new IllegalArgumentException();
return new HeapByteBuffer(capacity, capacity);
}區(qū)別
這兩種方式分別對應于不同的內存分配策略,具有不同的優(yōu)劣勢。
1. allocate:
- 使用
allocate方法分配的內存是在Java虛擬機的堆內存中。 ByteBuffer.allocate(capacity)分配的是非直接緩沖區(qū)(non-direct buffer)。- 非直接緩沖區(qū)的操作會在Java堆內存中進行,數(shù)據(jù)的讀寫會通過Java堆來傳遞。
```java ByteBuffer buffer = ByteBuffer.allocate(1024); ```
2. allocateDirect:
- 使用
allocateDirect方法分配的內存是在操作系統(tǒng)的本地內存中,而不是在Java堆內存中。 ByteBuffer.allocateDirect(capacity)分配的是直接緩沖區(qū)(direct buffer)。- 直接緩沖區(qū)的操作可以通過本地I/O傳遞,避免了在Java堆和本地堆之間的數(shù)據(jù)傳輸,可能在某些情況下提供更好的性能。
```java ByteBuffer directBuffer = ByteBuffer.allocateDirect(1024); ```
總結
選擇使用哪種方式取決于應用的需求和性能特征:
- allocate: 適用于較小的緩沖區(qū),對內存占用不太敏感的情況。由于是在Java堆上分配,垃圾回收器能夠管理這部分內存,但可能會有一些性能開銷。
- allocateDirect: 適用于需要較大緩沖區(qū)或對性能要求較高的情況。由于是在本地內存上分配,可能減少了一些垃圾回收器的開銷,但在分配和釋放直接緩沖區(qū)時可能涉及到一些本地資源的操作。
在使用allocateDirect時需要謹慎,因為它可能占用較多的本地內存,過度使用可能導致本地內存耗盡。
在選用這兩種技術方案中哪一種的時候需要根據(jù)具體的應用場景和需求權衡兩者之間的取舍。
到此這篇關于Java中ByteBuffer的allocate方法 和allocateDirect方法的區(qū)別和選用原則 的文章就介紹到這了,更多相關Java ByteBuffer的allocate方法 和allocateDirect方法內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
springboot 2.x整合mybatis實現(xiàn)增刪查和批量處理方式
這篇文章主要介紹了springboot 2.x整合mybatis實現(xiàn)增刪查和批量處理方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
Spark調優(yōu)多線程并行處理任務實現(xiàn)方式
這篇文章主要介紹了Spark調優(yōu)多線程并行處理任務實現(xiàn)方式,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-08-08
SpringBoot JdbcTemplate批量操作的示例代碼
本篇文章主要介紹了SpringBoot JdbcTemplate批量操作的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-04-04
springboot集成spring cache緩存示例代碼
本篇文章主要介紹了springboot集成spring cache示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-05-05
布隆過濾器(Bloom Filter)的Java實現(xiàn)方法
下面小編就為大家?guī)硪黄悸∵^濾器(Bloom Filter)的Java實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-12-12

