Як об’єднати два шари в керах?


96

У мене є приклад нейронної мережі з двома шарами. Перший шар приймає два аргументи і має один вихід. Другий повинен приймати один аргумент як результат першого шару і один додатковий аргумент. Це повинно виглядати так:

x1  x2  x3
 \  /   /
  y1   /
   \  /
    y2

Отже, я створив модель з двома шарами і намагався їх об’єднати, але вона повертає помилку: The first layer in a Sequential model must get an "input_shape" or "batch_input_shape" argument.на рядку result.add(merged).

Модель:

first = Sequential()
first.add(Dense(1, input_shape=(2,), activation='sigmoid'))

second = Sequential()
second.add(Dense(1, input_shape=(1,), activation='sigmoid'))

result = Sequential()
merged = Concatenate([first, second])
ada_grad = Adagrad(lr=0.1, epsilon=1e-08, decay=0.0)
result.add(merged)
result.compile(optimizer=ada_grad, loss=_loss_tensor, metrics=['accuracy'])

Відповіді:


122

Ви отримуєте помилку, оскільки resultвизначається як Sequential()просто контейнер для моделі, і ви не визначили вхід для неї.

Враховуючи те, що ви намагаєтеся побудувати, встановіть resultтретій вхід x3.

first = Sequential()
first.add(Dense(1, input_shape=(2,), activation='sigmoid'))

second = Sequential()
second.add(Dense(1, input_shape=(1,), activation='sigmoid'))

third = Sequential()
# of course you must provide the input to result with will be your x3
third.add(Dense(1, input_shape=(1,), activation='sigmoid'))

# lets say you add a few more layers to first and second.
# concatenate them
merged = Concatenate([first, second])

# then concatenate the two outputs

result = Concatenate([merged,  third])

ada_grad = Adagrad(lr=0.1, epsilon=1e-08, decay=0.0)

result.compile(optimizer=ada_grad, loss='binary_crossentropy',
               metrics=['accuracy'])

Однак найкращим способом побудови моделі, яка має такий тип вхідної структури, було б використання функціонального API .

Ось реалізація ваших вимог для початку:

from keras.models import Model
from keras.layers import Concatenate, Dense, LSTM, Input, concatenate
from keras.optimizers import Adagrad

first_input = Input(shape=(2, ))
first_dense = Dense(1, )(first_input)

second_input = Input(shape=(2, ))
second_dense = Dense(1, )(second_input)

merge_one = concatenate([first_dense, second_dense])

third_input = Input(shape=(1, ))
merge_two = concatenate([merge_one, third_input])

model = Model(inputs=[first_input, second_input, third_input], outputs=merge_two)
ada_grad = Adagrad(lr=0.1, epsilon=1e-08, decay=0.0)
model.compile(optimizer=ada_grad, loss='binary_crossentropy',
               metrics=['accuracy'])

Щоб відповісти на запитання в коментарях:

1) Як пов’язані результат та злиття? Припускаючи, що ви маєте на увазі, як вони об’єднані.

Конкатенація працює так:

  a        b         c
a b c   g h i    a b c g h i
d e f   j k l    d e f j k l

тобто рядки просто з’єднані.

2) Тепер x1вводиться на перший, x2вводиться на другий і x3вводиться на третій.


Як resultі merged(або merged2) шари пов'язані між собою в першій частині вашої відповіді?
rdo

і друге запитання. Як я розумію x1і x2є вхідною інформацією для first_input, x3для third_input. Про що second_input?
rdo

1
second_inputпропускається через Denseшар і об'єднується з first_inputяким також пропускається через Denseшар. third_inputпропускається через щільний шар і об'єднується в результаті попередньої конкатенації ( merged)
parsethis

2
@putonspectacles Другий спосіб використання функціонального API працює, однак перший спосіб використання послідовної моделі для мене не працює в Keras 2.0.2. Я грубо перевірив реалізацію, і виклик "Concatenate ([...])" не робить багато, крім того, ви не можете додати його до послідовної моделі. Я насправді вважаю, що все одно потрібно використовувати деприкований метод "Злиття ([...], 'concat')", поки вони не оновлять Keras. Що ти думаєш?
LFish

2
Яка різниця між Concatenate()і concatenate()шарами в Keras?
Лієво

8

Додавання до прийнятої вище відповіді, щоб вона допомогла тим, хто користується tensorflow 2.0


import tensorflow as tf

# some data
c1 = tf.constant([[1, 1, 1], [2, 2, 2]], dtype=tf.float32)
c2 = tf.constant([[2, 2, 2], [3, 3, 3]], dtype=tf.float32)
c3 = tf.constant([[3, 3, 3], [4, 4, 4]], dtype=tf.float32)

# bake layers x1, x2, x3
x1 = tf.keras.layers.Dense(10)(c1)
x2 = tf.keras.layers.Dense(10)(c2)
x3 = tf.keras.layers.Dense(10)(c3)

# merged layer y1
y1 = tf.keras.layers.Concatenate(axis=1)([x1, x2])

# merged layer y2
y2 = tf.keras.layers.Concatenate(axis=1)([y1, x3])

# print info
print("-"*30)
print("x1", x1.shape, "x2", x2.shape, "x3", x3.shape)
print("y1", y1.shape)
print("y2", y2.shape)
print("-"*30)

Результат:

------------------------------
x1 (2, 10) x2 (2, 10) x3 (2, 10)
y1 (2, 20)
y2 (2, 30)
------------------------------

7

Можна експериментувати model.summary() (зверніть увагу на розмір шару concatenate_XX (Concatenate))

# merge samples, two input must be same shape
inp1 = Input(shape=(10,32))
inp2 = Input(shape=(10,32))
cc1 = concatenate([inp1, inp2],axis=0) # Merge data must same row column
output = Dense(30, activation='relu')(cc1)
model = Model(inputs=[inp1, inp2], outputs=output)
model.summary()

# merge row must same column size
inp1 = Input(shape=(20,10))
inp2 = Input(shape=(32,10))
cc1 = concatenate([inp1, inp2],axis=1)
output = Dense(30, activation='relu')(cc1)
model = Model(inputs=[inp1, inp2], outputs=output)
model.summary()

# merge column must same row size
inp1 = Input(shape=(10,20))
inp2 = Input(shape=(10,32))
cc1 = concatenate([inp1, inp2],axis=1)
output = Dense(30, activation='relu')(cc1)
model = Model(inputs=[inp1, inp2], outputs=output)
model.summary()

Ви можете переглянути ноутбук тут для деталей: https://nbviewer.jupyter.org/github/anhhh11/DeepLearning/blob/master/Concanate_two_layer_keras.ipynb


3
У чому різниця між Concatenate()і concatenate()шарами в Keras?
Лієво

1
Ви зрозуміли різницю, один - це клас Кераса, а інший - метод
тензорного потоку
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.