All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ai.djl.training.dataset.Batch Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2019 Amazon.com, Inc. or its affiliates. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance
 * with the License. A copy of the License is located at
 *
 * http://aws.amazon.com/apache2.0/
 *
 * or in the "license" file accompanying this file. This file is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES
 * OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions
 * and limitations under the License.
 */
package ai.djl.training.dataset;

import ai.djl.Device;
import ai.djl.ndarray.NDList;
import ai.djl.ndarray.NDManager;
import ai.djl.translate.Batchifier;

import java.util.List;

/**
 * A {@code Batch} is used to hold multiple items (data and label pairs) from a {@link Dataset}.
 *
 * 

When training and performing inference, it is often more efficient to run multiple items * through a network simultaneously rather than one at a time. For this reason, much of the API is * oriented around the {@code Batch} class. * *

In a {@code Batch}, data and label are each an {@link NDList}. The data {@link NDList} * represents the data for each input in the batch. The number of {@link ai.djl.ndarray.NDArray}s in * the NDList is based on the number of different kinds of inputs, not the batch size. Similarly, * the label {@link NDList} represents the labels for each kind of output. * *

For example, an Image Question and Answer dataset has two inputs: an image and a question. In * this case, the data in the {@code Batch} will be an {@link NDList} containing an NCHW image * {@link ai.djl.ndarray.NDArray} and an NTC question {@link ai.djl.ndarray.NDArray}. The label will * be an {@link NDList} containing only an NTC answer {@link ai.djl.ndarray.NDArray}. * *

In order to differentiate a batch vs a single record (despite them both consisting of two * {@link NDList}s), we have the {@link Batch} and the {@link Record} respectively. */ public class Batch implements AutoCloseable { private NDManager manager; private NDList data; private NDList labels; private Batchifier dataBatchifier; private Batchifier labelBatchifier; private int size; private long progress; private long progressTotal; private List indices; /** * Creates a new instance of {@code Batch} with the given manager, data and labels. * * @param manager the manager for the {@code Batch} * @param data the {@link NDList} containing the data * @param labels the {@link NDList} containing the labels * @param size (batchSize) the number of {@link Record}s in the batch * @param dataBatchifier the {@link Batchifier} that is used to split data * @param labelBatchifier the {@link Batchifier} that is used for split labels * @param progress the progress of the batch if it is part of some kind of iteration like a * dataset iteration. Returns 0 if there is no iteration. * @param progressTotal the total or end value for the progress of the batch if it is part of */ public Batch( NDManager manager, NDList data, NDList labels, int size, Batchifier dataBatchifier, Batchifier labelBatchifier, long progress, long progressTotal) { this.manager = manager; this.manager.setName("batch"); data.attach(manager); labels.attach(manager); this.data = data; this.labels = labels; this.size = size; this.dataBatchifier = dataBatchifier; this.labelBatchifier = labelBatchifier; this.progress = progress; this.progressTotal = progressTotal; } /** * Creates a new instance of {@code Batch} with the given manager, data and labels. * * @param manager the manager for the {@code Batch} * @param data the {@link NDList} containing the data * @param labels the {@link NDList} containing the labels * @param size (batchSize) the number of {@link Record}s in the batch * @param dataBatchifier the {@link Batchifier} that is used to split data * @param labelBatchifier the {@link Batchifier} that is used for split labels * @param progress the progress of the batch if it is part of some kind of iteration like a * dataset iteration. Returns 0 if there is no iteration. * @param progressTotal the total or end value for the progress of the batch if it is part of * @param indices the indices used to extract the data and labels */ public Batch( NDManager manager, NDList data, NDList labels, int size, Batchifier dataBatchifier, Batchifier labelBatchifier, long progress, long progressTotal, List indices) { this.manager = manager; this.manager.setName("batch"); data.attach(manager); labels.attach(manager); this.data = data; this.labels = labels; this.size = size; this.dataBatchifier = dataBatchifier; this.labelBatchifier = labelBatchifier; this.progress = progress; this.progressTotal = progressTotal; this.indices = indices; } /** * Gets the {@link NDManager} that is attached to this {@code Batch}. * * @return the {@link NDManager} attached to this {@code Batch} */ public NDManager getManager() { return manager; } /** * Gets the data of this {@code Batch}. * * @return an {@link NDList} that contains the data */ public NDList getData() { return data; } /** * Gets the labels corresponding to the data of this {@code Batch}. * * @return an {@link NDList} that contains the labels */ public NDList getLabels() { return labels; } /** * Returns the batchSize. * * @return the batchSize or number of {@link Record}s in the batch */ public int getSize() { return size; } /** * Returns the progress of the batch if it is part of some kind of iteration like a dataset * iteration. * * @return the progress of the batch if it is part of some kind of iteration like a dataset * iteration. Returns 0 if there is no iteration */ public long getProgress() { return progress; } /** * Returns the total or end value for the progress of the batch if it is part of some kind of * iteration like a dataset iteration. * * @return the total or end value for the progress of the batch if it is part of some kind of * iteration like a dataset iteration. Returns 0 if there is no iteration */ public long getProgressTotal() { return progressTotal; } /** {@inheritDoc} */ @Override public void close() { manager.close(); manager = null; } /** * Splits the data and labels in the {@code Batch} across the given devices. * *

if {@code evenSplit} is {@code false}, that last device may have a smaller batch than the * rest. * * @param devices an array of {@link Device} across which the data must be split * @param evenSplit whether each slice must have the same shape * @return an array of {@code Batch}, each of which corresponds to a {@link Device} */ public Batch[] split(Device[] devices, boolean evenSplit) { int deviceCount = devices.length; if (deviceCount == 1) { // TODO: we should change to following once we support slice: // NDList d = data.asInDevice(devices[0], false); // avoid copy if data already in device if (data.head().getDevice().equals(devices[0])) { return new Batch[] { new Batch( manager.newSubManager(), data, labels, size, dataBatchifier, labelBatchifier, progress, progressTotal, indices) }; } else { NDList d = data.toDevice(devices[0], true); NDList l = labels.toDevice(devices[0], true); return new Batch[] { new Batch( manager.newSubManager(devices[0]), d, l, size, dataBatchifier, labelBatchifier, progress, progressTotal, indices) }; } } NDList[] splittedData = split(data, dataBatchifier, deviceCount, evenSplit); NDList[] splittedLabels = split(labels, labelBatchifier, deviceCount, evenSplit); Batch[] splitted = new Batch[splittedData.length]; int baseSplitSize = size / deviceCount; for (int i = 0; i < splittedData.length; ++i) { NDList d = splittedData[i].toDevice(devices[i], true); NDList l = splittedLabels[i].toDevice(devices[i], true); int subSize = (i == splittedData.length - 1) ? (size - i * baseSplitSize) : baseSplitSize; splitted[i] = new Batch( manager.newSubManager(devices[i]), d, l, subSize, dataBatchifier, labelBatchifier, progress, progressTotal, indices); } return splitted; } private NDList[] split(NDList list, Batchifier batchifier, int numOfSlices, boolean evenSplit) { if (batchifier == null) { throw new IllegalStateException( "Split can only be called on a batch containing a batchifier"); } return batchifier.split(list, numOfSlices, evenSplit); } /** * Returns the indices used to extract the data and labels from the {@link Dataset}. * * @return a list of {@link Long} if the {@link Dataset} is a {@link RandomAccessDataset}, * otherwise may return null. */ public List getIndices() { return indices; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy