DataView - JavaScript DataView 对象
DataView
DataView
视图是一个可以从二进制ArrayBuffer
对象中读写多种数值类型的底层接口,使用它时,不用考虑不同平台的字节序问题。
语法
new DataView(buffer [, byteOffset [, byteLength]])
参数
buffer
一个已经存在的ArrayBuffer
或SharedArrayBuffer
对象,DataView
对象的数据源。byteOffset
可选此DataView
对象的第一个字节在 buffer 中的字节偏移。如果未指定,则默认从第一个字节开始。byteLength
可选此 DataView 对象的字节长度。如果未指定,这个视图的长度将匹配buffer的长度。返回值
一个表示指定数据缓存区的新DataView
对象。(这句话也许不是非常有助于说明清楚)
你可以把返回的对象想象成一个二进制字节缓存区 array buffer 的“解释器”——它知道如何在读取或写入时正确地转换字节码。这意味着它能在二进制层面处理整数与浮点转化、字节顺序等其他有关的细节问题。
异常
RangeError
如果byteOffset
或者byteLength
参数的值导致视图超出了 buffer 的结束位置就会抛出此异常。例如,假设 buffer (缓冲对象)是 16 字节长度,
byteOffset
参数为 8,byteLength
参数为 10,这个错误就会抛出,这是因为结果视图试图超出 buffer 对象的总长度 2 个字节。描述
Endianness(字节序)
需要多个字节来表示的数值,在存储时其字节在内存中的相对顺序依据平台架构的不同而不同,参照 Endianness。而使用 DataView 的访问函数时,不需要考虑平台架构中所使用的是哪种字节序。
var littleEndian = (function() { var buffer = new ArrayBuffer(2); new DataView(buffer).setInt16(0, 256, true /* 设置值时,使用小端字节序 */); // Int16Array 使用系统字节序(由此可以判断系统字节序是否为小端字节序) return new Int16Array(buffer)[0] === 256; })(); console.log(littleEndian); // 返回 true 或 false
64 位整数值
因为 JavaScript 目前不包含对 64 位整数值支持的标准,所以DataView
不提供原生的 64 位操作。作为变通,您可以实现自己的getUint64()
函数,以获得精度高达Number.MAX_SAFE_INTEGER
的值,可以满足某些特定情况的需求。
function getUint64(dataview, byteOffset, littleEndian) { // 将 64 位整数值分成两份 32 位整数值 const left = dataview.getUint32(byteOffset, littleEndian); const right = dataview.getUint32(byteOffset+4, littleEndian); // 合并两个 32 位整数值 const combined = littleEndian? left + 2**32*right : 2**32*left + right; if (!Number.isSafeInteger(combined)) console.warn(combined, 'exceeds MAX_SAFE_INTEGER. Precision may be lost'); return combined; }
或者,如果需要填满 64 位,可以创建一个BigInt
。此外,尽管原生 BigInt 要比用户端的库中模拟的 BigInt 快得多,但在 JavaScript 中,BigInt 总是比 32 位整数慢得多,这是因为 BigInt 的大小是可变的。
const BigInt = window.BigInt, bigThirtyTwo = BigInt(32), bigZero = BigInt(0); function getUint64BigInt(dataview, byteOffset, littleEndian) { // 将 64 位整数值分成两份 32 位整数值 const left = BigInt(dataview.getUint32(byteOffset|0, !!littleEndian)>>>0); const right = BigInt(dataview.getUint32((byteOffset|0) + 4|0, !!littleEndian)>>>0); // 合并两个 32 位整数值并返回 return littleEndian ? (right
鹏仔微信 15129739599 鹏仔QQ344225443 鹏仔前端 pjxi.com 共享博客 sharedbk.com
图片声明:本站部分配图来自网络。本站只作为美观性配图使用,无任何非法侵犯第三方意图,一切解释权归图片著作权方,本站不承担任何责任。如有恶意碰瓷者,必当奉陪到底严惩不贷!