如何解决“预期看到2个阵列,但得到以下1个阵列的列表”

时间:2019-06-28 07:13:10

标签: python numpy keras

我正在训练多类cnn模型。 model.fit方法可以正常工作,但是当我使用fit_generator方法时,标题中会出现错误。

y_train_age = utils.to_categorical(y_train_age, 117)
y_test_age = utils.to_categorical(y_test_age, 117)

y_train_gender = utils.to_categorical(y_train_gender, 2)
y_test_gender = utils.to_categorical(y_test_gender, 2)

y_train = np.concatenate((y_train_age, y_train_gender), axis=1)
y_test = np.concatenate((y_test_age, y_test_gender), axis=1)

print(x_train.shape)
print(x_test.shape)
print(y_train.shape)
print(y_test.shape)
  

(15000,100,100,3),(8708,100,100,3),(15000,119),(8708,119)

型号:

from keras import layers
from keras.models import Model
from keras.layers import Input, Dense, Activation
from keras.layers import AveragePooling2D, MaxPooling2D, Flatten, Conv2D, ZeroPadding2D

x_input = Input((100,100,3))

x = Conv2D(64, (3,3))(x_input)
x = Activation('relu')(x)
x = MaxPooling2D((3, 3), strides=(2, 2))(x)

x = Conv2D(64, (3,3))(x)
x = Activation('relu')(x)
x = MaxPooling2D((3, 3), strides=(2, 2))(x)

x = Conv2D(128, (3,3))(x)
x = Activation('relu')(x)
x = MaxPooling2D((3, 3), strides=(2, 2))(x)

x = Conv2D(256, (3,3))(x)
x = Activation('relu')(x)
x = MaxPooling2D((3, 3), strides=(2, 2))(x)

x = Flatten()(x)
x = Dense(64, activation='relu')(x)
x = Dense(128, activation='relu')(x)
x = Dense(128, activation='relu')(x)

y1 = Dense(117, activation='softmax', name="Age")(x)
y2 = Dense(2, activation='softmax', name="Gender")(x)

model = Model(inputs=x_input, outputs=[y1, y2])
model.compile(loss=['categorical_crossentropy', 'categorical_crossentropy'], optimizer='adam', metrics=['accuracy'])
model.summary()

问题所在:

from keras.preprocessing.image import ImageDataGenerator

model.fit_generator(ImageDataGenerator(shear_range=0.3, zoom_range=0.1, 
                    horizontal_flip=True).flow(x_train, y_train, 32),
                    steps_per_epoch=len(x_train) / 32,
                    epochs=5, verbose=1,
                    validation_data=(x_test, y_test))

错误:

ValueError: Error when checking model target: the list of Numpy arrays that you are passing to your model is not the size the model expected. Expected to see 2 array(s), but instead got the following list of 1 arrays: [array([[0., 0., 0., ..., 0., 1., 0.],
       [0., 0., 0., ..., 0., 0., 1.],
       [0., 0., 0., ..., 0., 0., 1.],
       ...,
       [0., 0., 0., ..., 0., 0., 1.],
       [0., 0., 0., ..., 0., 0., 1....

请帮助我,谢谢。

答案

generator = ImageDataGenerator(...)

def generate_data_generator(generator, X, Y1, Y2):
    genX1 = generator.flow(X, Y1, seed=7)
    genX2 = generator.flow(X, Y2, seed=7)
    while True:
        X1i = genX1.next()
        X2i = genX2.next()
        yield X1i[0], [X1i[1], X2i[1]]

history = model.fit_generator(generate_data_generator(generator, x_train, y_train_age, y_train_gender),
                    steps_per_epoch=len(x_train) / 32,
                    epochs=5, 
                    verbose=1, 
                    callbacks = callbacks,
                    validation_data=(x_test, [y_test_age, y_test_gender]))

2 个答案:

答案 0 :(得分:1)

您定义了一个具有2个输出的模型:[y1,y2]
因此,它应该装有两个不同的标签数组,一个标签的大小为(, 119),另一个标签的大小为(,2),它们对应于您的2个Dense输出层。

使用fit函数,它看起来像这样:

model.fit( x = X_train,
           y = [y_train, y_train_gender],
           validation_data=(X_test, [y_test, y_test_gender]),
           batch_size = batch_size,
           epochs = num_epochs,
           verbose = 1)

我不太习惯ImageDataGenerator,但是尝试这样的事情:

from keras.preprocessing.image import ImageDataGenerator

model.fit_generator(ImageDataGenerator(shear_range=0.3, zoom_range=0.1, 
                    horizontal_flip=True).flow(x_train, [y_train, y_train_gender], 32),
                    steps_per_epoch=len(x_train) / 32,
                    epochs=5, verbose=1,
                    validation_data=(x_test, [y_test, y_test_gender]))

编辑

尝试这篇文章的这个小改编:
Keras: How to use fit_generator with multiple outputs of different type

generator = ImageDataGenerator(shear_range=0.3,
                                zoom_range=0.1, 
                                horizontal_flip=True) 


def generate_data_generator(generator, X, Y1, Y2):
    genX1 = generator.flow(X, Y1, seed=7)
    genX2 = generator.flow(X, Y2, seed=7)
    while True:
        X1i = genX1.next()
        X2i = genX2 .next()
        yield X1i[0], [X1i[1], X2i[1]]


model.fit_generator(generate_data_generator(generator, x_train, y_train, y_train_gender),
                    steps_per_epoch=len(x_train) / 32,
                    epochs=5, 
                    verbose=1)

答案 1 :(得分:0)

只需对现有答案进行一点修改,就使用y_train,它是AgeGender的级联向量,但由于其中包含Age,因此您已经有y_train_gender了,它对Gender进行了编码,我对代码的几个部分进行了更改以适应这一点

y1 = Dense(117, activation='softmax', name="Age")(x)    # and not 119
y2 = Dense(2, activation='softmax', name="Gender")(x)

只需在您的y_trainy_train_age方法中将.fit()替换为.fit_generator()。这样,我们将y1用作Age的输出,并将y2用作Gender的输出。

希望这会有所帮助!