keras 支持多处理的Tensorflow2.x自定义数据生成器

txu3uszq  于 2023-03-02  发布在  其他
关注(0)|答案(1)|浏览(143)

我刚刚升级到Tensorflow 2.3,我想制作自己的数据生成器用于训练。使用Tensorflow 1.x,我这样做:

def get_data_generator(test_flag):
  item_list = load_item_list(test_flag)
  print('data loaded')
  while True:
    X = []
    Y = []
    for _ in range(BATCH_SIZE):
      x, y = get_random_augmented_sample(item_list)
      X.append(x)
      Y.append(y)
    yield np.asarray(X), np.asarray(Y)

data_generator_train = get_data_generator(False)
data_generator_test = get_data_generator(True)
model.fit_generator(data_generator_train, validation_data=data_generator_test, 
                    epochs=10000, verbose=2,
                    use_multiprocessing=True,
                    workers=8,
                    validation_steps=100,
                    steps_per_epoch=500,
                    )

此代码在Tensorflow 1.x中运行良好。在系统中创建了8个进程。处理器和显卡加载完美。“数据加载”打印了8次。
使用Tensorflow 2.3时,我收到警告:
警告:tensorflow :多处理可能会与TensorFlow发生不良交互,从而导致不确定性死锁。对于高性能数据管道tf.data建议访问www.example.com。
“数据加载”被打印一次(应该是8次)。GPU没有被充分利用。它也有内存泄漏每个时期,所以训练将停止后,几个时期。使用_multiprocessing标志没有帮助。
如何在tensorflow(keras)2.x中制作一个生成器/迭代器,使其可以轻松地跨多个CPU进程并行化?死锁和数据顺序并不重要。

kzipqqlq

kzipqqlq1#

使用tf.data流水线,您可以在多个点进行并行化。根据数据存储和读取的方式,您可以并行阅读。您还可以并行扩展,并且可以在训练时预取数据,因此您的GPU(或其他硬件)永远不会对数据感到饥饿。
在下面的代码中,我演示了如何并行化扩展和添加预取。

import numpy as np
import tensorflow as tf

x_shape = (32, 32, 3)
y_shape = ()  # A single item (not array).
classes = 10

# This is tf.data.experimental.AUTOTUNE in older tensorflow.
AUTOTUNE = tf.data.AUTOTUNE

def generator_fn(n_samples):
    """Return a function that takes no arguments and returns a generator."""
    def generator():
        for i in range(n_samples):
            # Synthesize an image and a class label.
            x = np.random.random_sample(x_shape).astype(np.float32)
            y = np.random.randint(0, classes, size=y_shape, dtype=np.int32)
            yield x, y
    return generator

def augment(x, y):
    return x * tf.random.normal(shape=x_shape), y

samples = 10
batch_size = 5
epochs = 2

# Create dataset.
gen = generator_fn(n_samples=samples)
dataset = tf.data.Dataset.from_generator(
    generator=gen, 
    output_types=(np.float32, np.int32), 
    output_shapes=(x_shape, y_shape)
)
# Parallelize the augmentation.
dataset = dataset.map(
    augment, 
    num_parallel_calls=AUTOTUNE,
    # Order does not matter.
    deterministic=False
)
dataset = dataset.batch(batch_size, drop_remainder=True)
# Prefetch some batches.
dataset = dataset.prefetch(AUTOTUNE)

# Prepare model.
model = tf.keras.applications.VGG16(weights=None, input_shape=x_shape, classes=classes)
model.compile(optimizer="adam", loss="sparse_categorical_crossentropy")

# Train. Do not specify batch size because the dataset takes care of that.
model.fit(dataset, epochs=epochs)

相关问题