用 TensorFlow.js 在浏览器中训练一个计算机视觉模型(手写数字分类器)
创始人
2024-03-21 19:36:21
0

文章目录

  • Building a CNN in JavaScript
  • Using Callbacks for Visualization
  • Training with the MNIST Dataset
  • Running Inference on Images in TensorFlow.js
  • References


我们在《在浏览器中运行 TensorFlow.js 来训练模型并给出预测结果(Iris 数据集)》中已经对 TensorFlow.js 的使用有了大致的了解,现在我们进一步来看如何训练一个图片数据集,并做一些可视化工作。文章代码可从《AI and Machine Learning for Coders》一书 GitHub 找到。

在使用浏览器时,每当我们在一个 URL 上打开一个资源时,就会建立一个 HTTP 连接。我们用这个连接把命令传给服务器,然后服务器就会把结果回传。当涉及到计算机视觉时,我们通常会有大量的训练数据。例如,MNIST 和 Fashion MNIST,尽管它们已经是非常小型的图片数据集,但它们仍然包含了 70,000 张图片,这将是 70,000 个 HTTP 连接!这显然会造成大量的开销,稍后我们看如何处理这个问题。

Building a CNN in JavaScript

我们看在 keras 中定义的如下 针对手写数字数据集的 CNN 模型如何在 JavaScript 中定义:

model = tf.keras.models.Sequential([tf.keras.layers.Conv2D(64, (3, 3),activation='relu', input_shape=(28, 28, 1)),tf.keras.layers.MaxPooling2D(2, 2),tf.keras.layers.Conv2D(64, (3, 3), activation='relu'),tf.keras.layers.MaxPooling2D(2, 2),tf.keras.layers.Flatten(),tf.keras.layers.Dense(128, activation=tf.nn.relu),tf.keras.layers.Dense(10, activation=tf.nn.softmax)
])

我们分别来看看卷积层、池化层、全连接层是如何在 JavaScript 中定义的:

我们首先将模型定义为 sequential

model = tf.sequential();

第一个卷积层:

model.add(tf.layers.conv2d({inputShape: [28, 28, 1],kernelSize: 3,filters: 64,activation: 'relu'}));

第一个池化层:

model.add(tf.layers.maxPooling2d({poolSize: [2, 2]}));

第一个全连接层:

model.add(tf.layers.dense({units: 128, activation: 'relu'}));

因此完整的 JavaScript 定义为:

    model = tf.sequential();model.add(tf.layers.conv2d({inputShape: [28, 28, 1],kernelSize: 3,filters: 64,activation: 'relu'}));model.add(tf.layers.maxPooling2d({poolSize: [2, 2]}));model.add(tf.layers.conv2d({kernelSize: 3,filters: 64,activation: 'relu'}));model.add(tf.layers.maxPooling2d({poolSize: [2, 2]}));model.add(tf.layers.flatten());model.add(tf.layers.dense({units: 128, activation: 'relu'}));model.add(tf.layers.dense({units: 10, activation: 'softmax'}));

编译模型时的语法:

model.compile({optimizer: tf.train.adam(),loss: 'categoricalCrossentropy',metrics: ['accuracy']});

Using Callbacks for Visualization

我们直接使用《在浏览器中运行 TensorFlow.js 来训练模型并给出预测结果(Iris 数据集)》的现成代码来进行演示,代码如下:


Iris Classifier

为了使用可视化工具 tfjs-vis,我们需要在代码中添加如下的 script 标签:


并用 tfvis.show 定义一个回调,来在训练时进行可视化:

const metrics = ['loss', 'accuracy'];const container = {name: 'Model Training', styles: {height: '640px'},tab: 'Training Progress'};const fitCallbacks = tfvis.show.fitCallbacks(container, metrics);

将原有代码中的回调替换成 fitCallbacks,现在我们的完整代码为:


Iris Classifier

运行之后,会有如下结果:

在这里插入图片描述


Training with the MNIST Dataset

我们先如《在浏览器中运行 TensorFlow.js 来训练模型并给出预测结果(Iris 数据集)》一样新建一个项目,并在保持项目的本地路径复制一个 script.js 副本,将其命名为 data.js。下面列出的代码都会放在 data.js 中。

在 TensorFlow.js 中,处理数据训练的一个特殊的方法是将所有的图像附加在一起,成为一个单一的图像,通常称为 sprite sheet,而不是逐个下载每个图像。这种技术通常在游戏开发中使用,游戏的图形被存储在一个文件中,而不是多个小文件,以提高文件存储效率。如果我们把训练用的所有图片都存储在一个文件中,我们只需要打开一个 HTTP 连接,就可以一次性下载所有图片。例如,MNIST 的 sprite sheet 如下图所示:

在这里插入图片描述
这幅图片的维度为 65000×784(28×28),也就是说,我们只需逐行读取该图片文件,就能得到一张张 28×28 像素的图片。

我们可以在 JavaScript 中先将图像加载,然后定义一个画布(canvas),在从原始图像中提取出各个“线条”(行)后,在画布上画出这些“线条”。然后,画布上的字节可以被提取到一个数据集中用于训练。下面我们看具体流程:

训练集测试集比例为 5:1

const IMAGE_SIZE = 784;
const NUM_CLASSES = 10;
const NUM_DATASET_ELEMENTS = 65000;const TRAIN_TEST_RATIO = 5/6;const NUM_TRAIN_ELEMENTS = Math.floor(TRAIN_TEST_RATIO * NUM_DATASET_ELEMENTS);const NUM_TEST_ELEMENTS = NUM_DATASET_ELEMENTS - NUM_TRAIN_ELEMENTS;

定义 canvas:

const img = new Image();
const canvas = document.createElement('canvas');
const ctx = canvas.getContext('2d');

图片地址:

img.src = "https://storage.googleapis.com/learnjs-data/model-builder/mnist_images.png";

一旦图像被加载,我们就可以设置一个 buffer 来容纳其中的字节。该图像是一个 PNG 文件,每个像素有 4 个字节,所以需要为 buffer 预留 65,000×768×4 个字节。我们不需要逐个图像的分割文件,而是可以分块(chunks)分割。通过指定 chunkSize,我们可以一次取五千张图片:

img.onload = () => {img.width = img.naturalWidth;img.height = img.naturalHeight;const datasetBytesBuffer = new ArrayBuffer(NUM_DATASET_ELEMENTS * IMAGE_SIZE * 4);const chunkSize = 5000;canvas.width = img.width;anvas.height = chunkSize;

下面我们通过一个 for 循环来将图片读入 buffer 中,因为图片为灰度图,所以 R\G\B 三个通道的值都是一样的,我们任意选择其中之一:

for (let i = 0; i < NUM_DATASET_ELEMENTS / chunkSize; i++) {const datasetBytesView = new Float32Array(datasetBytesBuffer, i * IMAGE_SIZE * chunkSize * 4,IMAGE_SIZE * chunkSize);ctx.drawImage(img, 0, i * chunkSize, img.width, chunkSize, 0, 0, img.width, chunkSize);const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);for (let j = 0; j < imageData.data.length / 4; j++) {datasetBytesView[j] = imageData.data[j * 4] / 255;}
}this.datasetImages = new Float32Array(datasetBytesBuffer);

和图片类似,标签也是被存储在一个单独的文件中。这是一个二进制文件,对标签进行了稀疏编码。每个标签由 10 个字节表示,其中一个字节的值为 01,代表某个类别。因此,除了逐行下载和解码图像的字节外,我们还需要对标签进行解码。我们使用 arrayBuffer 将标签解码成整数数组。

const labelsRequest = fetch("https://storage.googleapis.com/learnjs-data/model-builder/mnist_labels_uint8");const [imgResponse, labelsResponse] = await Promise.all([imgRequest, labelsRequest]);this.datasetLabels = new Uint8Array(await labelsResponse.arrayBuffer());

然后我们就可以划分训练集和测试集:

this.trainImages = this.datasetImages.slice(0, IMAGE_SIZE * NUM_TRAIN_ELEMENTS);
this.testImages = this.datasetImages.slice(IMAGE_SIZE * NUM_TRAIN_ELEMENTS);this.trainLabels = this.datasetLabels.slice(0, NUM_CLASSES * NUM_TRAIN_ELEMENTS);
this.testLabels = this.datasetLabels.slice(NUM_CLASSES * NUM_TRAIN_ELEMENTS);

和常规流程一样,我们也可以对数据集进行分批打包(batch):

nextBatch(batchSize, data, index) {const batchImagesArray = new Float32Array(batchSize * IMAGE_SIZE);const batchLabelsArray = new Uint8Array(batchSize * NUM_CLASSES);for (let i = 0; i < batchSize; i++) {const idx = index();const image =data[0].slice(idx * IMAGE_SIZE, idx * IMAGE_SIZE + IMAGE_SIZE);batchImagesArray.set(image, i * IMAGE_SIZE);const label =data[1].slice(idx * NUM_CLASSES, idx * NUM_CLASSES + NUM_CLASSES);batchLabelsArray.set(label, i * NUM_CLASSES);}const xs = tf.tensor2d(batchImagesArray, [batchSize, IMAGE_SIZE]);const labels = tf.tensor2d(batchLabelsArray, [batchSize, NUM_CLASSES]);return {xs, labels};
}

然后,训练数据可以使用下面这个批处理函数来返回所需批次大小的且打乱顺序的训练批次:

nextTrainBatch(batchSize) {return this.nextBatch(batchSize, [this.trainImages, this.trainLabels], () => {this.shuffledTrainIndex =(this.shuffledTrainIndex + 1) % this.trainIndices.length;return this.trainIndices[this.shuffledTrainIndex];});
}

测试集数据的处理方式是完全一样的,我们在下面的完整代码中给出。

整个 data.js 文件中代码如下:我们定义了一个类 MnistData 用来封装我们刚刚定义的所有函数方法。之后在 index.html 中,我们直接从 data.js 文件中导入该类,并进行实例化 const data = new MnistData();,就可以直接调用类中的方法。

const IMAGE_SIZE = 784;
const NUM_CLASSES = 10;
const NUM_DATASET_ELEMENTS = 65000;const TRAIN_TEST_RATIO = 5 / 6;const NUM_TRAIN_ELEMENTS = Math.floor(TRAIN_TEST_RATIO * NUM_DATASET_ELEMENTS);
const NUM_TEST_ELEMENTS = NUM_DATASET_ELEMENTS - NUM_TRAIN_ELEMENTS;const MNIST_IMAGES_SPRITE_PATH ='https://storage.googleapis.com/learnjs-data/model-builder/mnist_images.png';
const MNIST_LABELS_PATH ='https://storage.googleapis.com/learnjs-data/model-builder/mnist_labels_uint8';export class MnistData {constructor() {this.shuffledTrainIndex = 0;this.shuffledTestIndex = 0;}async load() {// Make a request for the MNIST sprited image.const img = new Image();const canvas = document.createElement('canvas');const ctx = canvas.getContext('2d');const imgRequest = new Promise((resolve, reject) => {img.crossOrigin = '';img.onload = () => {img.width = img.naturalWidth;img.height = img.naturalHeight;const datasetBytesBuffer =new ArrayBuffer(NUM_DATASET_ELEMENTS * IMAGE_SIZE * 4);const chunkSize = 5000;canvas.width = img.width;canvas.height = chunkSize;for (let i = 0; i < NUM_DATASET_ELEMENTS / chunkSize; i++) {const datasetBytesView = new Float32Array(datasetBytesBuffer, i * IMAGE_SIZE * chunkSize * 4,IMAGE_SIZE * chunkSize);ctx.drawImage(img, 0, i * chunkSize, img.width, chunkSize, 0, 0, img.width,chunkSize);const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);for (let j = 0; j < imageData.data.length / 4; j++) {// All channels hold an equal value since the image is grayscale, so// just read the red channel.datasetBytesView[j] = imageData.data[j * 4] / 255;}}this.datasetImages = new Float32Array(datasetBytesBuffer);resolve();};img.src = MNIST_IMAGES_SPRITE_PATH;});const labelsRequest = fetch(MNIST_LABELS_PATH);const [imgResponse, labelsResponse] =await Promise.all([imgRequest, labelsRequest]);this.datasetLabels = new Uint8Array(await labelsResponse.arrayBuffer());// Create shuffled indices into the train/test set for when we select a// random dataset element for training / validation.this.trainIndices = tf.util.createShuffledIndices(NUM_TRAIN_ELEMENTS);this.testIndices = tf.util.createShuffledIndices(NUM_TEST_ELEMENTS);// Slice the the images and labels into train and test sets.this.trainImages =this.datasetImages.slice(0, IMAGE_SIZE * NUM_TRAIN_ELEMENTS);this.testImages = this.datasetImages.slice(IMAGE_SIZE * NUM_TRAIN_ELEMENTS);this.trainLabels =this.datasetLabels.slice(0, NUM_CLASSES * NUM_TRAIN_ELEMENTS);this.testLabels =this.datasetLabels.slice(NUM_CLASSES * NUM_TRAIN_ELEMENTS);}nextTrainBatch(batchSize) {return this.nextBatch(batchSize, [this.trainImages, this.trainLabels], () => {this.shuffledTrainIndex =(this.shuffledTrainIndex + 1) % this.trainIndices.length;return this.trainIndices[this.shuffledTrainIndex];});}nextTestBatch(batchSize) {return this.nextBatch(batchSize, [this.testImages, this.testLabels], () => {this.shuffledTestIndex =(this.shuffledTestIndex + 1) % this.testIndices.length;return this.testIndices[this.shuffledTestIndex];});}nextBatch(batchSize, data, index) {const batchImagesArray = new Float32Array(batchSize * IMAGE_SIZE);const batchLabelsArray = new Uint8Array(batchSize * NUM_CLASSES);for (let i = 0; i < batchSize; i++) {const idx = index();const image =data[0].slice(idx * IMAGE_SIZE, idx * IMAGE_SIZE + IMAGE_SIZE);batchImagesArray.set(image, i * IMAGE_SIZE);const label =data[1].slice(idx * NUM_CLASSES, idx * NUM_CLASSES + NUM_CLASSES);batchLabelsArray.set(label, i * NUM_CLASSES);}const xs = tf.tensor2d(batchImagesArray, [batchSize, IMAGE_SIZE]);const labels = tf.tensor2d(batchLabelsArray, [batchSize, NUM_CLASSES]);return {xs, labels};}
}

下面定义的回调以及训练代码,我们之后会直接封装到 index.html 中的 train 函数当中。

还记得我们刚刚使用的可视化回调吗?我们这里再将它定义出来:

const metrics = ['loss', 'val_loss', 'accuracy', 'val_accuracy'];
const container = { name: 'Model Training', styles: { height: '640px' },tab: 'Training Progress' };
const fitCallbacks = tfvis.show.fitCallbacks(container, metrics);

调用函数生成训练、测试数据集:

const [trainXs, trainYs] = tf.tidy(() => {const d = data.nextTrainBatch(TRAIN_DATA_SIZE);return [d.xs.reshape([TRAIN_DATA_SIZE, 28, 28, 1]),d.labels];
});const [testXs, testYs] = tf.tidy(() => {const d = data.nextTestBatch(TEST_DATA_SIZE);return [d.xs.reshape([TEST_DATA_SIZE, 28, 28, 1]),d.labels];
});

注意这里 tf.tidy 的使用。在 TensorFlow.js 中,它将帮助我们清理所有中间张量,除了那些函数返回的张量。在使用 TensorFlow.js 时,这对防止浏览器中的内存泄漏至关重要。

现在万事俱备,我们就可以进行训练啦!

return model.fit(trainXs, trainYs, {batchSize: BATCH_SIZE,validationData: [testXs, testYs],epochs: 20,shuffle: true,callbacks: fitCallbacks
});

大家这时可能还跑不出下面的训练过程,别着急,我们一会给出 index.html 文件的完整代码,大家到时直接运行即可。

在这里插入图片描述


Running Inference on Images in TensorFlow.js

推断时我们需要一张测试图片,我们可以直接创建一个画布对象,然后让用户使用鼠标在画布上写出要判断的数字:

rawImage = document.getElementById('canvasimg');
ctx = canvas.getContext("2d");
ctx.fillStyle = "black";
ctx.fillRect(0,0,280,280);

在用户通过下面的 draw 函数写好数字之后:

function draw(e) {if(e.buttons!=1) return;ctx.beginPath();ctx.lineWidth = 24;ctx.lineCap = 'round';ctx.strokeStyle = 'white';ctx.moveTo(pos.x, pos.y);setPosition(e);ctx.lineTo(pos.x, pos.y);ctx.stroke();rawImage.src = canvas.toDataURL('image/png');
}

我们从画布上抓取像素,并处理成模型可以处理的输入张量:

var raw = tf.browser.fromPixels(rawImage,1);var resized = tf.image.resizeBilinear(raw, [28,28]);var tensor = resized.expandDims(0);

之后我们就可以进行预测:

var prediction = model.predict(tensor);
var pIndex = tf.argMax(prediction, 1).dataSync();

index.thml 文件完整代码:



Handwriting Classifier!

当运行之后,我们等模型训练完毕,右侧就会出现下面的界面:
在这里插入图片描述

我们直接用鼠标在黑色画布上随便写一个数字,然后点击 classify:

在这里插入图片描述
在这里插入图片描述


References

AI and Machine Learning for Coders by Laurence Moroney.

相关内容

热门资讯

汽车油箱结构是什么(汽车油箱结... 本篇文章极速百科给大家谈谈汽车油箱结构是什么,以及汽车油箱结构原理图解对应的知识点,希望对各位有所帮...
美国2年期国债收益率上涨15个... 原标题:美国2年期国债收益率上涨15个基点 美国2年期国债收益率上涨15个基...
嵌入式 ADC使用手册完整版 ... 嵌入式 ADC使用手册完整版 (188977万字)💜&#...
重大消息战皇大厅开挂是真的吗... 您好:战皇大厅这款游戏可以开挂,确实是有挂的,需要了解加客服微信【8435338】很多玩家在这款游戏...
盘点十款牵手跑胡子为什么一直... 您好:牵手跑胡子这款游戏可以开挂,确实是有挂的,需要了解加客服微信【8435338】很多玩家在这款游...
senator香烟多少一盒(s... 今天给各位分享senator香烟多少一盒的知识,其中也会对sevebstars香烟进行解释,如果能碰...
终于懂了新荣耀斗牛真的有挂吗... 您好:新荣耀斗牛这款游戏可以开挂,确实是有挂的,需要了解加客服微信8435338】很多玩家在这款游戏...
盘点十款明星麻将到底有没有挂... 您好:明星麻将这款游戏可以开挂,确实是有挂的,需要了解加客服微信【5848499】很多玩家在这款游戏...
总结文章“新道游棋牌有透视挂吗... 您好:新道游棋牌这款游戏可以开挂,确实是有挂的,需要了解加客服微信【7682267】很多玩家在这款游...
终于懂了手机麻将到底有没有挂... 您好:手机麻将这款游戏可以开挂,确实是有挂的,需要了解加客服微信【8435338】很多玩家在这款游戏...