Introduction: Recently, I found in my work that some front-end partners rarely have access to binary data, so I would like to share with you the application of binary data in the project, which is suitable for beginners to understand, and experts are cautious to enter, welcome to clap bricks.
Project background: Implement Canvas drawing based on binary data files (self-defined binary data format) of the company’s original graphics processing.
In other words: AT the beginning of the project, I was also confused. There were so many technical difficulties to overcome. 1. How do I request a binary data stream? 2. How to decompress binary data? 3. How to read binary data? …
Next we break one by one: see recruit open recruit
1. Data requestarraybuffer
:
Based on Ajax requests, set the data format to be arrayBuffer. The stream-based file reading needs to be handled asynchronously, otherwise the data may be lost.
let oReq = null;
if (window.XMLHttpRequest) {
oReq = new XMLHttpRequest();
}
else {
oReq = new ActiveXObject('Microsoft.XMLHTTP'); } oReq.onprogress = this.updateProgress; // Download progress; oReq.responseType ="arraybuffer";
oReq.onload = function() {// Data download will be triggered;if((oReq.status >= 200 && oReq.status < 300) || oReq.status == 304) { var arrayBuffer = oReq.response; ReadFromByteArray(arrayBuffer); // Read the received data}if (oReq.status === 404) {
alert("Can't find corresponding file!")}}; oReq.open("GET", reqUrl, true);
oReq.send(null);
Copy the code
2. Read dataArrayBuffer
andDataView
:
The ArrayBuffer object is used to represent a generic, fixed-length buffer of raw binary data. An ArrayBuffer does not operate directly, but through a type array object or DataView object, which represents the data in the buffer in specific formats and reads and writes the contents of the buffer in those formats.
2.1 Size of verification data
functionReadFromByteArray(buffer){ parseInt(buffer.byteLength / 1024); // File size, in KB;if(buffer.bytelength < 64) {// fail console.log("Wrong file format: length less than 64");
return false;
} else{// Successfully decompress data}}Copy the code
DataView is a low-level interface that can read and write multiple numeric types from an ArrayBuffer object, regardless of platform byte order.
Next we can create a DataView object instance, this method is suitable for sequential storage data read, not sequential (such as incremental storage data can not be sequential, you need to install the table index read, otherwise it will read wrong). Js provides a basic binary reading API. To avoid manually calculating offsets, we can encapsulate the base API.
DataView is a commonly used API for reading data
getFloat32()
getFloat64()
getInt16()
getInt32()
getInt8()
getUint16()
getUint32()
getUint8()
Copy the code
2.2 Verification File name
letdataView = new DataView(buffer, 0); // Pass the buffer obtained above into the viewletheadstr = headerFiler.ReadUTFBytes(5); // Reads 5 UTF8 bytes, resulting in a file formatif(headstr ! ="DWG") {//DWG is a file format, stored in the head of the data structurereturn false;
}else{// Continue reading data}Copy the code
3. Decompress dataTypedArray
andpako.js
:
A TypedArray object describes an array-like view of an underlying binary data cache. In fact, there is no global object named TypedArray, nor is there a TypedArray constructor named TypedArray. Instead, there are many different global objects, and the constructors for these typed arrays for specific element types are listed below. In the following pages, you’ll find some properties and methods that are common regardless of type.
To reduce incoming data, the back end compresses the binary data, but does the front end decompress the code by hand? Even if you dared write, would you dare to use it? Of course, looking for tripartite plug-ins, about js binary data decompression plug-in is really not much, I chose pako.js, mobile terminal temporarily found serious compatibility problems, PC (IE) exist, must be careful to use. Expect more tripartite recommendations.
3.1 createTypedArray
The buffer is first converted to an array of types, TypedArray, for reading and manipulation.
letcompressdata = new Uint8Array(buffer, byteOffset, length); // Convert the 'buffer' obtained above into a controllable 'TypedArray'. Create an unsigned integer TypedArray with byteOffset and length.Copy the code
Tips: The offset is byteOffset, similar to the index of an array. The default value is 0. Such as:
const compressdata = new Uint8Array(buffer, 4, 10); // Start with the fourth byte and the length is 10 bytesCopy the code
3.2 Decompressing Data
Decompress the data using pako.js
letuncompress = pako.inflate(compressdata); // Decompress data;letuncompressdata = uncompress.buffer; // ArrayBuffer {}letdataViewData = new DataView(uncompressdata, 0); // Decompress data;Copy the code
The js number data type, regardless of the size of the number, takes up 8 bytes, or 64 bits, which is the length of a Double in Java. The 1 string takes up 2 bytes, or 16 bits. This provision in JS saves the calculation of data size when we declare variables, which is convenient to use. However, it will waste a lot of storage space and significantly increase the size of data. It is not convenient for the transmission of big data, so the data will be compressed.
Encapsulate the API for reading data to avoid manually calculating offsets
function WsFiler(dataView) {
this.dataView = dataView;
this.dataView.position = 0;
}
WsFiler.SEEK_BEGIN = 0;
WsFiler.SEEK_SET = 0;
WsFiler.SEEK_CUR = 1;
WsFiler.SEEK_END = 2;
WsFiler.prototype.ReadByte = function () {
var b = this.dataView.getUint8(this.dataView.position);
this.dataView.position++;
return b;
}
WsFiler.prototype.ReadShort = function () {
var s = this.dataView.getInt16(this.dataView.position, true);
this.dataView.position += 2;
return s;
};
WsFiler.prototype.ReadInt32 = function () {
var int32 = this.dataView.getInt32(this.dataView.position, true);
this.dataView.position += 4;
return int32;
};
WsFiler.prototype.ReadUInt32 = function () {
var uint32 = this.dataView.getUint32(this.dataView.position, true);
this.dataView.position += 4;
return uint32;
}
WsFiler.prototype.ReadUtf8String = function() { var len = this.ReadInt32(); // String length;return this.ReadUTFBytes(len);
};
WsFiler.prototype.ReadFloat = function () {
var ret = this.dataView.getFloat32(this.dataView.position, true);
this.dataView.position += 4;
return ret;
};
WsFiler.prototype.ReadDouble = function () {
var ret = this.dataView.getFloat64(this.dataView.position, true);
this.dataView.position += 8;
return ret;
}
Copy the code
4. Data Storage:
The read data can be manipulated arbitrarily, and an array can be created to store it. For our subsequent manipulation. Data storage is relatively simple, according to the need to split the data can be.
Now we are done requesting, extracting, reading, and storing binary data.
Continue to share, using canvas to draw the data we read to the web page.
Welcome to clap bricks. Thank you very much!
Reference: MDN
- Developer.mozilla.org/zh-CN/docs/…
- Developer.mozilla.org/zh-CN/docs/…
- Developer.mozilla.org/zh-CN/docs/…