Skip to content

Instantly share code, notes, and snippets.

@mjdietzx
Last active March 26, 2024 06:33
Show Gist options
  • Save mjdietzx/0cb95922aac14d446a6530f87b3a04ce to your computer and use it in GitHub Desktop.
Save mjdietzx/0cb95922aac14d446a6530f87b3a04ce to your computer and use it in GitHub Desktop.
Clean and simple Keras implementation of residual networks (ResNeXt and ResNet) accompanying accompanying Deep Residual Learning: https://blog.waya.ai/deep-residual-learning-9610bb62c355.
"""
Clean and simple Keras implementation of network architectures described in:
- (ResNet-50) [Deep Residual Learning for Image Recognition](https://arxiv.org/pdf/1512.03385.pdf).
- (ResNeXt-50 32x4d) [Aggregated Residual Transformations for Deep Neural Networks](https://arxiv.org/pdf/1611.05431.pdf).
Python 3.
"""
from keras import layers
from keras import models
#
# image dimensions
#
img_height = 224
img_width = 224
img_channels = 3
#
# network params
#
cardinality = 32
def residual_network(x):
"""
ResNeXt by default. For ResNet set `cardinality` = 1 above.
"""
def add_common_layers(y):
y = layers.BatchNormalization()(y)
y = layers.LeakyReLU()(y)
return y
def grouped_convolution(y, nb_channels, _strides):
# when `cardinality` == 1 this is just a standard convolution
if cardinality == 1:
return layers.Conv2D(nb_channels, kernel_size=(3, 3), strides=_strides, padding='same')(y)
assert not nb_channels % cardinality
_d = nb_channels // cardinality
# in a grouped convolution layer, input and output channels are divided into `cardinality` groups,
# and convolutions are separately performed within each group
groups = []
for j in range(cardinality):
group = layers.Lambda(lambda z: z[:, :, :, j * _d:j * _d + _d])(y)
groups.append(layers.Conv2D(_d, kernel_size=(3, 3), strides=_strides, padding='same')(group))
# the grouped convolutional layer concatenates them as the outputs of the layer
y = layers.concatenate(groups)
return y
def residual_block(y, nb_channels_in, nb_channels_out, _strides=(1, 1), _project_shortcut=False):
"""
Our network consists of a stack of residual blocks. These blocks have the same topology,
and are subject to two simple rules:
- If producing spatial maps of the same size, the blocks share the same hyper-parameters (width and filter sizes).
- Each time the spatial map is down-sampled by a factor of 2, the width of the blocks is multiplied by a factor of 2.
"""
shortcut = y
# we modify the residual building block as a bottleneck design to make the network more economical
y = layers.Conv2D(nb_channels_in, kernel_size=(1, 1), strides=(1, 1), padding='same')(y)
y = add_common_layers(y)
# ResNeXt (identical to ResNet when `cardinality` == 1)
y = grouped_convolution(y, nb_channels_in, _strides=_strides)
y = add_common_layers(y)
y = layers.Conv2D(nb_channels_out, kernel_size=(1, 1), strides=(1, 1), padding='same')(y)
# batch normalization is employed after aggregating the transformations and before adding to the shortcut
y = layers.BatchNormalization()(y)
# identity shortcuts used directly when the input and output are of the same dimensions
if _project_shortcut or _strides != (1, 1):
# when the dimensions increase projection shortcut is used to match dimensions (done by 1×1 convolutions)
# when the shortcuts go across feature maps of two sizes, they are performed with a stride of 2
shortcut = layers.Conv2D(nb_channels_out, kernel_size=(1, 1), strides=_strides, padding='same')(shortcut)
shortcut = layers.BatchNormalization()(shortcut)
y = layers.add([shortcut, y])
# relu is performed right after each batch normalization,
# expect for the output of the block where relu is performed after the adding to the shortcut
y = layers.LeakyReLU()(y)
return y
# conv1
x = layers.Conv2D(64, kernel_size=(7, 7), strides=(2, 2), padding='same')(x)
x = add_common_layers(x)
# conv2
x = layers.MaxPool2D(pool_size=(3, 3), strides=(2, 2), padding='same')(x)
for i in range(3):
project_shortcut = True if i == 0 else False
x = residual_block(x, 128, 256, _project_shortcut=project_shortcut)
# conv3
for i in range(4):
# down-sampling is performed by conv3_1, conv4_1, and conv5_1 with a stride of 2
strides = (2, 2) if i == 0 else (1, 1)
x = residual_block(x, 256, 512, _strides=strides)
# conv4
for i in range(6):
strides = (2, 2) if i == 0 else (1, 1)
x = residual_block(x, 512, 1024, _strides=strides)
# conv5
for i in range(3):
strides = (2, 2) if i == 0 else (1, 1)
x = residual_block(x, 1024, 2048, _strides=strides)
x = layers.GlobalAveragePooling2D()(x)
x = layers.Dense(1)(x)
return x
image_tensor = layers.Input(shape=(img_height, img_width, img_channels))
network_output = residual_network(image_tensor)
model = models.Model(inputs=[image_tensor], outputs=[network_output])
print(model.summary())
@Rabia-Metis
Copy link

image
Why Resnext ain't learning on my dataset? It's stuck at 50% from start till end. Help please

@emilk
Copy link

emilk commented Jan 13, 2018

Hey, I really enjoyed reading https://blog.waya.ai/deep-residual-learning-9610bb62c355 – thanks!

Small note on the code: project_shortcut = True if i == 0 else False is an obscure way to write project_shortcut = (i == 0) (parentheses optional)

@sungreong
Copy link

sungreong commented Jan 14, 2018

Hi i see your code. i think it 's good but i am newbie in keras
so can you show me process using mnist data sets ?

thanks : )

@kaltu
Copy link

kaltu commented Jul 14, 2018

Hi,
When i try to add plot_model(model, to_file='model.png') after print summary, the dot.exe crashes.
Any thoughts?

@iperov
Copy link

iperov commented Dec 24, 2018

why still using bias before BN ?

@gabrieldp
Copy link

gabrieldp commented Dec 30, 2018

@kaltu Has been a few months since you asked, but maybe this can help others:
The plot_model() function has some external dependencies. You will need graphviz installed (on path), python graphviz library and pydot:
http://www.graphviz.org/
https://pypi.org/project/graphviz/
https://pypi.org/project/pydot/

More info here:
https://www.codesofinterest.com/2017/02/visualizing-model-structures-in-keras.html

If you already have those installed, the problem may be something specific to this layer implementation (couldn't test it myself).

@xiechuxi
Copy link

great work but I don't understand your code below could you explain?
for j in range(cardinality):
group = layers.Lambda(lambda z: z[:, :, :, j * _d:j * _d + _d])(y)
groups.append(layers.Conv2D(_d, kernel_size=(3, 3), strides=_strides, padding='same')(group))

Really thankful

@HitLuca
Copy link

HitLuca commented Jun 21, 2019

double accompanying in the description

@iamsoroush
Copy link

It seems that you made a mistake in residual_block:
"shortcut = y" makes shortcut point to y and at the end shortcut and y are same. so there will be no skip connections.

@elJonathan
Copy link

The input I am trying to give is 3-dimensional. The code is giving an error of a 4D input requirement. How do I modify the model? Please help.

Capture

@pra-dan
Copy link

pra-dan commented Sep 27, 2020

@elJonathan The error means that the data that it was trained on, had 4 dimensions. Re-check it

@shah-scalpel
Copy link

y = layers.Conv2D(nb_channels_in, kernel_size=(1, 1), strides=(1, 1), padding='same')(y) should be y = layers.Conv2D(nb_channels_in, kernel_size=(1, 1), strides=_strides, padding='same')(y) as from conv3, the first conv layer at main path is with subsample=(2,2) And the shortcut should have subsample=(2,2) as well

grouped_convolution(y, nb_channels_in, _strides=_strides) should be grouped_convolution(y, nb_channels_in, _strides=(1, 1)), otherwise it will change the shape

@shah-scalpel
Copy link

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment