2022-03-24 13:42:03 +01:00
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <math.h>
# include <time.h>
2022-04-01 15:41:54 +02:00
# include "struct/neuron.h"
2022-03-24 13:42:03 +01:00
2022-03-25 19:19:27 +01:00
# define TAUX_APPRENTISSAGE 0.15 // Définit le taux d'apprentissage du réseau neuronal, donc la rapidité d'adaptation du modèle (compris entre 0 et 1)
2022-03-24 13:42:03 +01:00
2022-04-01 15:18:48 +02:00
void creation_du_reseau_neuronal ( Reseau * reseau_neuronal , int * neurones_par_couche , int nb_couches ) {
2022-03-24 13:42:03 +01:00
/* Créé les différentes variables dans la variable du réseau neuronal à
2022-03-25 14:26:46 +01:00
partir du nombre de couches et de la liste du nombre de neurones par couche */
2022-03-28 13:53:16 +02:00
reseau_neuronal - > nb_couches = nb_couches ;
2022-04-01 15:18:48 +02:00
reseau_neuronal - > couches = ( Couche * * ) malloc ( sizeof ( Couche * ) * nb_couches ) ; // Création des différentes couches
for ( int i = 0 ; i < nb_couches ; i + + ) {
reseau_neuronal - > couches [ i ] = ( Couche * ) malloc ( sizeof ( Couche ) ) ;
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > nb_neurones = neurones_par_couche [ i ] ; // nombre de neurones pour la couche
2022-03-28 13:53:16 +02:00
reseau_neuronal - > couches [ i ] - > neurones = ( Neurone * * ) malloc ( sizeof ( Neurone * ) * reseau_neuronal - > couches [ i ] - > nb_neurones ) ; // Création des différents neurones dans la couche
2022-04-01 15:32:39 +02:00
for ( int j = 0 ; j < reseau_neuronal - > couches [ i ] - > nb_neurones ; j + + ) {
reseau_neuronal - > couches [ i ] - > neurones [ j ] = ( Neurone * ) malloc ( sizeof ( Neurone ) ) ;
if ( i ! = reseau_neuronal - > nb_couches - 1 ) { // On exclut la dernière couche dont les neurones ne contiennent pas de poids sortants
2022-04-01 15:18:48 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > poids_sortants = ( float * ) malloc ( sizeof ( float ) * neurones_par_couche [ i + 1 ] ) ; // Création des poids sortants du neurone
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dw = ( float * ) malloc ( sizeof ( float ) * neurones_par_couche [ i + 1 ] ) ;
}
2022-03-24 13:42:03 +01:00
}
}
}
2022-04-01 15:18:48 +02:00
void suppression_du_reseau_neuronal ( Reseau * reseau_neuronal ) {
2022-03-24 13:42:03 +01:00
/* Libère l'espace mémoire alloué aux différentes variables dans la fonction
' creation_du_reseau_neuronal ' à partir du nombre de couche et de la liste du
nombre de neurone par couche */
2022-03-28 13:29:29 +02:00
for ( int i = 0 ; i < reseau_neuronal - > nb_couches ; i + + ) {
if ( i ! = reseau_neuronal - > nb_couches - 1 ) { // On exclut la dernière couche dont les neurones ne contiennent pas de poids sortants
2022-03-28 13:53:16 +02:00
for ( int j = 0 ; j < reseau_neuronal - > couches [ i ] - > nb_neurones ; j + + ) {
2022-03-28 13:29:29 +02:00
free ( reseau_neuronal - > couches [ i ] - > neurones [ j ] - > poids_sortants ) ; // On libère la variables des poids sortants
2022-03-24 13:42:03 +01:00
}
}
2022-03-28 13:29:29 +02:00
free ( reseau_neuronal - > couches [ i ] - > neurones ) ; // On libère enfin la liste des neurones de la couche
2022-03-24 13:42:03 +01:00
}
free ( reseau_neuronal ) ; // Pour finir, on libère le réseau neronal contenant la liste des couches
}
2022-04-01 15:18:48 +02:00
void forward_propagation ( Reseau * reseau_neuronal ) {
2022-03-24 13:42:03 +01:00
/* Effectue une propagation en avant du réseau neuronal à partir du nombre
2022-03-25 14:26:46 +01:00
de couches et de la liste du nombre de neurones par couche */
2022-03-24 13:42:03 +01:00
2022-03-28 13:29:29 +02:00
for ( int i = 1 ; i < reseau_neuronal - > nb_couches ; i + + ) { // La première couche contient déjà des valeurs
2022-03-28 13:53:16 +02:00
for ( int j = 0 ; j < reseau_neuronal - > couches [ i ] - > nb_neurones ; j + + ) { // Pour chaque neurone de la couche
2022-03-24 13:42:03 +01:00
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > z = reseau_neuronal - > couches [ i ] - > neurones [ j ] - > biais ; // On réinitialise l'utilisation actuelle du neurone à son biais
2022-03-28 13:53:16 +02:00
for ( int k = 0 ; k < reseau_neuronal - > couches [ i - 1 ] - > nb_neurones ; k + + ) {
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > z + = reseau_neuronal - > couches [ i - 1 ] - > neurones [ k ] - > activation * reseau_neuronal - > couches [ i - 1 ] - > neurones [ k ] - > z * reseau_neuronal - > couches [ i - 1 ] - > neurones [ k ] - > poids_sortants [ i ] ; // ???
2022-03-24 13:42:03 +01:00
}
2022-03-28 13:29:29 +02:00
if ( i < reseau_neuronal - > nb_couches - 1 ) { // Pour toutes les couches sauf la dernière on utilise la fonction relu
if ( reseau_neuronal - > couches [ i ] - > neurones [ j ] - > z < 0 )
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > activation = 0 ;
2022-03-24 13:42:03 +01:00
else
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > activation = reseau_neuronal - > couches [ i ] - > neurones [ j ] - > z ;
2022-03-24 13:42:03 +01:00
}
else { // Pour la dernière couche on utilise la fonction sigmoid permettant d'obtenir un résultat entre 0 et 1 étant une probabilité
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > activation = 1 / ( 1 + exp ( reseau_neuronal - > couches [ i ] - > neurones [ j ] - > activation ) ) ;
2022-03-24 13:42:03 +01:00
}
}
}
}
2022-04-01 15:18:48 +02:00
int * creation_de_la_sortie_voulue ( Reseau * reseau_neuronal , int pos_nombre_voulu ) {
2022-03-25 14:26:46 +01:00
/* Renvoie la liste des sorties voulues à partir du nombre
de couches , de la liste du nombre de neurones par couche et de la
2022-03-24 13:42:03 +01:00
position du résultat voulue , */
int * sortie_voulue = ( int * ) malloc ( sizeof ( int ) ) ;
2022-03-28 13:53:16 +02:00
for ( int i = 0 ; i < reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > nb_neurones ; i + + ) // On initialise toutes les sorties à 0 par défault
2022-03-24 13:42:03 +01:00
sortie_voulue [ i ] = 0 ;
sortie_voulue [ pos_nombre_voulu ] = 1 ; // Seule la sortie voulue vaut 1
return sortie_voulue ;
}
2022-04-01 15:18:48 +02:00
void backward_propagation ( Reseau * reseau_neuronal , int * sortie_voulue ) {
2022-03-24 13:42:03 +01:00
/* Effectue une propagation en arrière du réseau neuronal à partir du
2022-03-25 14:26:46 +01:00
nombre de couches , de la liste du nombre de neurone par couche et de
2022-03-24 13:42:03 +01:00
la liste des sorties voulues */
// On commence par parcourir tous les neurones de la couche finale
2022-03-28 13:53:16 +02:00
for ( int i = 0 ; i < reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > nb_neurones ; i + + ) {
2022-03-24 13:42:03 +01:00
// On applique la formule de propagation en arrière
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > dz = ( reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > activation - sortie_voulue [ i ] ) * ( reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > activation ) * ( 1 - reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > activation ) ;
2022-03-24 13:42:03 +01:00
2022-03-28 13:53:16 +02:00
for ( int k = 0 ; k < reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 2 ] - > nb_neurones ; k + + ) { // Pour chaque neurone de l'avant dernière couche
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 2 ] - > neurones [ k ] - > dw [ i ] = ( reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > dz * reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 2 ] - > neurones [ k ] - > activation ) ;
reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 2 ] - > neurones [ k ] - > dactivation = reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 2 ] - > neurones [ k ] - > poids_sortants [ i ] * reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > dz ;
2022-03-24 13:42:03 +01:00
}
// ???
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > dbiais = reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ i ] - > dz ;
2022-03-24 13:42:03 +01:00
}
2022-03-28 13:29:29 +02:00
for ( int i = reseau_neuronal - > nb_couches - 2 ; i > 0 ; i - - ) { // On remonte les couche de l'avant dernière jusqu'à la première
2022-03-28 13:53:16 +02:00
for ( int j = 0 ; j < reseau_neuronal - > couches [ i ] - > nb_neurones ; j + + ) {
2022-03-28 13:29:29 +02:00
if ( reseau_neuronal - > couches [ i ] - > neurones [ j ] - > z > = 0 ) // ??? ...
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dz = reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dactivation ;
2022-03-24 13:42:03 +01:00
else // ??? ...
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dz = 0 ;
2022-03-24 13:42:03 +01:00
2022-03-28 13:53:16 +02:00
for ( int k = 0 ; k < reseau_neuronal - > couches [ i - 1 ] - > nb_neurones ; k + + ) {
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i - 1 ] - > neurones [ k ] - > dw [ j ] = reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dz * reseau_neuronal - > couches [ i - 1 ] - > neurones [ k ] - > activation ;
2022-03-24 13:42:03 +01:00
if ( i > 1 ) // ??? ...
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i - 1 ] - > neurones [ k ] - > dactivation = reseau_neuronal - > couches [ i - 1 ] - > neurones [ k ] - > poids_sortants [ j ] * reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dz ;
2022-03-24 13:42:03 +01:00
}
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dbiais = reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dz ; // ??? ...
2022-03-24 13:42:03 +01:00
}
}
}
2022-04-01 15:18:48 +02:00
void modification_du_reseau_neuronal ( Reseau * reseau_neuronal ) {
2022-03-24 13:42:03 +01:00
/* Modifie les poids et le biais des neurones du réseau neuronal à partir
du nombre de couches et de la liste du nombre de neurone par couche */
2022-03-28 13:29:29 +02:00
for ( int i = 0 ; i < reseau_neuronal - > nb_couches - 1 ; i + + ) { // on exclut la dernière couche
2022-03-28 13:53:16 +02:00
for ( int j = 0 ; i < reseau_neuronal - > couches [ i ] - > nb_neurones ; j + + ) {
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > biais = reseau_neuronal - > couches [ i ] - > neurones [ j ] - > biais - ( TAUX_APPRENTISSAGE * reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dbiais ) ; // On modifie le biais du neurone à partir des données de la propagation en arrière
2022-03-28 13:53:16 +02:00
for ( int k = 0 ; k < reseau_neuronal - > couches [ i + 1 ] - > nb_neurones ; k + + ) {
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > poids_sortants [ k ] = reseau_neuronal - > couches [ i ] - > neurones [ j ] - > poids_sortants [ k ] - ( TAUX_APPRENTISSAGE * reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dw [ k ] ) ; // On modifie le poids du neurone à partir des données de la propagation en arrière
2022-03-24 13:42:03 +01:00
}
}
}
}
2022-04-01 15:18:48 +02:00
void initialisation_du_reseau_neuronal ( Reseau * reseau_neuronal ) {
2022-03-24 13:42:03 +01:00
/* Initialise les variables du réseau neuronal (activation, biais, poids, ...)
en suivant de la méthode de Xavier . . . . . . à partir du nombre de couches et de la liste du nombre de neurone par couche */
srand ( time ( 0 ) ) ;
2022-03-28 13:29:29 +02:00
for ( int i = 0 ; i < reseau_neuronal - > nb_couches - 1 ; i + + ) { // on exclut la dernière couche
2022-03-28 13:53:16 +02:00
for ( int j = 0 ; j < reseau_neuronal - > couches [ i ] - > nb_neurones - 1 ; j + + ) {
2022-03-25 14:26:46 +01:00
// Initialisation des bornes supérieure et inférieure
2022-03-28 13:53:16 +02:00
double borne_superieure = 1 / sqrt ( reseau_neuronal - > couches [ i ] - > nb_neurones ) ;
2022-03-24 13:42:03 +01:00
double borne_inferieure = - borne_superieure ;
2022-03-28 13:53:16 +02:00
for ( int k = 0 ; k < reseau_neuronal - > couches [ i + 1 ] - > nb_neurones - 1 ; k + + ) { // Pour chaque neurone de la couche suivante auquel le neurone est relié
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > poids_sortants [ k ] = borne_inferieure + ( ( double ) rand ( ) ) / ( ( double ) RAND_MAX ) * ( borne_superieure - borne_inferieure ) ; // Initialisation des poids sortants aléatoirement
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > dw [ k ] = 0.0 ; // ... ???
2022-03-24 13:42:03 +01:00
}
2022-04-01 15:18:48 +02:00
if ( i > 0 ) { // Pour tous les neurones n'étant pas dans la première couche
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ i ] - > neurones [ j ] - > biais = borne_inferieure + ( ( double ) rand ( ) ) / ( ( double ) RAND_MAX ) * ( borne_superieure - borne_inferieure ) ; // On initialise le biais aléatoirement
2022-04-01 15:18:48 +02:00
}
2022-03-24 13:42:03 +01:00
}
}
2022-03-28 13:53:16 +02:00
double borne_superieure = 1 / sqrt ( reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > nb_neurones ) ;
2022-03-24 13:42:03 +01:00
double borne_inferieure = - borne_superieure ;
2022-04-01 15:18:48 +02:00
for ( int j = 0 ; j < reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > nb_neurones ; j + + ) { // Pour chaque neurone de la dernière couche
2022-03-28 13:29:29 +02:00
reseau_neuronal - > couches [ reseau_neuronal - > nb_couches - 1 ] - > neurones [ j ] - > biais = borne_inferieure + ( ( double ) rand ( ) ) / ( ( double ) RAND_MAX ) * ( borne_superieure - borne_inferieure ) ; // On initialise le biais aléatoirement
2022-04-01 15:18:48 +02:00
}
2022-03-25 14:26:46 +01:00
}