Structures de données et algorithmes - Guide rapide

La structure des données est un moyen systématique d'organiser les données afin de les utiliser efficacement. Les termes suivants sont les termes fondamentaux d'une structure de données.

  • Interface- Chaque structure de données a une interface. L'interface représente l'ensemble des opérations prises en charge par une structure de données. Une interface fournit uniquement la liste des opérations prises en charge, le type de paramètres qu'elles peuvent accepter et le type de retour de ces opérations.

  • Implementation- La mise en œuvre fournit la représentation interne d'une structure de données. La mise en œuvre fournit également la définition des algorithmes utilisés dans les opérations de la structure de données.

Caractéristiques d'une structure de données

  • Correctness - L'implémentation de la structure de données doit implémenter correctement son interface.

  • Time Complexity - Le temps de fonctionnement ou le temps d'exécution des opérations de structure de données doit être le plus petit possible.

  • Space Complexity - L'utilisation de la mémoire d'une opération de structure de données doit être aussi faible que possible.

Besoin de structure de données

À mesure que les applications deviennent complexes et riches en données, il existe trois problèmes courants auxquels les applications sont confrontées de nos jours.

  • Data Search- Considérez un inventaire de 1 million (10 6 ) articles d'un magasin. Si l'application cherche à rechercher un élément, elle doit rechercher un élément dans 1 million (10 6 ) éléments à chaque fois, ce qui ralentit la recherche. À mesure que les données augmentent, la recherche deviendra plus lente.

  • Processor speed - La vitesse du processeur, bien qu'étant très élevée, est limitée si les données atteignent des milliards d'enregistrements.

  • Multiple requests - Comme des milliers d'utilisateurs peuvent rechercher des données simultanément sur un serveur Web, même le serveur rapide échoue lors de la recherche des données.

Pour résoudre les problèmes mentionnés ci-dessus, les structures de données viennent à la rescousse. Les données peuvent être organisées dans une structure de données de telle sorte que tous les éléments ne doivent pas nécessairement être recherchés, et les données requises peuvent être recherchées presque instantanément.

Cas de temps d'exécution

Il existe trois cas qui sont généralement utilisés pour comparer le temps d'exécution de diverses structures de données de manière relative.

  • Worst Case- Il s'agit du scénario dans lequel une opération de structure de données particulière prend le plus de temps possible. Si le pire des cas d'une opération est ƒ (n) alors cette opération ne prendra pas plus de ƒ (n) temps où ƒ (n) représente la fonction de n.

  • Average Case- Il s'agit du scénario représentant le temps d'exécution moyen d'une opération d'une structure de données. Si une opération prend ƒ (n) temps d'exécution, alors m opérations prendront mƒ (n) temps.

  • Best Case- Il s'agit du scénario représentant le temps d'exécution le moins élevé possible d'une opération d'une structure de données. Si une opération prend ƒ (n) temps d'exécution, alors l'opération réelle peut prendre du temps en tant que nombre aléatoire qui serait au maximum comme ƒ (n).

Terminologie de base

  • Data - Les données sont des valeurs ou un ensemble de valeurs.

  • Data Item - L'élément de données fait référence à une seule unité de valeurs.

  • Group Items - Les éléments de données divisés en sous-éléments sont appelés éléments de groupe.

  • Elementary Items - Les éléments de données qui ne peuvent pas être divisés sont appelés éléments élémentaires.

  • Attribute and Entity - Une entité est celle qui contient certains attributs ou propriétés auxquels des valeurs peuvent être attribuées.

  • Entity Set - Les entités d'attributs similaires forment un ensemble d'entités.

  • Field - Le champ est une seule unité élémentaire d'information représentant un attribut d'une entité.

  • Record - Record est une collection de valeurs de champ d'une entité donnée.

  • File - Le fichier est une collection d'enregistrements des entités dans un ensemble d'entités donné.

Essayez-le Option en ligne

Vous n'avez vraiment pas besoin de configurer votre propre environnement pour commencer à apprendre le langage de programmation C. La raison est très simple, nous avons déjà mis en place un environnement de programmation C en ligne, afin que vous puissiez compiler et exécuter tous les exemples disponibles en ligne en même temps que vous faites votre travail théorique. Cela vous donne confiance dans ce que vous lisez et de vérifier le résultat avec différentes options. N'hésitez pas à modifier n'importe quel exemple et à l'exécuter en ligne.

Essayez l'exemple suivant en utilisant le Try it option disponible dans le coin supérieur droit de la boîte d'exemple de code -

#include <stdio.h>
int main(){
   /* My first program in C */
   printf("Hello, World! \n");
   return 0;
}

Pour la plupart des exemples donnés dans ce didacticiel, vous trouverez l'option Essayer, alors utilisez-la et profitez de votre apprentissage.

Configuration de l'environnement local

Si vous êtes toujours prêt à configurer votre environnement pour le langage de programmation C, vous avez besoin des deux outils suivants disponibles sur votre ordinateur, (a) l'éditeur de texte et (b) le compilateur C.

Éditeur de texte

Cela sera utilisé pour taper votre programme. Quelques exemples d'éditeurs incluent le Bloc-notes Windows, la commande d'édition du système d'exploitation, Brief, Epsilon, EMACS et vim ou vi.

Le nom et la version de l'éditeur de texte peuvent varier selon les systèmes d'exploitation. Par exemple, le Bloc-notes sera utilisé sous Windows, et vim ou vi pourra être utilisé sous Windows ainsi que Linux ou UNIX.

Les fichiers que vous créez avec votre éditeur sont appelés fichiers source et contiennent le code source du programme. Les fichiers source des programmes C sont généralement nommés avec l'extension ".c".

Avant de commencer votre programmation, assurez-vous d'avoir un éditeur de texte en place et que vous avez suffisamment d'expérience pour écrire un programme informatique, l'enregistrer dans un fichier, le compiler et enfin l'exécuter.

Le compilateur C

Le code source écrit dans le fichier source est la source lisible par l'homme pour votre programme. Il doit être "compilé", pour se transformer en langage machine afin que votre CPU puisse réellement exécuter le programme selon les instructions données.

Ce compilateur de langage de programmation C sera utilisé pour compiler votre code source dans un programme exécutable final. Nous supposons que vous avez les connaissances de base sur un compilateur de langage de programmation.

Le compilateur le plus fréquemment utilisé et disponible gratuitement est le compilateur GNU C / C ++. Sinon, vous pouvez avoir des compilateurs de HP ou de Solaris si vous disposez de systèmes d'exploitation (SE) respectifs.

La section suivante vous explique comment installer le compilateur GNU C / C ++ sur différents systèmes d'exploitation. Nous mentionnons ensemble C / C ++ car le compilateur GNU GCC fonctionne pour les langages de programmation C et C ++.

Installation sous UNIX / Linux

Si vous utilisez Linux or UNIX, puis vérifiez si GCC est installé sur votre système en entrant la commande suivante à partir de la ligne de commande -

$ gcc -v

Si le compilateur GNU est installé sur votre machine, il devrait afficher un message tel que le suivant -

Using built-in specs.
Target: i386-redhat-linux
Configured with: ../configure --prefix = /usr .......
Thread model: posix
gcc version 4.1.2 20080704 (Red Hat 4.1.2-46)

Si GCC n'est pas installé, vous devrez l'installer vous-même en suivant les instructions détaillées disponibles sur https://gcc.gnu.org/install/

Ce tutoriel a été écrit sur la base de Linux et tous les exemples donnés ont été compilés sur la version Cent OS du système Linux.

Installation sous Mac OS

Si vous utilisez Mac OS X, le moyen le plus simple d'obtenir GCC est de télécharger l'environnement de développement Xcode à partir du site Web d'Apple et de suivre les instructions d'installation simples. Une fois que vous avez configuré Xcode, vous pourrez utiliser le compilateur GNU pour C / C ++.

Xcode est actuellement disponible sur developer.apple.com/technologies/tools/

Installation sous Windows

Pour installer GCC sur Windows, vous devez installer MinGW. Pour installer MinGW, allez sur la page d'accueil de MinGW, www.mingw.org , et suivez le lien vers la page de téléchargement de MinGW. Téléchargez la dernière version du programme d'installation MinGW, qui doit être nommée MinGW- <version> .exe.

Lors de l'installation de MinWG, au minimum, vous devez installer gcc-core, gcc-g ++, binutils et le runtime MinGW, mais vous souhaiterez peut-être en installer d'autres.

Ajoutez le sous-répertoire bin de votre installation MinGW à votre PATH variable d'environnement, afin que vous puissiez spécifier ces outils sur la ligne de commande par leurs noms simples.

Une fois l'installation terminée, vous pourrez exécuter gcc, g ++, ar, ranlib, dlltool et plusieurs autres outils GNU à partir de la ligne de commande Windows.

L'algorithme est une procédure étape par étape, qui définit un ensemble d'instructions à exécuter dans un certain ordre pour obtenir la sortie souhaitée. Les algorithmes sont généralement créés indépendamment des langages sous-jacents, c'est-à-dire qu'un algorithme peut être implémenté dans plus d'un langage de programmation.

Du point de vue de la structure des données, voici quelques catégories importantes d'algorithmes -

  • Search - Algorithme pour rechercher un élément dans une structure de données.

  • Sort - Algorithme pour trier les éléments dans un certain ordre.

  • Insert - Algorithme pour insérer un élément dans une structure de données.

  • Update - Algorithme pour mettre à jour un élément existant dans une structure de données.

  • Delete - Algorithme pour supprimer un élément existant d'une structure de données.

Caractéristiques d'un algorithme

Toutes les procédures ne peuvent pas être appelées un algorithme. Un algorithme doit avoir les caractéristiques suivantes -

  • Unambiguous- L'algorithme doit être clair et sans ambiguïté. Chacune de ses étapes (ou phases) et leurs entrées / sorties doivent être claires et ne doivent conduire qu'à une seule signification.

  • Input - Un algorithme doit avoir 0 ou plus d'entrées bien définies.

  • Output - Un algorithme doit avoir une ou plusieurs sorties bien définies et doit correspondre à la sortie souhaitée.

  • Finiteness - Les algorithmes doivent se terminer après un nombre fini d'étapes.

  • Feasibility - Doit être réalisable avec les ressources disponibles.

  • Independent - Un algorithme doit avoir des directions étape par étape, qui doivent être indépendantes de tout code de programmation.

Comment écrire un algorithme?

Il n'y a pas de normes bien définies pour l'écriture d'algorithmes. Il dépend plutôt du problème et des ressources. Les algorithmes ne sont jamais écrits pour prendre en charge un code de programmation particulier.

Comme nous savons que tous les langages de programmation partagent des constructions de code de base comme des boucles (do, for, while), flow-control (if-else), etc. Ces constructions communes peuvent être utilisées pour écrire un algorithme.

Nous écrivons des algorithmes étape par étape, mais ce n'est pas toujours le cas. L'écriture d'algorithme est un processus et est exécutée une fois que le domaine du problème est bien défini. Autrement dit, nous devons connaître le domaine du problème, pour lequel nous concevons une solution.

Exemple

Essayons d'apprendre l'écriture d'algorithmes en utilisant un exemple.

Problem - Concevez un algorithme pour ajouter deux nombres et afficher le résultat.

Step 1 − START
Step 2 − declare three integers a, b & c
Step 3 − define values of a & b
Step 4 − add values of a & b
Step 5 − store output of step 4 to c
Step 6 − print c
Step 7 − STOP

Les algorithmes indiquent aux programmeurs comment coder le programme. Alternativement, l'algorithme peut être écrit comme -

Step 1 − START ADD
Step 2 − get values of a & b
Step 3 − c ← a + b
Step 4 − display c
Step 5 − STOP

Dans la conception et l'analyse des algorithmes, la deuxième méthode est généralement utilisée pour décrire un algorithme. Cela permet à l'analyste d'analyser facilement l'algorithme en ignorant toutes les définitions indésirables. Il peut observer quelles opérations sont utilisées et comment le processus se déroule.

L'écriture step numbers, est facultatif.

Nous concevons un algorithme pour obtenir une solution à un problème donné. Un problème peut être résolu de plusieurs manières.

Par conséquent, de nombreux algorithmes de solution peuvent être dérivés pour un problème donné. L'étape suivante consiste à analyser les algorithmes de solution proposés et à mettre en œuvre la meilleure solution appropriée.

Analyse d'algorithme

L'efficacité d'un algorithme peut être analysée à deux étapes différentes, avant et après la mise en œuvre. Ce sont les suivants -

  • A Priori Analysis- Il s'agit d'une analyse théorique d'un algorithme. L'efficacité d'un algorithme est mesurée en supposant que tous les autres facteurs, par exemple la vitesse du processeur, sont constants et n'ont aucun effet sur l'implémentation.

  • A Posterior Analysis- Il s'agit d'une analyse empirique d'un algorithme. L'algorithme sélectionné est implémenté en utilisant le langage de programmation. Ceci est ensuite exécuté sur la machine informatique cible. Dans cette analyse, des statistiques réelles telles que le temps de fonctionnement et l'espace requis sont collectées.

Nous apprendrons l' analyse d'algorithme a priori . L'analyse des algorithmes traite de l'exécution ou du temps d'exécution des diverses opérations impliquées. Le temps d'exécution d'une opération peut être défini comme le nombre d'instructions informatiques exécutées par opération.

Complexité de l'algorithme

Supposer X est un algorithme et n est la taille des données d'entrée, le temps et l'espace utilisés par l'algorithme X sont les deux principaux facteurs qui décident de l'efficacité de X.

  • Time Factor - Le temps est mesuré en comptant le nombre d'opérations clés telles que des comparaisons dans l'algorithme de tri.

  • Space Factor - L'espace est mesuré en comptant l'espace mémoire maximal requis par l'algorithme.

La complexité d'un algorithme f(n) donne le temps de fonctionnement et / ou l'espace de stockage requis par l'algorithme en termes de n comme la taille des données d'entrée.

Complexité spatiale

La complexité spatiale d'un algorithme représente la quantité d'espace mémoire requise par l'algorithme dans son cycle de vie. L'espace requis par un algorithme est égal à la somme des deux composantes suivantes -

  • Une partie fixe qui est un espace nécessaire pour stocker certaines données et variables, qui sont indépendantes de la taille du problème. Par exemple, les variables simples et les constantes utilisées, la taille du programme, etc.

  • Une partie variable est un espace requis par des variables, dont la taille dépend de la taille du problème. Par exemple, l'allocation de mémoire dynamique, l'espace de pile de récursivité, etc.

La complexité spatiale S (P) de tout algorithme P est S (P) = C + SP (I), où C est la partie fixe et S (I) est la partie variable de l'algorithme, qui dépend de la caractéristique d'instance I. est un exemple simple qui tente d'expliquer le concept -

Algorithm: SUM(A, B)
Step 1 -  START
Step 2 -  C ← A + B + 10
Step 3 -  Stop

Ici, nous avons trois variables A, B et C et une constante. D'où S (P) = 1 + 3. Maintenant, l'espace dépend des types de données des variables données et des types de constantes et il sera multiplié en conséquence.

Complexité temporelle

La complexité temporelle d'un algorithme représente le temps nécessaire à l'algorithme pour s'exécuter jusqu'à son terme. Les exigences de temps peuvent être définies comme une fonction numérique T (n), où T (n) peut être mesurée comme le nombre d'étapes, à condition que chaque étape consomme un temps constant.

Par exemple, l'ajout de deux entiers de n bits prend npas. Par conséquent, le temps de calcul total est T (n) = c ∗ n, où c est le temps pris pour l'addition de deux bits. Ici, nous observons que T (n) croît linéairement à mesure que la taille d'entrée augmente.

L'analyse asymptotique d'un algorithme fait référence à la définition de la limite / cadrage mathématique de ses performances d'exécution. En utilisant une analyse asymptotique, nous pouvons très bien conclure le meilleur cas, le cas moyen et le pire des scénarios d'un algorithme.

L'analyse asymptotique est liée à l'entrée, c'est-à-dire que s'il n'y a pas d'entrée dans l'algorithme, elle est conclue qu'elle fonctionne en temps constant. Hormis l '«entrée», tous les autres facteurs sont considérés comme constants.

L'analyse asymptotique fait référence au calcul du temps d'exécution de toute opération en unités mathématiques de calcul. Par exemple, le temps d'exécution d'une opération est calculé comme f (n) et peut être pour une autre opération, il est calculé comme g (n 2 ). Cela signifie que le temps de fonctionnement de la première opération augmentera linéairement avec l'augmentation den et le temps de fonctionnement de la deuxième opération augmentera de façon exponentielle lorsque naugmente. De même, la durée d'exécution des deux opérations sera presque la même sin est significativement petit.

Habituellement, le temps requis par un algorithme relève de trois types -

  • Best Case - Temps minimum requis pour l'exécution du programme.

  • Average Case - Temps moyen requis pour l'exécution du programme.

  • Worst Case - Temps maximum requis pour l'exécution du programme.

Notations asymptotiques

Voici les notations asymptotiques couramment utilisées pour calculer la complexité du temps d'exécution d'un algorithme.

  • Ο Notation
  • Notation Ω
  • Notation θ

Grosse Oh Notation,

La notation Ο (n) est la manière formelle d'exprimer la limite supérieure du temps d'exécution d'un algorithme. Il mesure la complexité temporelle du cas le plus défavorable ou le temps le plus long qu'un algorithme peut prendre pour se terminer.

Par exemple, pour une fonction f(n)

Ο(f(n)) = { g(n) : there exists c > 0 and n0 such that f(n) ≤ c.g(n) for all n > n0. }

Notation oméga, Ω

La notation Ω (n) est la manière formelle d'exprimer la borne inférieure du temps d'exécution d'un algorithme. Il mesure la meilleure complexité temporelle du cas ou le meilleur temps qu'un algorithme peut prendre pour se terminer.

Par exemple, pour une fonction f(n)

Ω(f(n)) ≥ { g(n) : there exists c > 0 and n0 such that g(n) ≤ c.f(n) for all n > n0. }

Notation thêta, θ

La notation θ (n) est la manière formelle d'exprimer à la fois la borne inférieure et la borne supérieure du temps d'exécution d'un algorithme. Il est représenté comme suit -

θ(f(n)) = { g(n) if and only if g(n) =  Ο(f(n)) and g(n) = Ω(f(n)) for all n > n0. }

Notations asymptotiques courantes

Voici une liste de quelques notations asymptotiques courantes -

constant - Ο (1)
logarithmique - Ο (log n)
linéaire - Ο (n)
n log n - Ο (n log n)
quadratique - Ο (n 2 )
cubique - Ο (n 3 )
polynôme - n Ο (1)
exponentiel - 2 Ο (n)

Un algorithme est conçu pour parvenir à une solution optimale pour un problème donné. Dans l'approche d'algorithme glouton, les décisions sont prises à partir du domaine de solution donné. En tant qu'être gourmand, la solution la plus proche qui semble fournir une solution optimale est choisie.

Les algorithmes gourmands tentent de trouver une solution optimale localisée, qui peut éventuellement conduire à des solutions optimisées globalement. Cependant, les algorithmes généralement gourmands ne fournissent pas de solutions globalement optimisées.

Compter les pièces

Ce problème consiste à compter jusqu'à une valeur souhaitée en choisissant le moins de pièces possible et l'approche gourmande oblige l'algorithme à choisir la plus grosse pièce possible. Si on nous fournit des pièces de 1, 2, 5 et 10 ₹ et qu'on nous demande de compter 18 ₹, la procédure gourmande sera -

  • 1 - Sélectionnez une pièce de 10 ₹, le nombre restant est de 8

  • 2 - Sélectionnez ensuite une pièce de 5 ₹, le nombre restant est de 3

  • 3 - Sélectionnez ensuite une pièce de 2 ₹, le nombre restant est de 1

  • 4 - Et enfin, la sélection d'une pièce de 1 ₹ résout le problème

Cependant, cela semble fonctionner correctement, pour ce décompte, nous devons choisir seulement 4 pièces. Mais si nous modifions légèrement le problème, la même approche peut ne pas être en mesure de produire le même résultat optimal.

Pour le système monétaire, où nous avons des pièces d'une valeur de 1, 7, 10, compter les pièces pour la valeur 18 sera absolument optimal, mais pour un compte comme 15, il peut utiliser plus de pièces que nécessaire. Par exemple, l'approche gourmande utilisera 10 + 1 + 1 + 1 + 1 + 1, totalisant 6 pièces. Alors que le même problème pourrait être résolu en utilisant seulement 3 pièces (7 + 7 + 1)

Par conséquent, nous pouvons conclure que l'approche gourmande choisit une solution optimisée immédiate et peut échouer là où l'optimisation globale est une préoccupation majeure.

Exemples

La plupart des algorithmes de mise en réseau utilisent l'approche gourmande. Voici une liste de quelques-uns d'entre eux -

  • Problème de vendeur itinérant
  • Algorithme d'arbre couvrant minimal de Prim
  • Algorithme d'arbre couvrant minimal de Kruskal
  • Algorithme d'arbre couvrant minimal de Dijkstra
  • Graphique - Coloration de la carte
  • Graphique - Couverture de sommet
  • Problème de sac à dos
  • Problème de planification des travaux

Il existe de nombreux problèmes similaires qui utilisent l'approche gourmande pour trouver une solution optimale.

Dans l'approche de division pour conquérir, le problème en cours est divisé en sous-problèmes plus petits, puis chaque problème est résolu indépendamment. Lorsque nous continuons à diviser les sous-problèmes en sous-problèmes encore plus petits, nous pouvons éventuellement atteindre un stade où plus aucune division n'est possible. Ces plus petits sous-problèmes "atomiques" (fractions) sont résolus. La solution de tous les sous-problèmes est finalement fusionnée afin d'obtenir la solution d'un problème original.

Globalement, nous pouvons comprendre divide-and-conquer approche dans un processus en trois étapes.

Diviser / Pause

Cette étape consiste à diviser le problème en sous-problèmes plus petits. Les sous-problèmes doivent représenter une partie du problème d'origine. Cette étape adopte généralement une approche récursive pour diviser le problème jusqu'à ce qu'aucun sous-problème ne soit davantage divisible. À ce stade, les sous-problèmes deviennent de nature atomique mais représentent encore une partie du problème réel.

Conquérir / Résoudre

Cette étape reçoit beaucoup de sous-problèmes plus petits à résoudre. Généralement, à ce niveau, les problèmes sont considérés comme «résolus» d'eux-mêmes.

Fusionner / Combiner

Lorsque les sous-problèmes plus petits sont résolus, cette étape les combine de manière récursive jusqu'à ce qu'ils formulent une solution du problème d'origine. Cette approche algorithmique fonctionne de manière récursive et les étapes de conquête et de fusion fonctionnent si étroitement qu'elles apparaissent comme une seule.

Exemples

Les algorithmes informatiques suivants sont basés sur divide-and-conquer approche de programmation -

  • Tri par fusion
  • Tri rapide
  • Recherche binaire
  • Multiplication matricielle de Strassen
  • Paire la plus proche (points)

Il existe différentes façons de résoudre n'importe quel problème informatique, mais celles mentionnées sont un bon exemple d'approche diviser pour conquérir.

L'approche de programmation dynamique est similaire à diviser pour vaincre en décomposant le problème en sous-problèmes possibles plus petits et plus petits. Mais contrairement à, diviser pour conquérir, ces sous-problèmes ne sont pas résolus indépendamment. Au contraire, les résultats de ces sous-problèmes plus petits sont mémorisés et utilisés pour des sous-problèmes similaires ou se chevauchant.

La programmation dynamique est utilisée là où nous avons des problèmes, qui peuvent être divisés en sous-problèmes similaires, afin que leurs résultats puissent être réutilisés. La plupart du temps, ces algorithmes sont utilisés pour l'optimisation. Avant de résoudre le sous-problème en cours, l'algorithme dynamique essaiera d'examiner les résultats des sous-problèmes précédemment résolus. Les solutions des sous-problèmes sont combinées afin d'obtenir la meilleure solution.

Nous pouvons donc dire que -

  • Le problème devrait pouvoir être divisé en sous-problèmes plus petits qui se chevauchent.

  • Une solution optimale peut être obtenue en utilisant une solution optimale de sous-problèmes plus petits.

  • Les algorithmes dynamiques utilisent la mémorisation.

Comparaison

Contrairement aux algorithmes gourmands, où l'optimisation locale est abordée, les algorithmes dynamiques sont motivés pour une optimisation globale du problème.

Contrairement aux algorithmes de division et de conquête, où les solutions sont combinées pour obtenir une solution globale, les algorithmes dynamiques utilisent la sortie d'un sous-problème plus petit, puis tentent d'optimiser un sous-problème plus grand. Les algorithmes dynamiques utilisent la mémorisation pour se souvenir de la sortie des sous-problèmes déjà résolus.

Exemple

Les problèmes informatiques suivants peuvent être résolus en utilisant une approche de programmation dynamique -

  • Série de numéros de Fibonacci
  • Problème de sac à dos
  • La tour de Hanoi
  • Chemin le plus court de toutes les paires par Floyd-Warshall
  • Chemin le plus court par Dijkstra
  • Planification du projet

La programmation dynamique peut être utilisée de manière descendante et ascendante. Et bien sûr, la plupart du temps, se référer à la sortie de la solution précédente est moins cher que de recalculer en termes de cycles CPU.

Ce chapitre explique les termes de base liés à la structure des données.

Définition des données

La définition de données définit une donnée particulière avec les caractéristiques suivantes.

  • Atomic - La définition doit définir un concept unique.

  • Traceable - La définition doit pouvoir être mappée à certains éléments de données.

  • Accurate - La définition doit être sans ambiguïté.

  • Clear and Concise - La définition doit être compréhensible.

Objet de données

L'objet de données représente un objet ayant une donnée.

Type de données

Le type de données est un moyen de classer divers types de données tels que des entiers, des chaînes, etc. qui détermine les valeurs qui peuvent être utilisées avec le type de données correspondant, le type d'opérations qui peuvent être effectuées sur le type de données correspondant. Il existe deux types de données -

  • Type de données intégré
  • Type de données dérivé

Type de données intégré

Les types de données pour lesquels un langage a une prise en charge intégrée sont appelés types de données intégrés. Par exemple, la plupart des langages fournissent les types de données intégrés suivants.

  • Integers
  • Booléen (vrai, faux)
  • Flottant (nombres décimaux)
  • Caractère et chaînes

Type de données dérivé

Les types de données qui sont indépendants de l'implémentation car ils peuvent être implémentés de l'une ou l'autre manière sont appelés types de données dérivés. Ces types de données sont normalement générés par la combinaison de types de données primaires ou intégrés et d'opérations associées sur eux. Par exemple -

  • List
  • Array
  • Stack
  • Queue

Opérations de base

Les données des structures de données sont traitées par certaines opérations. La structure de données particulière choisie dépend largement de la fréquence de l'opération qui doit être effectuée sur la structure de données.

  • Traversing
  • Searching
  • Insertion
  • Deletion
  • Sorting
  • Merging

Array est un conteneur qui peut contenir un nombre fixe d'éléments et ces éléments doivent être du même type. La plupart des structures de données utilisent des tableaux pour implémenter leurs algorithmes. Voici les termes importants pour comprendre le concept de Array.

  • Element - Chaque élément stocké dans un tableau est appelé un élément.

  • Index - Chaque emplacement d'un élément dans un tableau a un index numérique, qui est utilisé pour identifier l'élément.

Représentation du tableau

Les tableaux peuvent être déclarés de différentes manières dans différentes langues. Par exemple, prenons la déclaration de tableau C.

Les tableaux peuvent être déclarés de différentes manières dans différentes langues. Par exemple, prenons la déclaration de tableau C.

Conformément à l'illustration ci-dessus, voici les points importants à considérer.

  • L'index commence par 0.

  • La longueur du tableau est de 10, ce qui signifie qu'il peut stocker 10 éléments.

  • Chaque élément est accessible via son index. Par exemple, nous pouvons récupérer un élément à l'index 6 comme 9.

Opérations de base

Voici les opérations de base prises en charge par une baie.

  • Traverse - imprimer tous les éléments du tableau un par un.

  • Insertion - Ajoute un élément à l'index donné.

  • Deletion - Supprime un élément à l'index donné.

  • Search - Recherche un élément en utilisant l'index donné ou par la valeur.

  • Update - Met à jour un élément à l'index donné.

En C, lorsqu'un tableau est initialisé avec size, il attribue des valeurs par défaut à ses éléments dans l'ordre suivant.

Type de données Valeur par défaut
booléen faux
carboniser 0
int 0
flotte 0,0
double 0,0f
néant
wchar_t 0

Fonctionnement de la traversée

Cette opération consiste à parcourir les éléments d'un tableau.

Exemple

Le programme suivant parcourt et imprime les éléments d'un tableau:

#include <stdio.h>
main() {
   int LA[] = {1,3,5,7,8};
   int item = 10, k = 3, n = 5;
   int i = 0, j = n;   
   printf("The original array elements are :\n");
   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }
}

Lorsque nous compilons et exécutons le programme ci-dessus, il produit le résultat suivant -

Production

The original array elements are :
LA[0] = 1 
LA[1] = 3 
LA[2] = 5 
LA[3] = 7 
LA[4] = 8

Opération d'insertion

L'opération d'insertion consiste à insérer un ou plusieurs éléments de données dans un tableau. En fonction de l'exigence, un nouvel élément peut être ajouté au début, à la fin ou à tout index donné du tableau.

Ici, nous voyons une implémentation pratique de l'opération d'insertion, où nous ajoutons des données à la fin du tableau -

Exemple

Voici la mise en œuvre de l'algorithme ci-dessus -

#include <stdio.h>

main() {
   int LA[] = {1,3,5,7,8};
   int item = 10, k = 3, n = 5;
   int i = 0, j = n;
   
   printf("The original array elements are :\n");

   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }

   n = n + 1;
	
   while( j >= k) {
      LA[j+1] = LA[j];
      j = j - 1;
   }

   LA[k] = item;

   printf("The array elements after insertion :\n");

   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }
}

Lorsque nous compilons et exécutons le programme ci-dessus, il produit le résultat suivant -

Production

The original array elements are :
LA[0] = 1 
LA[1] = 3 
LA[2] = 5 
LA[3] = 7 
LA[4] = 8 
The array elements after insertion :
LA[0] = 1 
LA[1] = 3 
LA[2] = 5 
LA[3] = 10 
LA[4] = 7 
LA[5] = 8

Pour d'autres variantes d'opération d'insertion de tableau, cliquez ici

Opération de suppression

La suppression fait référence à la suppression d'un élément existant du tableau et à la réorganisation de tous les éléments d'un tableau.

Algorithme

Considérer LA est un tableau linéaire avec N éléments et K est un entier positif tel que K<=N. Voici l'algorithme pour supprimer un élément disponible à la K e position de LA.

1. Start
2. Set J = K
3. Repeat steps 4 and 5 while J < N
4. Set LA[J] = LA[J + 1]
5. Set J = J+1
6. Set N = N-1
7. Stop

Exemple

Voici la mise en œuvre de l'algorithme ci-dessus -

#include <stdio.h>

void main() {
   int LA[] = {1,3,5,7,8};
   int k = 3, n = 5;
   int i, j;
   
   printf("The original array elements are :\n");
	
   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }
    
   j = k;
	
   while( j < n) {
      LA[j-1] = LA[j];
      j = j + 1;
   }
	
   n = n -1;
   
   printf("The array elements after deletion :\n");
	
   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }
}

Lorsque nous compilons et exécutons le programme ci-dessus, il produit le résultat suivant -

Production

The original array elements are :
LA[0] = 1 
LA[1] = 3 
LA[2] = 5 
LA[3] = 7 
LA[4] = 8 
The array elements after deletion :
LA[0] = 1 
LA[1] = 3 
LA[2] = 7 
LA[3] = 8

Opération de recherche

Vous pouvez rechercher un élément de tableau en fonction de sa valeur ou de son index.

Algorithme

Considérer LA est un tableau linéaire avec N éléments et K est un entier positif tel que K<=N. Voici l'algorithme pour trouver un élément avec une valeur de ITEM en utilisant la recherche séquentielle.

1. Start
2. Set J = 0
3. Repeat steps 4 and 5 while J < N
4. IF LA[J] is equal ITEM THEN GOTO STEP 6
5. Set J = J +1
6. PRINT J, ITEM
7. Stop

Exemple

Voici la mise en œuvre de l'algorithme ci-dessus -

#include <stdio.h>

void main() {
   int LA[] = {1,3,5,7,8};
   int item = 5, n = 5;
   int i = 0, j = 0;
   
   printf("The original array elements are :\n");
	
   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }
    
   while( j < n){
      if( LA[j] == item ) {
         break;
      }
		
      j = j + 1;
   }
	
   printf("Found element %d at position %d\n", item, j+1);
}

Lorsque nous compilons et exécutons le programme ci-dessus, il produit le résultat suivant -

Production

The original array elements are :
LA[0] = 1 
LA[1] = 3 
LA[2] = 5 
LA[3] = 7 
LA[4] = 8 
Found element 5 at position 3

Opération de mise à jour

L'opération de mise à jour fait référence à la mise à jour d'un élément existant du tableau à un index donné.

Algorithme

Considérer LA est un tableau linéaire avec N éléments et K est un entier positif tel que K<=N. Voici l'algorithme pour mettre à jour un élément disponible à la K e position de LA.

1. Start
2. Set LA[K-1] = ITEM
3. Stop

Exemple

Voici la mise en œuvre de l'algorithme ci-dessus -

#include <stdio.h>

void main() {
   int LA[] = {1,3,5,7,8};
   int k = 3, n = 5, item = 10;
   int i, j;
   
   printf("The original array elements are :\n");
	
   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }
    
   LA[k-1] = item;

   printf("The array elements after updation :\n");
	
   for(i = 0; i<n; i++) {
      printf("LA[%d] = %d \n", i, LA[i]);
   }
}

Lorsque nous compilons et exécutons le programme ci-dessus, il produit le résultat suivant -

Production

The original array elements are :
LA[0] = 1 
LA[1] = 3 
LA[2] = 5 
LA[3] = 7 
LA[4] = 8 
The array elements after updation :
LA[0] = 1 
LA[1] = 3 
LA[2] = 10 
LA[3] = 7 
LA[4] = 8

Une liste chaînée est une séquence de structures de données, qui sont connectées entre elles via des liens.

La liste liée est une séquence de liens contenant des éléments. Chaque lien contient une connexion à un autre lien. La liste liée est la deuxième structure de données la plus utilisée après le tableau. Voici les termes importants pour comprendre le concept de liste liée.

  • Link - Chaque lien d'une liste liée peut stocker une donnée appelée élément.

  • Next - Chaque lien d'une liste liée contient un lien vers le lien suivant appelé Suivant.

  • LinkedList - Une liste liée contient le lien de connexion vers le premier lien appelé Premier.

Représentation de liste liée

La liste liée peut être visualisée comme une chaîne de nœuds, où chaque nœud pointe vers le nœud suivant.

Conformément à l'illustration ci-dessus, voici les points importants à considérer.

  • La liste liée contient un élément de lien appelé en premier.

  • Chaque lien porte un (des) champ (s) de données et un champ de lien appelé ensuite.

  • Chaque lien est lié à son lien suivant en utilisant son lien suivant.

  • Le dernier lien porte un lien nul pour marquer la fin de la liste.

Types de liste liée

Voici les différents types de liste chaînée.

  • Simple Linked List - La navigation des articles est uniquement en avant.

  • Doubly Linked List - Les éléments peuvent être parcourus en avant et en arrière.

  • Circular Linked List - Le dernier élément contient le lien du premier élément comme suivant et le premier élément a un lien vers le dernier élément comme précédent.

Opérations de base

Voici les opérations de base prises en charge par une liste.

  • Insertion - Ajoute un élément au début de la liste.

  • Deletion - Supprime un élément au début de la liste.

  • Display - Affiche la liste complète.

  • Search - Recherche un élément en utilisant la clé donnée.

  • Delete - Supprime un élément en utilisant la clé donnée.

Opération d'insertion

L'ajout d'un nouveau nœud dans une liste liée est une activité en plusieurs étapes. Nous allons apprendre cela avec des schémas ici. Tout d'abord, créez un nœud en utilisant la même structure et trouvez l'emplacement où il doit être inséré.

Imaginez que nous insérons un nœud B (NewNode), entre A (LeftNode) et C(RightNode). Puis pointez B suivant vers C -

NewNode.next −> RightNode;

Cela devrait ressembler à ceci -

Maintenant, le nœud suivant à gauche doit pointer vers le nouveau nœud.

LeftNode.next −> NewNode;

Cela placera le nouveau nœud au milieu des deux. La nouvelle liste devrait ressembler à ceci -

Des étapes similaires doivent être prises si le nœud est inséré au début de la liste. Lors de son insertion à la fin, l'avant-dernier nœud de la liste doit pointer vers le nouveau nœud et le nouveau nœud pointera vers NULL.

Opération de suppression

La suppression est également un processus en plusieurs étapes. Nous apprendrons avec la représentation picturale. Tout d'abord, localisez le nœud cible à supprimer à l'aide d'algorithmes de recherche.

Le nœud gauche (précédent) du nœud cible doit maintenant pointer vers le nœud suivant du nœud cible -

LeftNode.next −> TargetNode.next;

Cela supprimera le lien qui pointait vers le nœud cible. Maintenant, en utilisant le code suivant, nous allons supprimer ce sur quoi le nœud cible pointe.

TargetNode.next −> NULL;

Nous devons utiliser le nœud supprimé. Nous pouvons garder cela en mémoire, sinon nous pouvons simplement désallouer de la mémoire et effacer complètement le nœud cible.

Opération inverse

Cette opération est approfondie. Nous devons faire en sorte que le dernier nœud soit pointé par le nœud principal et inverser toute la liste chaînée.

Tout d'abord, nous allons à la fin de la liste. Il doit pointer vers NULL. Maintenant, nous allons le faire pointer vers son nœud précédent -

Nous devons nous assurer que le dernier nœud n'est pas le dernier. Nous aurons donc un nœud temporaire, qui ressemble au nœud principal pointant vers le dernier nœud. Maintenant, nous allons faire pointer tous les nœuds du côté gauche vers leurs nœuds précédents un par un.

À l'exception du nœud (premier nœud) pointé par le nœud principal, tous les nœuds doivent pointer vers leur prédécesseur, ce qui en fait leur nouveau successeur. Le premier nœud pointera vers NULL.

Nous allons faire pointer le nœud principal vers le nouveau premier nœud en utilisant le nœud temporaire.

La liste chaînée est maintenant inversée. Pour voir l'implémentation des listes liées dans le langage de programmation C, veuillez cliquer ici .

La liste double liée est une variante de la liste liée dans laquelle la navigation est possible dans les deux sens, en avant et en arrière facilement par rapport à la liste liée unique. Voici les termes importants pour comprendre le concept de liste à double chaînage.

  • Link - Chaque lien d'une liste liée peut stocker une donnée appelée élément.

  • Next - Chaque lien d'une liste liée contient un lien vers le lien suivant appelé Suivant.

  • Prev - Chaque lien d'une liste liée contient un lien vers le lien précédent appelé Prev.

  • LinkedList - Une liste liée contient le lien de connexion vers le premier lien appelé Premier et vers le dernier lien appelé Dernier.

Représentation de liste doublement liée

Conformément à l'illustration ci-dessus, voici les points importants à considérer.

  • La liste doublement liée contient un élément de lien appelé premier et dernier.

  • Chaque lien porte un (des) champ (s) de données et deux champs de lien appelés next et prev.

  • Chaque lien est lié à son lien suivant en utilisant son lien suivant.

  • Chaque lien est lié à son lien précédent en utilisant son lien précédent.

  • Le dernier lien porte un lien nul pour marquer la fin de la liste.

Opérations de base

Voici les opérations de base prises en charge par une liste.

  • Insertion - Ajoute un élément au début de la liste.

  • Deletion - Supprime un élément au début de la liste.

  • Insert Last - Ajoute un élément à la fin de la liste.

  • Delete Last - Supprime un élément de la fin de la liste.

  • Insert After - Ajoute un élément après un élément de la liste.

  • Delete - Supprime un élément de la liste à l'aide de la touche.

  • Display forward - Affiche la liste complète d'une manière avant.

  • Display backward - Affiche la liste complète à l'envers.

Opération d'insertion

Le code suivant illustre l'opération d'insertion au début d'une liste doublement liée.

Exemple

//insert link at the first location
void insertFirst(int key, int data) {

   //create a link
   struct node *link = (struct node*) malloc(sizeof(struct node));
   link->key = key;
   link->data = data;
	
   if(isEmpty()) {
      //make it the last link
      last = link;
   } else {
      //update first prev link
      head->prev = link;
   }

   //point it to old first link
   link->next = head;
	
   //point first to new first link
   head = link;
}

Opération de suppression

Le code suivant illustre l'opération de suppression au début d'une liste doublement liée.

Exemple

//delete first item
struct node* deleteFirst() {

   //save reference to first link
   struct node *tempLink = head;
	
   //if only one link
   if(head->next == NULL) {
      last = NULL;
   } else {
      head->next->prev = NULL;
   }
	
   head = head->next;
	
   //return the deleted link
   return tempLink;
}

Insertion à la fin d'une opération

Le code suivant illustre l'opération d'insertion à la dernière position d'une liste doublement liée.

Exemple

//insert link at the last location
void insertLast(int key, int data) {

   //create a link
   struct node *link = (struct node*) malloc(sizeof(struct node));
   link->key = key;
   link->data = data;
	
   if(isEmpty()) {
      //make it the last link
      last = link;
   } else {
      //make link a new last link
      last->next = link;     
      
      //mark old last node as prev of new link
      link->prev = last;
   }

   //point last to new last node
   last = link;
}

Pour voir l'implémentation en langage de programmation C, veuillez cliquer ici .

La liste liée circulaire est une variante de la liste liée dans laquelle le premier élément pointe vers le dernier élément et le dernier élément pointe vers le premier élément. La liste à liaison unique et la liste à liaison double peuvent être transformées en une liste liée circulaire.

Liste à lien unique en tant que circulaire

Dans une liste à liaison unique, le pointeur suivant du dernier nœud pointe vers le premier nœud.

Liste doublement liée en tant que circulaire

Dans une liste doublement liée, le pointeur suivant du dernier nœud pointe vers le premier nœud et le pointeur précédent du premier nœud pointe vers le dernier nœud faisant la circulaire dans les deux sens.

Conformément à l'illustration ci-dessus, voici les points importants à considérer.

  • Le prochain lien du dernier lien pointe vers le premier lien de la liste dans les deux cas de liste simple ou double.

  • Le précédent du premier lien pointe vers le dernier de la liste en cas de liste doublement liée.

Opérations de base

Voici les opérations importantes soutenues par une liste circulaire.

  • insert - Insère un élément au début de la liste.

  • delete - Supprime un élément du début de la liste.

  • display - Affiche la liste.

Opération d'insertion

Le code suivant illustre l'opération d'insertion dans une liste liée circulaire basée sur une seule liste liée.

Exemple

//insert link at the first location
void insertFirst(int key, int data) {
   //create a link
   struct node *link = (struct node*) malloc(sizeof(struct node));
   link->key = key;
   link->data= data;
	
   if (isEmpty()) {
      head = link;
      head->next = head;
   } else {
      //point it to old first node
      link->next = head;
		
      //point first to new first node
      head = link;
   }   
}

Opération de suppression

Le code suivant illustre l'opération de suppression dans une liste liée circulaire basée sur une seule liste liée.

//delete first item
struct node * deleteFirst() {
   //save reference to first link
   struct node *tempLink = head;
	
   if(head->next == head) {  
      head = NULL;
      return tempLink;
   }     

   //mark next to first link as first 
   head = head->next;
	
   //return the deleted link
   return tempLink;
}

Fonctionnement de la liste d'affichage

Le code suivant illustre l'opération d'affichage de la liste dans une liste liée circulaire.

//display the list
void printList() {
   struct node *ptr = head;
   printf("\n[ ");
	
   //start from the beginning
   if(head != NULL) {
      while(ptr->next != ptr) {     
         printf("(%d,%d) ",ptr->key,ptr->data);
         ptr = ptr->next;
      }
   }
	
   printf(" ]");
}

Pour connaître son implémentation en langage de programmation C, veuillez cliquer ici .

Une pile est un type de données abstrait (ADT), couramment utilisé dans la plupart des langages de programmation. Il est nommé pile car il se comporte comme une pile du monde réel, par exemple - un jeu de cartes ou une pile de plaques, etc.

Une pile du monde réel permet les opérations à une seule extrémité. Par exemple, nous pouvons placer ou retirer une carte ou une plaque du haut de la pile uniquement. De même, Stack ADT autorise toutes les opérations de données à une seule extrémité. À tout moment, nous ne pouvons accéder qu'à l'élément supérieur d'une pile.

Cette caractéristique en fait une structure de données LIFO. LIFO signifie Last-in-first-out. Ici, on accède en premier à l'élément placé (inséré ou ajouté) en dernier. Dans la terminologie de la pile, l'opération d'insertion est appeléePUSH l'opération et l'opération de suppression est appelée POP opération.

Représentation de la pile

Le diagramme suivant illustre une pile et ses opérations -

Une pile peut être implémentée au moyen de Array, Structure, Pointer et Linked List. La pile peut être de taille fixe ou avoir un sens de redimensionnement dynamique. Ici, nous allons implémenter la pile à l'aide de tableaux, ce qui en fait une implémentation de pile de taille fixe.

Opérations de base

Les opérations de pile peuvent impliquer l'initialisation de la pile, son utilisation, puis sa désinitialisation. En dehors de ces éléments de base, une pile est utilisée pour les deux opérations principales suivantes -

  • push() - Pousser (stocker) un élément sur la pile.

  • pop() - Suppression (accès) à un élément de la pile.

Lorsque les données sont PUSHed sur la pile.

Pour utiliser efficacement une pile, nous devons également vérifier l'état de la pile. Dans le même but, la fonctionnalité suivante est ajoutée aux piles -

  • peek() - récupère l'élément de données supérieur de la pile, sans le supprimer.

  • isFull() - vérifier si la pile est pleine.

  • isEmpty() - vérifier si la pile est vide.

À tout moment, nous maintenons un pointeur vers les dernières données PUSHed de la pile. Comme ce pointeur représente toujours le haut de la pile, donc nommétop. letop pointer fournit la valeur supérieure de la pile sans la supprimer.

Nous devons d'abord en apprendre davantage sur les procédures de prise en charge des fonctions de pile -

coup d'oeil ()

Algorithme de la fonction peek () -

begin procedure peek
   return stack[top]
end procedure

Implémentation de la fonction peek () en langage de programmation C -

Example

int peek() {
   return stack[top];
}

est rempli()

Algorithme de la fonction isfull () -

begin procedure isfull

   if top equals to MAXSIZE
      return true
   else
      return false
   endif
   
end procedure

Implémentation de la fonction isfull () en langage de programmation C -

Example

bool isfull() {
   if(top == MAXSIZE)
      return true;
   else
      return false;
}

est vide()

Algorithme de la fonction isempty () -

begin procedure isempty

   if top less than 1
      return true
   else
      return false
   endif
   
end procedure

L'implémentation de la fonction isempty () dans le langage de programmation C est légèrement différente. Nous initialisons top à -1, car l'index du tableau commence à 0. Nous vérifions donc si le haut est en dessous de zéro ou -1 pour déterminer si la pile est vide. Voici le code -

Example

bool isempty() {
   if(top == -1)
      return true;
   else
      return false;
}

Opération Push

Le processus de mise en pile d'un nouvel élément de données est appelé opération Push. L'opération de poussée implique une série d'étapes -

  • Step 1 - Vérifie si la pile est pleine.

  • Step 2 - Si la pile est pleine, produit une erreur et quitte.

  • Step 3 - Si la pile n'est pas pleine, incrémente top pour pointer le prochain espace vide.

  • Step 4 - Ajoute un élément de données à l'emplacement de la pile, où le haut pointe.

  • Step 5 - Renvoie le succès.

Si la liste liée est utilisée pour implémenter la pile, à l'étape 3, nous devons allouer de l'espace de manière dynamique.

Algorithme pour l'opération PUSH

Un algorithme simple pour l'opération Push peut être dérivé comme suit -

begin procedure push: stack, data

   if stack is full
      return null
   endif
   
   top ← top + 1
   stack[top] ← data

end procedure

L'implémentation de cet algorithme en C, est très simple. Voir le code suivant -

Example

void push(int data) {
   if(!isFull()) {
      top = top + 1;   
      stack[top] = data;
   } else {
      printf("Could not insert data, Stack is full.\n");
   }
}

Opération Pop

L'accès au contenu tout en le supprimant de la pile est appelé opération Pop. Dans une implémentation de tableau de l'opération pop (), l'élément de données n'est pas réellement supprimé, à la placetopest décrémenté à une position inférieure dans la pile pour pointer vers la valeur suivante. Mais dans l'implémentation de liste chaînée, pop () supprime en fait l'élément de données et libère l'espace mémoire.

Une opération Pop peut impliquer les étapes suivantes -

  • Step 1 - Vérifie si la pile est vide.

  • Step 2 - Si la pile est vide, produit une erreur et quitte.

  • Step 3 - Si la pile n'est pas vide, accède à l'élément de données auquel top pointe.

  • Step 4 - Diminue la valeur de top de 1.

  • Step 5 - Renvoie le succès.

Algorithme pour l'opération Pop

Un algorithme simple pour l'opération Pop peut être dérivé comme suit -

begin procedure pop: stack

   if stack is empty
      return null
   endif
   
   data ← stack[top]
   top ← top - 1
   return data

end procedure

La mise en œuvre de cet algorithme en C, est la suivante -

Example

int pop(int data) {

   if(!isempty()) {
      data = stack[top];
      top = top - 1;   
      return data;
   } else {
      printf("Could not retrieve data, Stack is empty.\n");
   }
}

Pour un programme complet de pile en langage de programmation C, veuillez cliquer ici .

La façon d'écrire une expression arithmétique est connue sous le nom de notation. Une expression arithmétique peut être écrite dans trois notations différentes mais équivalentes, c'est-à-dire sans changer l'essence ou la sortie d'une expression. Ces notations sont -

  • Notation infixe
  • Notation de préfixe (polonais)
  • Notation Postfix (polonaise inversée)

Ces notations sont nommées selon la manière dont elles utilisent l'opérateur dans l'expression. Nous apprendrons la même chose ici dans ce chapitre.

Notation infixe

Nous écrivons l'expression dans infix notation, par exemple a - b + c, où les opérateurs sont utilisés in-entre les opérandes. Il est facile pour nous, humains, de lire, d'écrire et de parler en notation infixe, mais la même chose ne va pas bien avec les appareils informatiques. Un algorithme pour traiter la notation infixe pourrait être difficile et coûteux en termes de consommation de temps et d'espace.

Notation de préfixe

Dans cette notation, l'opérateur est prefixed aux opérandes, c'est-à-dire que l'opérateur est écrit avant les opérandes. Par exemple,+ab. Ceci est équivalent à sa notation infixea + b. La notation de préfixe est également connue sous le nom dePolish Notation.

Notation Postfix

Ce style de notation est connu sous le nom de Reversed Polish Notation. Dans ce style de notation, l'opérateur estpostfixed aux opérandes, c'est-à-dire que l'opérateur est écrit après les opérandes. Par exemple,ab+. Ceci est équivalent à sa notation infixea + b.

Le tableau suivant essaie brièvement de montrer la différence entre les trois notations -

Sr.No. Notation infixe Notation de préfixe Notation Postfix
1 a + b + ab ab +
2 (a + b) ∗ c ∗ + abc ab + c ∗
3 a ∗ (b + c) ∗ a + bc abc + ∗
4 a / b + c / d + / ab / cd ab / cd / +
5 (a + b) ∗ (c + d) ∗ + ab + cd ab + cd + ∗
6 ((a + b) ∗ c) - d - ∗ + abcd ab + c ∗ d -

Analyse des expressions

Comme nous l'avons vu, ce n'est pas un moyen très efficace de concevoir un algorithme ou un programme pour analyser les notations d'infixe. Au lieu de cela, ces notations d'infixe sont d'abord converties en notations de suffixe ou de préfixe, puis calculées.

Pour analyser une expression arithmétique, nous devons également prendre en compte la priorité des opérateurs et l'associativité.

Priorité

Lorsqu'un opérande est entre deux opérateurs différents, quel opérateur prendra l'opérande en premier, est décidé par la priorité d'un opérateur sur les autres. Par exemple -

Comme l'opération de multiplication a priorité sur l'addition, b * c sera évalué en premier. Un tableau de priorité des opérateurs est fourni ultérieurement.

Associativité

L'associativité décrit la règle dans laquelle les opérateurs de même priorité apparaissent dans une expression. Par exemple, dans l'expression a + b - c, + et - ont la même priorité, alors quelle partie de l'expression sera évaluée en premier est déterminée par l'associativité de ces opérateurs. Ici, + et - sont laissés associatifs, donc l'expression sera évaluée comme(a + b) − c.

La préséance et l'associativité déterminent l'ordre d'évaluation d'une expression. Voici une table de priorité des opérateurs et d'associativité (du plus élevé au plus bas) -

Sr.No. Opérateur Priorité Associativité
1 Exponentiation ^ Le plus élevé Associatif droit
2 Multiplication (∗) & Division (/) Deuxième plus haut Associatif gauche
3 Addition (+) et soustraction (-) Le plus bas Associatif gauche

Le tableau ci-dessus montre le comportement par défaut des opérateurs. À tout moment de l'évaluation de l'expression, l'ordre peut être modifié à l'aide de parenthèses. Par exemple -

Dans a + b*c, la partie expression b*csera évalué en premier, avec la multiplication comme priorité sur l'addition. Nous utilisons ici des parenthèses poura + b à évaluer en premier, comme (a + b)*c.

Algorithme d'évaluation de Postfix

Nous allons maintenant examiner l'algorithme sur la façon d'évaluer la notation postfixe -

Step 1 − scan the expression from left to right 
Step 2 − if it is an operand push it to stack 
Step 3 − if it is an operator pull operand from stack and perform operation 
Step 4 − store the output of step 3, back to stack 
Step 5 − scan the expression until all operands are consumed 
Step 6 − pop the stack and perform operation

Pour voir l'implémentation en langage de programmation C, veuillez cliquer ici .

Queue est une structure de données abstraite, quelque peu similaire à Stacks. Contrairement aux piles, une file d'attente est ouverte à ses deux extrémités. Une extrémité est toujours utilisée pour insérer des données (mise en file d'attente) et l'autre est utilisée pour supprimer des données (retrait de la file d'attente). La file d'attente suit la méthodologie du premier entré, premier sorti, c'est-à-dire que l'élément de données stocké en premier sera consulté en premier.

Un exemple concret de file d'attente peut être une route à sens unique à voie unique, où le véhicule entre en premier, sort en premier. D'autres exemples concrets peuvent être considérés comme des files d'attente aux guichets et aux arrêts de bus.

Représentation de la file d'attente

Comme nous comprenons maintenant que dans la file d'attente, nous accédons aux deux extrémités pour des raisons différentes. Le diagramme suivant donné ci-dessous tente d'expliquer la représentation de la file d'attente sous forme de structure de données -

Comme dans les piles, une file d'attente peut également être implémentée à l'aide de tableaux, de listes liées, de pointeurs et de structures. Par souci de simplicité, nous allons implémenter des files d'attente en utilisant un tableau unidimensionnel.

Opérations de base

Les opérations de file d'attente peuvent impliquer l'initialisation ou la définition de la file d'attente, son utilisation, puis son effacement complet de la mémoire. Ici, nous allons essayer de comprendre les opérations de base associées aux files d'attente -

  • enqueue() - ajouter (stocker) un élément à la file d'attente.

  • dequeue() - supprimer (accéder) un élément de la file d'attente.

Peu de fonctions supplémentaires sont nécessaires pour rendre efficace l'opération de file d'attente mentionnée ci-dessus. Ce sont -

  • peek() - Obtient l'élément au début de la file d'attente sans le supprimer.

  • isfull() - Vérifie si la file d'attente est pleine.

  • isempty() - Vérifie si la file d'attente est vide.

En file d'attente, nous retirons (ou accédons) toujours aux données, pointées par front pointeur et en enregistrant (ou en stockant) des données dans la file d'attente, nous prenons l'aide de rear aiguille.

Découvrons d'abord les fonctions de support d'une file d'attente -

coup d'oeil ()

Cette fonction permet de voir les données au frontde la file d'attente. L'algorithme de la fonction peek () est le suivant -

Algorithm

begin procedure peek
   return queue[front]
end procedure

Implémentation de la fonction peek () en langage de programmation C -

Example

int peek() {
   return queue[front];
}

est rempli()

Comme nous utilisons un tableau à dimension unique pour implémenter la file d'attente, nous vérifions simplement que le pointeur arrière atteint MAXSIZE pour déterminer que la file d'attente est pleine. Si nous maintenons la file d'attente dans une liste chaînée circulaire, l'algorithme sera différent. Algorithme de la fonction isfull () -

Algorithm

begin procedure isfull

   if rear equals to MAXSIZE
      return true
   else
      return false
   endif
   
end procedure

Implémentation de la fonction isfull () en langage de programmation C -

Example

bool isfull() {
   if(rear == MAXSIZE - 1)
      return true;
   else
      return false;
}

est vide()

Algorithme de la fonction isempty () -

Algorithm

begin procedure isempty

   if front is less than MIN  OR front is greater than rear
      return true
   else
      return false
   endif
   
end procedure

Si la valeur de front est inférieur à MIN ou 0, il indique que la file d'attente n'est pas encore initialisée, donc vide.

Voici le code de programmation C -

Example

bool isempty() {
   if(front < 0 || front > rear) 
      return true;
   else
      return false;
}

Opération de mise en file d'attente

Les files d'attente conservent deux pointeurs de données, front et rear. Par conséquent, ses opérations sont comparativement difficiles à mettre en œuvre que celles des piles.

Les étapes suivantes doivent être prises pour mettre en file d'attente (insérer) des données dans une file d'attente -

  • Step 1 - Vérifiez si la file d'attente est pleine.

  • Step 2 - Si la file d'attente est pleine, générer une erreur de dépassement de capacité et quitter.

  • Step 3 - Si la file d'attente n'est pas pleine, incrémentez rear pointeur pour pointer le prochain espace vide.

  • Step 4 - Ajoutez un élément de données à l'emplacement de la file d'attente, où pointe l'arrière.

  • Step 5 - retour de succès.

Parfois, nous vérifions également si une file d'attente est initialisée ou non, pour gérer d'éventuelles situations imprévues.

Algorithme pour l'opération de mise en file d'attente

procedure enqueue(data)      
   
   if queue is full
      return overflow
   endif
   
   rear ← rear + 1
   queue[rear] ← data
   return true
   
end procedure

Implémentation de enqueue () en langage de programmation C -

Example

int enqueue(int data)      
   if(isfull())
      return 0;
   
   rear = rear + 1;
   queue[rear] = data;
   
   return 1;
end procedure

Opération de retrait de la file d'attente

L'accès aux données de la file d'attente est un processus de deux tâches: accéder aux données où frontpointe et supprime les données après l'accès. Les étapes suivantes sont prises pour effectuerdequeue opération -

  • Step 1 - Vérifiez si la file d'attente est vide.

  • Step 2 - Si la file d'attente est vide, générer une erreur de sous-dépassement et quitter.

  • Step 3 - Si la file d'attente n'est pas vide, accédez aux données où front pointe.

  • Step 4 - Incrément front pointeur pour pointer vers le prochain élément de données disponible.

  • Step 5 - Retournez le succès.

Algorithme pour l'opération de retrait de la file d'attente

procedure dequeue
   
   if queue is empty
      return underflow
   end if

   data = queue[front]
   front ← front + 1
   return true

end procedure

Implémentation de dequeue () en langage de programmation C -

Example

int dequeue() {
   if(isempty())
      return 0;

   int data = queue[front];
   front = front + 1;

   return data;
}

Pour un programme Queue complet en langage de programmation C, veuillez cliquer ici .

La recherche linéaire est un algorithme de recherche très simple. Dans ce type de recherche, une recherche séquentielle est effectuée sur tous les éléments un par un. Chaque élément est vérifié et si une correspondance est trouvée, cet élément particulier est renvoyé, sinon la recherche se poursuit jusqu'à la fin de la collecte de données.

Algorithme

Linear Search ( Array A, Value x)

Step 1: Set i to 1
Step 2: if i > n then go to step 7
Step 3: if A[i] = x then go to step 6
Step 4: Set i to i + 1
Step 5: Go to Step 2
Step 6: Print Element x Found at index i and go to step 8
Step 7: Print element not found
Step 8: Exit

Pseudocode

procedure linear_search (list, value)

   for each item in the list
      if match item == value
         return the item's location
      end if
   end for

end procedure

Pour en savoir plus sur l'implémentation de la recherche linéaire dans le langage de programmation C, veuillez cliquer ici .

La recherche binaire est un algorithme de recherche rapide avec une complexité d'exécution de Ο (log n). Cet algorithme de recherche fonctionne sur le principe de diviser pour conquérir. Pour que cet algorithme fonctionne correctement, la collecte de données doit être triée.

La recherche binaire recherche un élément particulier en comparant l'élément le plus au milieu de la collection. Si une correspondance se produit, l'index de l'élément est renvoyé. Si l'élément du milieu est supérieur à l'élément, l'élément est recherché dans le sous-tableau à gauche de l'élément du milieu. Sinon, l'élément est recherché dans le sous-tableau à droite de l'élément du milieu. Ce processus se poursuit également sur le sous-tableau jusqu'à ce que la taille du sous-tableau soit réduite à zéro.

Comment fonctionne la recherche binaire?

Pour qu'une recherche binaire fonctionne, il est obligatoire que le tableau cible soit trié. Nous apprendrons le processus de recherche binaire avec un exemple pictural. Ce qui suit est notre tableau trié et supposons que nous devons rechercher l'emplacement de la valeur 31 à l'aide de la recherche binaire.

Tout d'abord, nous allons déterminer la moitié du tableau en utilisant cette formule -

mid = low + (high - low) / 2

Le voici, 0 + (9 - 0) / 2 = 4 (valeur entière de 4,5). Donc, 4 est le milieu du tableau.

Maintenant, nous comparons la valeur stockée à l'emplacement 4, avec la valeur recherchée, c'est-à-dire 31. Nous trouvons que la valeur à l'emplacement 4 est 27, ce qui n'est pas une correspondance. Comme la valeur est supérieure à 27 et que nous avons un tableau trié, nous savons également que la valeur cible doit être dans la partie supérieure du tableau.

Nous changeons notre bas en mid + 1 et retrouvons la nouvelle valeur moyenne.

low = mid + 1
mid = low + (high - low) / 2

Notre nouveau milieu a maintenant 7 ans. Nous comparons la valeur stockée à l'emplacement 7 avec notre valeur cible 31.

La valeur stockée à l'emplacement 7 n'est pas une correspondance, c'est plutôt plus que ce que nous recherchons. Ainsi, la valeur doit être dans la partie inférieure de cet emplacement.

Par conséquent, nous calculons à nouveau le milieu. Cette fois, il est 5.

Nous comparons la valeur stockée à l'emplacement 5 avec notre valeur cible. Nous trouvons que c'est un match.

Nous concluons que la valeur cible 31 est stockée à l'emplacement 5.

La recherche binaire divise par deux les éléments interrogeables et réduit ainsi le nombre de comparaisons à effectuer à un nombre très réduit.

Pseudocode

Le pseudo-code des algorithmes de recherche binaire devrait ressembler à ceci -

Procedure binary_search
   A ← sorted array
   n ← size of array
   x ← value to be searched

   Set lowerBound = 1
   Set upperBound = n 

   while x not found
      if upperBound < lowerBound 
         EXIT: x does not exists.
   
      set midPoint = lowerBound + ( upperBound - lowerBound ) / 2
      
      if A[midPoint] < x
         set lowerBound = midPoint + 1
         
      if A[midPoint] > x
         set upperBound = midPoint - 1 

      if A[midPoint] = x 
         EXIT: x found at location midPoint
   end while
   
end procedure

Pour en savoir plus sur l'implémentation de la recherche binaire à l'aide de array en langage de programmation C, veuillez cliquer ici .

La recherche par interpolation est une variante améliorée de la recherche binaire. Cet algorithme de recherche fonctionne sur la position de palpage de la valeur requise. Pour que cet algorithme fonctionne correctement, la collecte de données doit être triée et répartie de manière égale.

La recherche binaire présente un énorme avantage de complexité temporelle par rapport à la recherche linéaire. La recherche linéaire a une complexité dans le pire des cas de Ο (n) alors que la recherche binaire a Ο (log n).

Il y a des cas où l'emplacement des données cibles peut être connu à l'avance. Par exemple, dans le cas d'un annuaire téléphonique, si nous voulons rechercher le numéro de téléphone de Morphius. Ici, la recherche linéaire et même la recherche binaire sembleront lentes car nous pouvons directement sauter dans l'espace mémoire où les noms commençant par «M» sont stockés.

Positionnement dans la recherche binaire

Dans la recherche binaire, si les données souhaitées ne sont pas trouvées, le reste de la liste est divisé en deux parties, inférieure et supérieure. La recherche est effectuée dans l'un ou l'autre.

Même lorsque les données sont triées, la recherche binaire n'en profite pas pour sonder la position des données souhaitées.

Sondage de position dans la recherche d'interpolation

La recherche par interpolation trouve un élément particulier en calculant la position de la sonde. Au départ, la position de la sonde est la position de l'élément le plus au milieu de la collection.

Si une correspondance se produit, l'index de l'élément est renvoyé. Pour diviser la liste en deux parties, nous utilisons la méthode suivante -

mid = Lo + ((Hi - Lo) / (A[Hi] - A[Lo])) * (X - A[Lo])

where −
   A    = list
   Lo   = Lowest index of the list
   Hi   = Highest index of the list
   A[n] = Value stored at index n in the list

Si l'élément du milieu est supérieur à l'élément, la position de la sonde est à nouveau calculée dans le sous-tableau à droite de l'élément du milieu. Sinon, l'élément est recherché dans le sous-tableau à gauche de l'élément du milieu. Ce processus se poursuit également sur le sous-tableau jusqu'à ce que la taille du sous-tableau soit réduite à zéro.

La complexité d'exécution de l'algorithme de recherche par interpolation est Ο(log (log n)) par rapport à Ο(log n) de BST dans des situations favorables.

Algorithme

Comme il s'agit d'une improvisation de l'algorithme BST existant, nous mentionnons les étapes pour rechercher l'index de valeur de données `` cible '', en utilisant le palpage de position -

Step 1 − Start searching data from middle of the list.
Step 2 − If it is a match, return the index of the item, and exit.
Step 3 − If it is not a match, probe position.
Step 4 − Divide the list using probing formula and find the new midle.
Step 5 − If data is greater than middle, search in higher sub-list.
Step 6 − If data is smaller than middle, search in lower sub-list.
Step 7 − Repeat until match.

Pseudocode

A → Array list
N → Size of A
X → Target Value

Procedure Interpolation_Search()

   Set Lo  →  0
   Set Mid → -1
   Set Hi  →  N-1

   While X does not match
   
      if Lo equals to Hi OR A[Lo] equals to A[Hi]
         EXIT: Failure, Target not found
      end if
      
      Set Mid = Lo + ((Hi - Lo) / (A[Hi] - A[Lo])) * (X - A[Lo]) 

      if A[Mid] = X
         EXIT: Success, Target found at Mid
      else 
         if A[Mid] < X
            Set Lo to Mid+1
         else if A[Mid] > X
            Set Hi to Mid-1
         end if
      end if
   End While

End Procedure

Pour connaître l'implémentation de la recherche d'interpolation dans le langage de programmation C, cliquez ici .

Hash Table est une structure de données qui stocke les données de manière associative. Dans une table de hachage, les données sont stockées dans un format de tableau, où chaque valeur de données a sa propre valeur d'index unique. L'accès aux données devient très rapide si l'on connaît l'index des données souhaitées.

Ainsi, il devient une structure de données dans laquelle les opérations d'insertion et de recherche sont très rapides quelle que soit la taille des données. Hash Table utilise un tableau comme support de stockage et utilise la technique de hachage pour générer un index où un élément doit être inséré ou à partir duquel il doit être localisé.

Hashing

Le hachage est une technique permettant de convertir une plage de valeurs de clé en une plage d'index d'un tableau. Nous allons utiliser l'opérateur modulo pour obtenir une plage de valeurs clés. Prenons un exemple de table de hachage de taille 20, et les éléments suivants doivent être stockés. Les éléments sont au format (clé, valeur).

  • (1,20)
  • (2,70)
  • (42,80)
  • (4,25)
  • (12,44)
  • (14,32)
  • (17,11)
  • (13,78)
  • (37,98)
Sr.No. Clé Hacher Index du tableau
1 1 1% 20 = 1 1
2 2 2% 20 = 2 2
3 42 42% 20 = 2 2
4 4 4% 20 = 4 4
5 12 12% 20 = 12 12
6 14 14% 20 = 14 14
sept 17 17% 20 = 17 17
8 13 13% 20 = 13 13
9 37 37% 20 = 17 17

Palpage linéaire

Comme nous pouvons le voir, il peut arriver que la technique de hachage soit utilisée pour créer un index déjà utilisé du tableau. Dans un tel cas, nous pouvons rechercher l'emplacement vide suivant dans le tableau en regardant dans la cellule suivante jusqu'à ce que nous trouvions une cellule vide. Cette technique est appelée sondage linéaire.

Sr.No. Clé Hacher Index du tableau Après un palpage linéaire, index de tableau
1 1 1% 20 = 1 1 1
2 2 2% 20 = 2 2 2
3 42 42% 20 = 2 2 3
4 4 4% 20 = 4 4 4
5 12 12% 20 = 12 12 12
6 14 14% 20 = 14 14 14
sept 17 17% 20 = 17 17 17
8 13 13% 20 = 13 13 13
9 37 37% 20 = 17 17 18

Opérations de base

Voici les principales opérations de base d'une table de hachage.

  • Search - Recherche un élément dans une table de hachage.

  • Insert - insère un élément dans une table de hachage.

  • delete - Supprime un élément d'une table de hachage.

Élément de données

Définissez un élément de données ayant des données et une clé, sur la base desquels la recherche doit être effectuée dans une table de hachage.

struct DataItem {
   int data;
   int key;
};

Méthode de hachage

Définissez une méthode de hachage pour calculer le code de hachage de la clé de la donnée élémentaire.

int hashCode(int key){
   return key % SIZE;
}

Opération de recherche

Chaque fois qu'un élément doit être recherché, calculez le code de hachage de la clé transmise et recherchez l'élément en utilisant ce code de hachage comme index dans le tableau. Utilisez le sondage linéaire pour faire avancer l'élément si l'élément n'est pas trouvé dans le code de hachage calculé.

Exemple

struct DataItem *search(int key) {
   //get the hash
   int hashIndex = hashCode(key);
	
   //move in array until an empty
   while(hashArray[hashIndex] != NULL) {
	
      if(hashArray[hashIndex]->key == key)
         return hashArray[hashIndex];
			
      //go to next cell
      ++hashIndex;
		
      //wrap around the table
      hashIndex %= SIZE;
   }

   return NULL;        
}

Insérer une opération

Chaque fois qu'un élément doit être inséré, calculez le code de hachage de la clé transmise et localisez l'index en utilisant ce code de hachage comme index dans le tableau. Utilisez le sondage linéaire pour un emplacement vide, si un élément est trouvé au niveau du code de hachage calculé.

Exemple

void insert(int key,int data) {
   struct DataItem *item = (struct DataItem*) malloc(sizeof(struct DataItem));
   item->data = data;  
   item->key = key;     

   //get the hash 
   int hashIndex = hashCode(key);

   //move in array until an empty or deleted cell
   while(hashArray[hashIndex] != NULL && hashArray[hashIndex]->key != -1) {
      //go to next cell
      ++hashIndex;
		
      //wrap around the table
      hashIndex %= SIZE;
   }
	
   hashArray[hashIndex] = item;        
}

Supprimer l'opération

Chaque fois qu'un élément doit être supprimé, calculez le code de hachage de la clé transmise et recherchez l'index en utilisant ce code de hachage comme index dans le tableau. Utilisez le sondage linéaire pour faire avancer l'élément si un élément n'est pas trouvé au niveau du code de hachage calculé. Une fois trouvé, stockez-y un élément factice pour conserver intactes les performances de la table de hachage.

Exemple

struct DataItem* delete(struct DataItem* item) {
   int key = item->key;

   //get the hash 
   int hashIndex = hashCode(key);

   //move in array until an empty 
   while(hashArray[hashIndex] !=NULL) {
	
      if(hashArray[hashIndex]->key == key) {
         struct DataItem* temp = hashArray[hashIndex]; 
			
         //assign a dummy item at deleted position
         hashArray[hashIndex] = dummyItem; 
         return temp;
      } 
		
      //go to next cell
      ++hashIndex;
		
      //wrap around the table
      hashIndex %= SIZE;
   }  
	
   return NULL;        
}

Pour en savoir plus sur l'implémentation du hachage dans le langage de programmation C, veuillez cliquer ici .

Le tri fait référence à l'organisation des données dans un format particulier. L'algorithme de tri spécifie la manière d'organiser les données dans un ordre particulier. Les ordres les plus courants sont dans l'ordre numérique ou lexicographique.

L'importance du tri réside dans le fait que la recherche de données peut être optimisée à un niveau très élevé, si les données sont stockées de manière triée. Le tri est également utilisé pour représenter les données dans des formats plus lisibles. Voici quelques exemples de tri dans des scénarios réels -

  • Telephone Directory - L'annuaire téléphonique stocke les numéros de téléphone des personnes triés par leurs noms, afin que les noms puissent être recherchés facilement.

  • Dictionary - Le dictionnaire stocke les mots dans un ordre alphabétique afin que la recherche de n'importe quel mot devienne facile.

Tri sur place et tri sur place

Les algorithmes de tri peuvent nécessiter un espace supplémentaire pour la comparaison et le stockage temporaire de quelques éléments de données. Ces algorithmes ne nécessitent pas d'espace supplémentaire et le tri est censé se produire sur place, ou par exemple, dans le tableau lui-même. C'est appeléin-place sorting. Le tri à bulles est un exemple de tri sur place.

Cependant, dans certains algorithmes de tri, le programme requiert un espace supérieur ou égal aux éléments triés. Le tri qui utilise un espace égal ou supérieur est appelénot-in-place sorting. Le tri par fusion est un exemple de tri non en place.

Tri stable et non stable

Si un algorithme de tri, après avoir trié les contenus, ne modifie pas la séquence de contenus similaires dans lesquels ils apparaissent, il est appelé stable sorting.

Si un algorithme de tri, après avoir trié le contenu, modifie la séquence de contenu similaire dans laquelle ils apparaissent, il est appelé unstable sorting.

La stabilité d'un algorithme est importante lorsque l'on souhaite conserver la séquence des éléments d'origine, comme dans un tuple par exemple.

Algorithme de tri adaptatif et non adaptatif

Un algorithme de tri est dit adaptatif s'il tire parti d'éléments déjà «triés» dans la liste à trier. Autrement dit, lors du tri si la liste source a un élément déjà trié, les algorithmes adaptatifs en tiendront compte et essaieront de ne pas les réorganiser.

Un algorithme non adaptatif est un algorithme qui ne prend pas en compte les éléments déjà triés. Ils essaient de forcer chaque élément à être réordonné pour confirmer leur tri.

Termes importants

Certains termes sont généralement inventés lors de la discussion des techniques de tri, voici une brève introduction à eux -

Ordre croissant

On dit qu'une séquence de valeurs est dans increasing order, si l'élément successif est supérieur au précédent. Par exemple, 1, 3, 4, 6, 8, 9 sont dans l'ordre croissant, car chaque élément suivant est supérieur à l'élément précédent.

Ordre décroissant

On dit qu'une séquence de valeurs est dans decreasing order, si l'élément successif est inférieur à l'élément courant. Par exemple, 9, 8, 6, 4, 3, 1 sont dans l'ordre décroissant, car chaque élément suivant est inférieur à l'élément précédent.

Ordre non croissant

On dit qu'une séquence de valeurs est dans non-increasing order, si l'élément successif est inférieur ou égal à son élément précédent dans la séquence. Cet ordre se produit lorsque la séquence contient des valeurs en double. Par exemple, 9, 8, 6, 3, 3, 1 sont dans un ordre non croissant, car chaque élément suivant est inférieur ou égal à (dans le cas de 3) mais pas supérieur à tout élément précédent.

Ordre non décroissant

On dit qu'une séquence de valeurs est dans non-decreasing order, si l'élément successif est supérieur ou égal à son élément précédent dans la séquence. Cet ordre se produit lorsque la séquence contient des valeurs en double. Par exemple, 1, 3, 3, 6, 8, 9 sont dans un ordre non décroissant, car chaque élément suivant est supérieur ou égal à (dans le cas de 3) mais pas inférieur au précédent.

Le tri à bulles est un algorithme de tri simple. Cet algorithme de tri est un algorithme basé sur la comparaison dans lequel chaque paire d'éléments adjacents est comparée et les éléments sont échangés s'ils ne sont pas dans l'ordre. Cet algorithme ne convient pas aux grands ensembles de données car sa complexité moyenne et dans le pire des cas est de Ο (n 2 ) oùn est le nombre d'éléments.

Comment fonctionne le tri à bulles?

Nous prenons un tableau non trié pour notre exemple. Le tri des bulles prend Ο (n 2 ) temps, nous le gardons donc court et précis.

Le tri à bulles commence par les deux premiers éléments, en les comparant pour vérifier lequel est le plus grand.

Dans ce cas, la valeur 33 est supérieure à 14, elle se trouve donc déjà dans des emplacements triés. Ensuite, nous comparons 33 à 27.

Nous trouvons que 27 est plus petit que 33 et ces deux valeurs doivent être permutées.

Le nouveau tableau devrait ressembler à ceci -

Ensuite, nous comparons 33 et 35. Nous constatons que les deux sont dans des positions déjà triées.

Ensuite, nous passons aux deux valeurs suivantes, 35 et 10.

Nous savons alors que 10 est plus petit 35. Par conséquent, ils ne sont pas triés.

Nous échangeons ces valeurs. Nous constatons que nous avons atteint la fin du tableau. Après une itération, le tableau devrait ressembler à ceci -

Pour être précis, nous montrons maintenant à quoi doit ressembler un tableau après chaque itération. Après la deuxième itération, cela devrait ressembler à ceci -

Notez qu'après chaque itération, au moins une valeur se déplace à la fin.

Et quand aucun échange n'est requis, le tri par bulles apprend qu'un tableau est complètement trié.

Nous devons maintenant examiner certains aspects pratiques du tri à bulles.

Algorithme

Nous supposons list est un tableau de néléments. Nous supposons en outre queswap function permute les valeurs des éléments de tableau donnés.

begin BubbleSort(list)

   for all elements of list
      if list[i] > list[i+1]
         swap(list[i], list[i+1])
      end if
   end for
   
   return list
   
end BubbleSort

Pseudocode

Nous observons dans l'algorithme que Bubble Sort compare chaque paire d'éléments de tableau à moins que le tableau entier ne soit complètement trié dans un ordre croissant. Cela peut entraîner quelques problèmes de complexité comme si le tableau n'a plus besoin d'être permuté car tous les éléments sont déjà ascendants.

Pour résoudre le problème, nous utilisons une variable d'indicateur swappedce qui nous aidera à voir si un échange a eu lieu ou non. Si aucun échange n'a eu lieu, c'est-à-dire que le tableau ne nécessite plus de traitement pour être trié, il sortira de la boucle.

Le pseudocode de l'algorithme BubbleSort peut être écrit comme suit -

procedure bubbleSort( list : array of items )

   loop = list.count;
   
   for i = 0 to loop-1 do:
      swapped = false
		
      for j = 0 to loop-1 do:
      
         /* compare the adjacent elements */   
         if list[j] > list[j+1] then
            /* swap them */
            swap( list[j], list[j+1] )		 
            swapped = true
         end if
         
      end for
      
      /*if no number was swapped that means 
      array is sorted now, break the loop.*/
      
      if(not swapped) then
         break
      end if
      
   end for
   
end procedure return list

la mise en oeuvre

Un autre problème que nous n'avons pas abordé dans notre algorithme d'origine et son pseudo-code improvisé, est que, après chaque itération, les valeurs les plus élevées s'établissent à la fin du tableau. Par conséquent, la prochaine itération n'a pas besoin d'inclure des éléments déjà triés. Pour cela, dans notre implémentation, nous limitons la boucle interne pour éviter les valeurs déjà triées.

Pour en savoir plus sur l'implémentation du tri à bulles dans le langage de programmation C, veuillez cliquer ici .

Il s'agit d'un algorithme de tri basé sur la comparaison sur place. Ici, une sous-liste est maintenue qui est toujours triée. Par exemple, la partie inférieure d'un tableau est conservée pour être triée. Un élément qui doit être «inséré» dans cette sous-liste triée doit trouver sa place appropriée, puis il doit y être inséré. D'où le nom,insertion sort.

Le tableau est recherché séquentiellement et les éléments non triés sont déplacés et insérés dans la sous-liste triée (dans le même tableau). Cet algorithme ne convient pas aux grands ensembles de données car sa complexité moyenne et dans le pire des cas est de Ο (n 2 ), oùn est le nombre d'éléments.

Comment fonctionne le tri par insertion?

Nous prenons un tableau non trié pour notre exemple.

Le tri par insertion compare les deux premiers éléments.

Il constate que les deux 14 et 33 sont déjà dans l'ordre croissant. Pour l'instant, 14 est dans une sous-liste triée.

Le tri par insertion avance et compare 33 à 27.

Et constate que 33 n'est pas dans la bonne position.

Il échange 33 contre 27. Il vérifie également avec tous les éléments de la sous-liste triée. Ici, nous voyons que la sous-liste triée n'a qu'un seul élément 14 et 27 est supérieur à 14. Par conséquent, la sous-liste triée reste triée après l'échange.

À présent, nous avons 14 et 27 dans la sous-liste triée. Ensuite, il compare 33 à 10.

Ces valeurs ne sont pas triées.

Alors on les échange.

Cependant, l'échange rend 27 et 10 non triés.

Par conséquent, nous les échangeons aussi.

Encore une fois, nous trouvons 14 et 10 dans un ordre non trié.

Nous les échangeons à nouveau. À la fin de la troisième itération, nous avons une sous-liste triée de 4 éléments.

Ce processus se poursuit jusqu'à ce que toutes les valeurs non triées soient couvertes dans une sous-liste triée. Nous allons maintenant voir quelques aspects de programmation du tri par insertion.

Algorithme

Nous avons maintenant une vue d'ensemble du fonctionnement de cette technique de tri, nous pouvons donc en déduire des étapes simples par lesquelles nous pouvons réaliser le tri par insertion.

Step 1 − If it is the first element, it is already sorted. return 1;
Step 2 − Pick next element
Step 3 − Compare with all elements in the sorted sub-list
Step 4 − Shift all the elements in the sorted sub-list that is greater than the 
         value to be sorted
Step 5 − Insert the value
Step 6 − Repeat until list is sorted

Pseudocode

procedure insertionSort( A : array of items )
   int holePosition
   int valueToInsert
	
   for i = 1 to length(A) inclusive do:
	
      /* select value to be inserted */
      valueToInsert = A[i]
      holePosition = i
      
      /*locate hole position for the element to be inserted */
		
      while holePosition > 0 and A[holePosition-1] > valueToInsert do:
         A[holePosition] = A[holePosition-1]
         holePosition = holePosition -1
      end while
		
      /* insert the number at hole position */
      A[holePosition] = valueToInsert
      
   end for
	
end procedure

Pour connaître l'implémentation du tri par insertion dans le langage de programmation C, veuillez cliquer ici .

Le tri par sélection est un algorithme de tri simple. Cet algorithme de tri est un algorithme basé sur une comparaison sur place dans lequel la liste est divisée en deux parties, la partie triée à l'extrémité gauche et la partie non triée à l'extrémité droite. Au départ, la partie triée est vide et la partie non triée est la liste entière.

Le plus petit élément est sélectionné dans le tableau non trié et échangé avec l'élément le plus à gauche, et cet élément devient une partie du tableau trié. Ce processus continue de déplacer la limite de tableau non triée d'un élément vers la droite.

Cet algorithme n'est pas adapté aux grands ensembles de données car ses complexités moyenne et pire sont de Ο (n 2 ), oùn est le nombre d'éléments.

Comment fonctionne le tri par sélection?

Considérez le tableau suivant comme exemple.

Pour la première position dans la liste triée, la liste entière est balayée séquentiellement. La première position où 14 est actuellement stocké, nous recherchons la liste entière et trouvons que 10 est la valeur la plus basse.

Nous remplaçons donc 14 par 10. Après une itération, 10, qui se trouve être la valeur minimale de la liste, apparaît en première position de la liste triée.

Pour la deuxième position, où 33 réside, nous commençons à balayer le reste de la liste de manière linéaire.

Nous trouvons que 14 est la deuxième valeur la plus basse de la liste et qu'elle devrait apparaître à la deuxième place. Nous échangeons ces valeurs.

Après deux itérations, deux moindres valeurs sont positionnées au début de manière triée.

Le même processus est appliqué au reste des éléments du tableau.

Voici une représentation graphique de l'ensemble du processus de tri -

Maintenant, apprenons quelques aspects de programmation du tri sélectif.

Algorithme

Step 1 − Set MIN to location 0
Step 2 − Search the minimum element in the list
Step 3 − Swap with value at location MIN
Step 4 − Increment MIN to point to next element
Step 5 − Repeat until list is sorted

Pseudocode

procedure selection sort 
   list  : array of items
   n     : size of list

   for i = 1 to n - 1
   /* set current element as minimum*/
      min = i    
  
      /* check the element to be minimum */

      for j = i+1 to n 
         if list[j] < list[min] then
            min = j;
         end if
      end for

      /* swap the minimum element with the current element*/
      if indexMin != i  then
         swap list[min] and list[i]
      end if
   end for
	
end procedure

Pour en savoir plus sur l'implémentation du tri de sélection dans le langage de programmation C, veuillez cliquer ici .

Le tri par fusion est une technique de tri basée sur la technique de division et de conquête. Dans le pire des cas, la complexité temporelle étant Ο (n log n), c'est l'un des algorithmes les plus respectés.

Le tri par fusion divise d'abord le tableau en deux moitiés égales, puis les combine de manière triée.

Comment fonctionne le tri par fusion?

Pour comprendre le tri par fusion, nous prenons un tableau non trié comme suit -

Nous savons que le tri par fusion divise d'abord le tableau entier de manière itérative en deux moitiés égales à moins que les valeurs atomiques ne soient atteintes. On voit ici qu'un tableau de 8 éléments est divisé en deux tableaux de taille 4.

Cela ne modifie pas la séquence d'apparition des éléments dans l'original. Maintenant, nous divisons ces deux tableaux en deux.

Nous divisons encore ces tableaux et nous obtenons une valeur atomique qui ne peut plus être divisée.

Maintenant, nous les combinons exactement de la même manière qu'ils ont été décomposés. Veuillez noter les codes couleur donnés à ces listes.

Nous comparons d'abord les éléments de chaque liste, puis les combinons dans une autre liste de manière triée. Nous voyons que 14 et 33 sont dans des positions triées. Nous comparons 27 et 10 et dans la liste cible de 2 valeurs, nous mettons 10 en premier, suivi de 27. Nous changeons l'ordre de 19 et 35 tandis que 42 et 44 sont placés séquentiellement.

Dans l'itération suivante de la phase de combinaison, nous comparons des listes de deux valeurs de données et les fusionnons dans une liste de valeurs de données trouvées en les plaçant toutes dans un ordre trié.

Après la fusion finale, la liste devrait ressembler à ceci -

Nous devrions maintenant apprendre quelques aspects de programmation du tri par fusion.

Algorithme

Le tri par fusion continue de diviser la liste en deux moitiés égales jusqu'à ce qu'elle ne puisse plus être divisée. Par définition, s'il ne s'agit que d'un élément de la liste, il est trié. Ensuite, le tri par fusion combine les listes triées plus petites en gardant également la nouvelle liste triée.

Step 1 − if it is only one element in the list it is already sorted, return.
Step 2 − divide the list recursively into two halves until it can no more be divided.
Step 3 − merge the smaller lists into new list in sorted order.

Pseudocode

Nous allons maintenant voir les pseudocodes des fonctions de tri par fusion. Comme nos algorithmes le soulignent, deux fonctions principales - diviser et fusionner.

Le tri par fusion fonctionne avec la récursivité et nous verrons notre implémentation de la même manière.

procedure mergesort( var a as array )
   if ( n == 1 ) return a

   var l1 as array = a[0] ... a[n/2]
   var l2 as array = a[n/2+1] ... a[n]

   l1 = mergesort( l1 )
   l2 = mergesort( l2 )

   return merge( l1, l2 )
end procedure

procedure merge( var a as array, var b as array )

   var c as array
   while ( a and b have elements )
      if ( a[0] > b[0] )
         add b[0] to the end of c
         remove b[0] from b
      else
         add a[0] to the end of c
         remove a[0] from a
      end if
   end while
   
   while ( a has elements )
      add a[0] to the end of c
      remove a[0] from a
   end while
   
   while ( b has elements )
      add b[0] to the end of c
      remove b[0] from b
   end while
   
   return c
	
end procedure

Pour en savoir plus sur l'implémentation du tri par fusion dans le langage de programmation C, veuillez cliquer ici .

Le tri shell est un algorithme de tri très efficace basé sur un algorithme de tri par insertion. Cet algorithme évite les grands décalages comme dans le cas du tri par insertion, si la valeur la plus petite est à l'extrême droite et doit être déplacée vers l'extrême gauche.

Cet algorithme utilise le tri par insertion sur des éléments largement répandus, d'abord pour les trier, puis trie les éléments les moins espacés. Cet espacement est appeléinterval. Cet intervalle est calculé sur la base de la formule de Knuth comme -

Formule de Knuth

h = h * 3 + 1
where −
   h is interval with initial value 1

Cet algorithme est assez efficace pour les ensembles de données de taille moyenne car sa complexité moyenne et dans le pire des cas est de Ο (n), où n est le nombre d'éléments.

Comment fonctionne le tri Shell?

Prenons l'exemple suivant pour avoir une idée du fonctionnement du tri shell. Nous prenons le même tableau que nous avons utilisé dans nos exemples précédents. Pour notre exemple et la facilité de compréhension, nous prenons l'intervalle de 4. Faites une sous-liste virtuelle de toutes les valeurs situées à l'intervalle de 4 positions. Ici, ces valeurs sont {35, 14}, {33, 19}, {42, 27} et {10, 44}

Nous comparons les valeurs de chaque sous-liste et les échangeons (si nécessaire) dans le tableau d'origine. Après cette étape, le nouveau tableau devrait ressembler à ceci -

Ensuite, nous prenons un intervalle de 2 et cet écart génère deux sous-listes - {14, 27, 35, 42}, {19, 10, 33, 44}

Nous comparons et échangeons les valeurs, si nécessaire, dans le tableau d'origine. Après cette étape, le tableau devrait ressembler à ceci -

Enfin, nous trions le reste du tableau en utilisant l'intervalle de valeur 1. Le tri shell utilise le tri par insertion pour trier le tableau.

Voici la représentation étape par étape -

Nous voyons qu'il n'a fallu que quatre swaps pour trier le reste du tableau.

Algorithme

Voici l'algorithme de tri par shell.

Step 1 − Initialize the value of h
Step 2 − Divide the list into smaller sub-list of equal interval h
Step 3 − Sort these sub-lists using insertion sort
Step 3 − Repeat until complete list is sorted

Pseudocode

Voici le pseudocode pour le tri shell.

procedure shellSort()
   A : array of items 
	
   /* calculate interval*/
   while interval < A.length /3 do:
      interval = interval * 3 + 1	    
   end while
   
   while interval > 0 do:

      for outer = interval; outer < A.length; outer ++ do:

      /* select value to be inserted */
      valueToInsert = A[outer]
      inner = outer;

         /*shift element towards right*/
         while inner > interval -1 && A[inner - interval] >= valueToInsert do:
            A[inner] = A[inner - interval]
            inner = inner - interval
         end while

      /* insert the number at hole position */
      A[inner] = valueToInsert

      end for

   /* calculate interval*/
   interval = (interval -1) /3;	  

   end while
   
end procedure

Pour connaître l'implémentation du tri shell en langage de programmation C, veuillez cliquer ici .

Le tri shell est un algorithme de tri très efficace basé sur un algorithme de tri par insertion. Cet algorithme évite les grands décalages comme dans le cas du tri par insertion, si la valeur la plus petite est à l'extrême droite et doit être déplacée vers l'extrême gauche.

Cet algorithme utilise le tri par insertion sur des éléments largement répandus, d'abord pour les trier, puis trie les éléments les moins espacés. Cet espacement est appeléinterval. Cet intervalle est calculé sur la base de la formule de Knuth comme -

Formule de Knuth

h = h * 3 + 1
where −
   h is interval with initial value 1

Cet algorithme est assez efficace pour les ensembles de données de taille moyenne car sa complexité moyenne et dans le pire des cas de cet algorithme dépend de la séquence de brèches la plus connue est Ο (n), où n est le nombre d'éléments. Et le pire des cas de complexité spatiale est O (n).

Comment fonctionne le tri Shell?

Prenons l'exemple suivant pour avoir une idée du fonctionnement du tri shell. Nous prenons le même tableau que nous avons utilisé dans nos exemples précédents. Pour notre exemple et la facilité de compréhension, nous prenons l'intervalle de 4. Faites une sous-liste virtuelle de toutes les valeurs situées à l'intervalle de 4 positions. Ici, ces valeurs sont {35, 14}, {33, 19}, {42, 27} et {10, 44}

Nous comparons les valeurs de chaque sous-liste et les échangeons (si nécessaire) dans le tableau d'origine. Après cette étape, le nouveau tableau devrait ressembler à ceci -

Ensuite, nous prenons un intervalle de 1 et cet écart génère deux sous-listes - {14, 27, 35, 42}, {19, 10, 33, 44}

Nous comparons et échangeons les valeurs, si nécessaire, dans le tableau d'origine. Après cette étape, le tableau devrait ressembler à ceci -

Enfin, nous trions le reste du tableau en utilisant l'intervalle de valeur 1. Le tri shell utilise le tri par insertion pour trier le tableau.

Voici la représentation étape par étape -

Nous voyons qu'il n'a fallu que quatre swaps pour trier le reste du tableau.

Algorithme

Voici l'algorithme de tri par shell.

Step 1 − Initialize the value of h
Step 2 − Divide the list into smaller sub-list of equal interval h
Step 3 − Sort these sub-lists using insertion sort
Step 3 − Repeat until complete list is sorted

Pseudocode

Voici le pseudocode pour le tri shell.

procedure shellSort()
   A : array of items 
	
   /* calculate interval*/
   while interval < A.length /3 do:
      interval = interval * 3 + 1	    
   end while
   
   while interval > 0 do:

      for outer = interval; outer < A.length; outer ++ do:

      /* select value to be inserted */
      valueToInsert = A[outer]
      inner = outer;

         /*shift element towards right*/
         while inner > interval -1 && A[inner - interval] >= valueToInsert do:
            A[inner] = A[inner - interval]
            inner = inner - interval
         end while

      /* insert the number at hole position */
      A[inner] = valueToInsert

      end for

   /* calculate interval*/
   interval = (interval -1) /3;	  

   end while
   
end procedure

Pour connaître l'implémentation du tri shell en langage de programmation C, veuillez cliquer ici .

Le tri rapide est un algorithme de tri très efficace basé sur le partitionnement d'un tableau de données en tableaux plus petits. Un grand tableau est partitionné en deux tableaux dont l'un contient des valeurs inférieures à la valeur spécifiée, disons pivot, en fonction de laquelle la partition est créée et un autre tableau contient des valeurs supérieures à la valeur pivot.

Quicksort partitionne un tableau puis s'appelle lui-même deux fois de manière récursive pour trier les deux sous-tableaux résultants. Cet algorithme est assez efficace pour les ensembles de données de grande taille car sa complexité moyenne et dans le pire des cas sont respectivement O (nLogn) et image.png (n 2 ).

Partition en tri rapide

La représentation animée suivante explique comment trouver la valeur de pivot dans un tableau.

La valeur pivot divise la liste en deux parties. Et récursivement, nous trouvons le pivot pour chaque sous-liste jusqu'à ce que toutes les listes contiennent un seul élément.

Algorithme de tri rapide de pivot

Sur la base de notre compréhension du partitionnement en tri rapide, nous allons maintenant essayer d'écrire un algorithme pour celui-ci, qui est le suivant.

Step 1 − Choose the highest index value has pivot
Step 2 − Take two variables to point left and right of the list excluding pivot
Step 3 − left points to the low index
Step 4 − right points to the high
Step 5 − while value at left is less than pivot move right
Step 6 − while value at right is greater than pivot move left
Step 7 − if both step 5 and step 6 does not match swap left and right
Step 8 − if left ≥ right, the point where they met is new pivot

Pseudocode du pivot de tri rapide

Le pseudocode de l'algorithme ci-dessus peut être dérivé comme -

function partitionFunc(left, right, pivot)
   leftPointer = left
   rightPointer = right - 1

   while True do
      while A[++leftPointer] < pivot do
         //do-nothing            
      end while
		
      while rightPointer > 0 && A[--rightPointer] > pivot do
         //do-nothing         
      end while
		
      if leftPointer >= rightPointer
         break
      else                
         swap leftPointer,rightPointer
      end if
		
   end while 
	
   swap leftPointer,right
   return leftPointer
	
end function

Algorithme de tri rapide

En utilisant l'algorithme de pivot récursivement, nous nous retrouvons avec des partitions possibles plus petites. Chaque partition est ensuite traitée pour un tri rapide. Nous définissons l'algorithme récursif pour le tri rapide comme suit -

Step 1 − Make the right-most index value pivot
Step 2 − partition the array using pivot value
Step 3 − quicksort left partition recursively
Step 4 − quicksort right partition recursively

Pseudocode de tri rapide

Pour en savoir plus, voyons le pseudocode de l'algorithme de tri rapide -

procedure quickSort(left, right)

   if right-left <= 0
      return
   else     
      pivot = A[right]
      partition = partitionFunc(left, right, pivot)
      quickSort(left,partition-1)
      quickSort(partition+1,right)    
   end if		
   
end procedure

Pour en savoir plus sur l'implémentation du tri rapide dans le langage de programmation C, veuillez cliquer ici .

Un graphique est une représentation picturale d'un ensemble d'objets où certaines paires d'objets sont reliées par des liens. Les objets interconnectés sont représentés par des points appelésvertices, et les liens qui relient les sommets sont appelés edges.

Formellement, un graphique est une paire d'ensembles (V, E), où V est l'ensemble des sommets et Eest l'ensemble des arêtes, reliant les paires de sommets. Jetez un œil au graphique suivant -

Dans le graphique ci-dessus,

V = {a, b, c, d, e}

E = {ab, ac, bd, cd, de}

Structure des données graphiques

Les graphiques mathématiques peuvent être représentés dans une structure de données. Nous pouvons représenter un graphe en utilisant un tableau de sommets et un tableau à deux dimensions d'arêtes. Avant d'aller plus loin, familiarisons-nous avec quelques termes importants -

  • Vertex- Chaque nœud du graphe est représenté comme un sommet. Dans l'exemple suivant, le cercle étiqueté représente des sommets. Ainsi, A à G sont des sommets. Nous pouvons les représenter à l'aide d'un tableau comme indiqué dans l'image suivante. Ici, A peut être identifié par l'indice 0. B peut être identifié à l'aide de l'indice 1 et ainsi de suite.

  • Edge- Edge représente un chemin entre deux sommets ou une ligne entre deux sommets. Dans l'exemple suivant, les lignes de A à B, de B à C, etc. représentent des arêtes. Nous pouvons utiliser un tableau à deux dimensions pour représenter un tableau comme indiqué dans l'image suivante. Ici, AB peut être représenté par 1 à la ligne 0, colonne 1, BC par 1 à la ligne 1, colonne 2 et ainsi de suite, en gardant les autres combinaisons à 0.

  • Adjacency- Deux nœuds ou sommets sont adjacents s'ils sont connectés l'un à l'autre via une arête. Dans l'exemple suivant, B est adjacent à A, C est adjacent à B, et ainsi de suite.

  • Path- Le chemin représente une séquence d'arêtes entre les deux sommets. Dans l'exemple suivant, ABCD représente un chemin de A à D.

Opérations de base

Voici les opérations principales de base d'un graphique -

  • Add Vertex - Ajoute un sommet au graphique.

  • Add Edge - Ajoute une arête entre les deux sommets du graphe.

  • Display Vertex - Affiche un sommet du graphique.

Pour en savoir plus sur Graph, veuillez lire le didacticiel sur la théorie des graphes . Nous apprendrons à parcourir un graphe dans les prochains chapitres.

L'algorithme de recherche en profondeur (DFS) parcourt un graphique dans un mouvement vers la profondeur et utilise une pile pour se souvenir d'obtenir le sommet suivant pour démarrer une recherche, lorsqu'une impasse se produit dans une itération.

Comme dans l'exemple donné ci-dessus, l'algorithme DFS passe de S à A à D à G à E à B d'abord, puis à F et enfin à C. Il utilise les règles suivantes.

  • Rule 1- Visitez le sommet non visité adjacent. Marquez-le comme visité. Affichez-le. Poussez-le dans une pile.

  • Rule 2- Si aucun sommet adjacent n'est trouvé, affiche un sommet de la pile. (Il fera apparaître tous les sommets de la pile, qui n'ont pas de sommets adjacents.)

  • Rule 3 - Répétez la règle 1 et la règle 2 jusqu'à ce que la pile soit vide.

Étape Traversée La description
1
Initialisez la pile.
2
marque Scomme visité et mettez-le sur la pile. Explorez n'importe quel nœud adjacent non visité depuisS. Nous avons trois nœuds et nous pouvons choisir n'importe lequel d'entre eux. Pour cet exemple, nous prendrons le nœud dans un ordre alphabétique.
3
marque Acomme visité et mettez-le sur la pile. Explorez n'importe quel nœud adjacent non visité de A. Les deuxS et D sont adjacents à A mais nous nous préoccupons uniquement des nœuds non visités.
4
Visite Det marquez-le comme visité et mettez-le sur la pile. Ici nous avonsB et C nœuds, qui sont adjacents à Det les deux ne sont pas visités. Cependant, nous choisirons à nouveau par ordre alphabétique.
5
Nous choisissons B, marquez-le comme visité et mettez-le sur la pile. IciBn'a aucun nœud adjacent non visité. Alors, on popB de la pile.
6
Nous vérifions le haut de la pile pour revenir au nœud précédent et vérifions s'il contient des nœuds non visités. Ici, on trouveD être au sommet de la pile.
sept
Seul le nœud adjacent non visité provient de D est Cmaintenant. Alors on visiteC, marquez-le comme visité et placez-le sur la pile.

Comme Cn'a pas de nœud adjacent non visité, nous continuons donc à sauter la pile jusqu'à ce que nous trouvions un nœud qui a un nœud adjacent non visité. Dans ce cas, il n'y en a pas et nous continuons à sauter jusqu'à ce que la pile soit vide.

Pour connaître l'implémentation de cet algorithme en langage de programmation C, cliquez ici .

L'algorithme BFS (Breadth First Search) parcourt un graphe dans un mouvement vers la largeur et utilise une file d'attente pour se souvenir d'obtenir le sommet suivant pour démarrer une recherche, lorsqu'une impasse se produit dans une itération.

Comme dans l'exemple donné ci-dessus, l'algorithme BFS parcourt de A à B à E à F d'abord puis à C et G enfin à D. Il utilise les règles suivantes.

  • Rule 1- Visitez le sommet non visité adjacent. Marquez-le comme visité. Affichez-le. Insérez-le dans une file d'attente.

  • Rule 2 - Si aucun sommet adjacent n'est trouvé, supprimez le premier sommet de la file d'attente.

  • Rule 3 - Répétez la règle 1 et la règle 2 jusqu'à ce que la file d'attente soit vide.

Étape Traversée La description
1
Initialisez la file d'attente.
2
Nous commençons par visiter S (nœud de départ) et marquez-le comme visité.
3
Nous voyons alors un nœud adjacent non visité de S. Dans cet exemple, nous avons trois nœuds mais nous choisissons par ordre alphabétiqueA, marquez-le comme visité et placez-le en file d'attente.
4
Ensuite, le nœud adjacent non visité de S est B. Nous le marquons comme visité et le mettons en file d'attente.
5
Ensuite, le nœud adjacent non visité de S est C. Nous le marquons comme visité et le mettons en file d'attente.
6
Maintenant, Sest laissé sans nœuds adjacents non visités. Alors, nous sortons de la file d'attente et trouvonsA.
sept
De A nous avons Dcomme nœud adjacent non visité. Nous le marquons comme visité et le mettons en file d'attente.

À ce stade, il ne nous reste aucun nœud non marqué (non visité). Mais selon l'algorithme, nous continuons à retirer la file d'attente afin d'obtenir tous les nœuds non visités. Lorsque la file d'attente est vidée, le programme est terminé.

L'implémentation de cet algorithme en langage de programmation C peut être vue ici .

L'arbre représente les nœuds connectés par des arêtes. Nous discuterons spécifiquement de l'arbre binaire ou de l'arbre de recherche binaire.

L'arbre binaire est une structure de données spéciale utilisée à des fins de stockage de données. Un arbre binaire a une condition spéciale selon laquelle chaque nœud peut avoir un maximum de deux enfants. Un arbre binaire présente les avantages à la fois d'un tableau ordonné et d'une liste liée car la recherche est aussi rapide que dans un tableau trié et les opérations d'insertion ou de suppression sont aussi rapides que dans une liste liée.

Termes importants

Voici les termes importants concernant l'arbre.

  • Path - Chemin fait référence à la séquence de nœuds le long des bords d'un arbre.

  • Root- Le nœud en haut de l'arbre s'appelle root. Il n'y a qu'une seule racine par arbre et un chemin entre le nœud racine et n'importe quel nœud.

  • Parent - Tout nœud à l'exception du nœud racine a une arête vers le haut jusqu'à un nœud appelé parent.

  • Child - Le nœud sous un nœud donné connecté par son bord descendant est appelé son nœud enfant.

  • Leaf - Le nœud qui n'a pas de nœud enfant est appelé nœud feuille.

  • Subtree - Le sous-arbre représente les descendants d'un nœud.

  • Visiting - La visite fait référence à la vérification de la valeur d'un nœud lorsque le contrôle est sur le nœud.

  • Traversing - Traverser signifie passer par des nœuds dans un ordre spécifique.

  • Levels- Le niveau d'un nœud représente la génération d'un nœud. Si le nœud racine est au niveau 0, alors son nœud enfant suivant est au niveau 1, son petit-enfant est au niveau 2, et ainsi de suite.

  • keys - La clé représente une valeur d'un nœud sur la base de laquelle une opération de recherche doit être effectuée pour un nœud.

Représentation de l'arbre de recherche binaire

L'arbre de recherche binaire présente un comportement spécial. L'enfant gauche d'un nœud doit avoir une valeur inférieure à la valeur de son parent et l'enfant droit du nœud doit avoir une valeur supérieure à sa valeur parent.

Nous allons implémenter l'arborescence en utilisant l'objet nœud et en les connectant via des références.

Noeud d'arbre

Le code pour écrire un nœud d'arbre serait similaire à ce qui est donné ci-dessous. Il a une partie de données et des références à ses nœuds enfants gauche et droit.

struct node {
   int data;   
   struct node *leftChild;
   struct node *rightChild;
};

Dans une arborescence, tous les nœuds partagent une construction commune.

Opérations de base BST

Les opérations de base qui peuvent être effectuées sur une structure de données d'arbre de recherche binaire sont les suivantes:

  • Insert - Insère un élément dans un arbre / crée un arbre.

  • Search - Recherche un élément dans un arbre.

  • Preorder Traversal - Traverse un arbre en pré-commande.

  • Inorder Traversal - Traverse un arbre dans l'ordre.

  • Postorder Traversal - Traverse un arbre de manière post-ordre.

Nous allons apprendre à créer (insérer dans) une structure arborescente et à rechercher une donnée dans un arbre dans ce chapitre. Nous en apprendrons davantage sur les méthodes de traversée des arbres dans le prochain chapitre.

Insérer une opération

La toute première insertion crée l'arborescence. Ensuite, chaque fois qu'un élément doit être inséré, recherchez d'abord son emplacement approprié. Commencez la recherche à partir du nœud racine, puis si les données sont inférieures à la valeur de clé, recherchez l'emplacement vide dans le sous-arbre de gauche et insérez les données. Sinon, recherchez l'emplacement vide dans le sous-arbre de droite et insérez les données.

Algorithme

If root is NULL 
   then create root node
return

If root exists then
   compare the data with node.data
   
   while until insertion position is located

      If data is greater than node.data
         goto right subtree
      else
         goto left subtree

   endwhile 
   
   insert data
	
end If

la mise en oeuvre

L'implémentation de la fonction d'insertion devrait ressembler à ceci -

void insert(int data) {
   struct node *tempNode = (struct node*) malloc(sizeof(struct node));
   struct node *current;
   struct node *parent;

   tempNode->data = data;
   tempNode->leftChild = NULL;
   tempNode->rightChild = NULL;

   //if tree is empty, create root node
   if(root == NULL) {
      root = tempNode;
   } else {
      current = root;
      parent  = NULL;

      while(1) {                
         parent = current;

         //go to left of the tree
         if(data < parent->data) {
            current = current->leftChild;                
            
            //insert to the left
            if(current == NULL) {
               parent->leftChild = tempNode;
               return;
            }
         }
			
         //go to right of the tree
         else {
            current = current->rightChild;
            
            //insert to the right
            if(current == NULL) {
               parent->rightChild = tempNode;
               return;
            }
         }
      }            
   }
}

Opération de recherche

Chaque fois qu'un élément doit être recherché, commencez la recherche à partir du nœud racine, puis si les données sont inférieures à la valeur de clé, recherchez l'élément dans le sous-arbre de gauche. Sinon, recherchez l'élément dans le sous-arbre de droite. Suivez le même algorithme pour chaque nœud.

Algorithme

If root.data is equal to search.data
   return root
else
   while data not found

      If data is greater than node.data
         goto right subtree
      else
         goto left subtree
         
      If data found
         return node
   endwhile 
   
   return data not found
   
end if

L'implémentation de cet algorithme devrait ressembler à ceci.

struct node* search(int data) {
   struct node *current = root;
   printf("Visiting elements: ");

   while(current->data != data) {
      if(current != NULL)
      printf("%d ",current->data); 
      
      //go to left tree

      if(current->data > data) {
         current = current->leftChild;
      }
      //else go to right tree
      else {                
         current = current->rightChild;
      }

      //not found
      if(current == NULL) {
         return NULL;
      }

      return current;
   }  
}

Pour connaître la mise en œuvre de la structure de données de l'arborescence de recherche binaire, veuillez cliquer ici .

La traversée est un processus permettant de visiter tous les nœuds d'un arbre et peut également imprimer leurs valeurs. Parce que tous les nœuds sont connectés via des bords (liens), nous partons toujours du nœud racine (tête). Autrement dit, nous ne pouvons pas accéder de manière aléatoire à un nœud dans un arbre. Il y a trois façons que nous utilisons pour traverser un arbre -

  • Traversée en ordre
  • Précommander Traversal
  • Traversée après commande

Généralement, nous parcourons un arbre pour rechercher ou localiser un élément ou une clé donné dans l'arborescence ou pour imprimer toutes les valeurs qu'il contient.

Traversée en ordre

Dans cette méthode de traversée, le sous-arbre gauche est visité en premier, puis la racine et plus tard le sous-arbre droit. Nous devons toujours nous rappeler que chaque nœud peut représenter un sous-arbre lui-même.

Si un arbre binaire est parcouru in-order, la sortie produira des valeurs de clé triées dans un ordre croissant.

Nous partons de A, et suivant la traversée dans l'ordre, nous nous déplaçons vers son sous-arbre gauche B. Best également parcouru dans l'ordre. Le processus se poursuit jusqu'à ce que tous les nœuds soient visités. La sortie de la traversée en ordre de cet arbre sera -

D → B → E → A → F → C → G

Algorithme

Until all nodes are traversed −
Step 1 − Recursively traverse left subtree.
Step 2 − Visit root node.
Step 3 − Recursively traverse right subtree.

Précommander Traversal

Dans cette méthode de traversée, le nœud racine est visité en premier, puis le sous-arbre gauche et enfin le sous-arbre droit.

Nous partons de A, et après la traversée de pré-commande, nous visitons d'abord A lui-même, puis passez à son sous-arbre gauche B. Best également parcouru en pré-commande. Le processus se poursuit jusqu'à ce que tous les nœuds soient visités. La sortie du parcours de pré-commande de cet arbre sera -

A → B → D → E → C → F → G

Algorithme

Until all nodes are traversed −
Step 1 − Visit root node.
Step 2 − Recursively traverse left subtree.
Step 3 − Recursively traverse right subtree.

Traversée après commande

Dans cette méthode de traversée, le nœud racine est visité en dernier, d'où le nom. Nous traversons d'abord le sous-arbre gauche, puis le sous-arbre droit et enfin le nœud racine.

Nous partons de A, et après le parcours post-ordre, nous visitons d'abord le sous-arbre de gauche B. Best également traversée après la commande. Le processus se poursuit jusqu'à ce que tous les nœuds soient visités. La sortie de la traversée post-ordre de cet arbre sera -

D → E → B → F → G → C → A

Algorithme

Until all nodes are traversed −
Step 1 − Recursively traverse left subtree.
Step 2 − Recursively traverse right subtree.
Step 3 − Visit root node.

Pour vérifier l'implémentation en C de la traversée d'arbres, veuillez cliquer ici .

Un arbre de recherche binaire (BST) est un arbre dans lequel tous les nœuds suivent les propriétés mentionnées ci-dessous -

  • La valeur de la clé du sous-arbre gauche est inférieure à la valeur de la clé de son nœud parent (racine).

  • La valeur de la clé du sous-arbre droit est supérieure ou égale à la valeur de la clé de son nœud parent (racine).

Ainsi, BST divise tous ses sous-arbres en deux segments; le sous-arbre gauche et le sous-arbre droit et peuvent être définis comme -

left_subtree (keys) < node (key) ≤ right_subtree (keys)

Représentation

BST est une collection de nœuds disposés de manière à conserver les propriétés BST. Chaque nœud a une clé et une valeur associée. Lors de la recherche, la clé souhaitée est comparée aux clés dans BST et si elle est trouvée, la valeur associée est récupérée.

Voici une représentation picturale de BST -

Nous observons que la clé du nœud racine (27) a toutes les clés de moindre valeur sur le sous-arbre de gauche et les clés de valeur supérieure sur le sous-arbre de droite.

Opérations de base

Voici les opérations de base d'un arbre -

  • Search - Recherche un élément dans un arbre.

  • Insert - Insère un élément dans une arborescence.

  • Pre-order Traversal - Traverse un arbre en pré-commande.

  • In-order Traversal - Traverse un arbre dans l'ordre.

  • Post-order Traversal - Traverse un arbre de manière post-ordre.

Nœud

Définissez un nœud contenant des données, des références à ses nœuds enfants gauche et droit.

struct node {
   int data;   
   struct node *leftChild;
   struct node *rightChild;
};

Opération de recherche

Chaque fois qu'un élément doit être recherché, lancez la recherche à partir du nœud racine. Ensuite, si les données sont inférieures à la valeur de clé, recherchez l'élément dans le sous-arbre de gauche. Sinon, recherchez l'élément dans le sous-arbre de droite. Suivez le même algorithme pour chaque nœud.

Algorithme

struct node* search(int data){
   struct node *current = root;
   printf("Visiting elements: ");
	
   while(current->data != data){
	
      if(current != NULL) {
         printf("%d ",current->data);
			
         //go to left tree
         if(current->data > data){
            current = current->leftChild;
         }  //else go to right tree
         else {                
            current = current->rightChild;
         }
			
         //not found
         if(current == NULL){
            return NULL;
         }
      }			
   }
   
   return current;
}

Insérer une opération

Chaque fois qu'un élément doit être inséré, localisez d'abord son emplacement approprié. Commencez la recherche à partir du nœud racine, puis si les données sont inférieures à la valeur de clé, recherchez l'emplacement vide dans le sous-arbre de gauche et insérez les données. Sinon, recherchez l'emplacement vide dans le sous-arbre de droite et insérez les données.

Algorithme

void insert(int data) {
   struct node *tempNode = (struct node*) malloc(sizeof(struct node));
   struct node *current;
   struct node *parent;

   tempNode->data = data;
   tempNode->leftChild = NULL;
   tempNode->rightChild = NULL;

   //if tree is empty
   if(root == NULL) {
      root = tempNode;
   } else {
      current = root;
      parent = NULL;

      while(1) {                
         parent = current;
			
         //go to left of the tree
         if(data < parent->data) {
            current = current->leftChild;                
            //insert to the left
				
            if(current == NULL) {
               parent->leftChild = tempNode;
               return;
            }
         }  //go to right of the tree
         else {
            current = current->rightChild;
            
            //insert to the right
            if(current == NULL) {
               parent->rightChild = tempNode;
               return;
            }
         }
      }            
   }
}

Que faire si l'entrée de l'arbre de recherche binaire est triée (par ordre croissant ou décroissant)? Cela ressemblera alors à ceci -

On observe que les performances les plus défavorables de BST sont les plus proches des algorithmes de recherche linéaire, c'est-à-dire Ο (n). Dans les données en temps réel, nous ne pouvons pas prédire le modèle de données et leurs fréquences. Il est donc nécessaire d’équilibrer la BST existante.

Nommé d'après leur inventeur Adelson, Velski & Landis, AVL treessont un arbre de recherche binaire d'équilibrage de la hauteur. L'arbre AVL vérifie la hauteur des sous-arbres gauche et droit et assure que la différence n'est pas supérieure à 1. Cette différence est appeléeBalance Factor.

Ici, nous voyons que le premier arbre est équilibré et les deux arbres suivants ne sont pas équilibrés -

Dans le deuxième arbre, le sous-arbre gauche de C a la hauteur 2 et le sous-arbre droit a la hauteur 0, donc la différence est 2. Dans le troisième arbre, le sous-arbre droit de Aa la hauteur 2 et la gauche est manquante, donc c'est 0, et la différence est à nouveau 2. L'arbre AVL permet à la différence (facteur d'équilibre) d'être seulement 1.

BalanceFactor = height(left-sutree) − height(right-sutree)

Si la différence de hauteur des sous-arbres gauche et droit est supérieure à 1, l'arbre est équilibré en utilisant certaines techniques de rotation.

Rotations AVL

Pour s'équilibrer, un arbre AVL peut effectuer les quatre types de rotations suivants -

  • Rotation gauche
  • Rotation droite
  • Rotation gauche-droite
  • Rotation droite-gauche

Les deux premières rotations sont des rotations simples et les deux suivantes sont des rotations doubles. Pour avoir un arbre déséquilibré, il faut au moins un arbre de hauteur 2. Avec cet arbre simple, comprenons-les un par un.

Rotation gauche

Si un arbre devient déséquilibré, lorsqu'un nœud est inséré dans le sous-arbre droit du sous-arbre droit, nous effectuons une seule rotation à gauche -

Dans notre exemple, node Aest devenu déséquilibré lorsqu'un nœud est inséré dans le sous-arbre droit du sous-arbre droit de A. Nous effectuons la rotation gauche en faisantA le sous-arbre gauche de B.

Rotation à droite

L'arbre AVL peut devenir déséquilibré, si un nœud est inséré dans le sous-arbre gauche du sous-arbre gauche. L'arbre a alors besoin d'une bonne rotation.

Comme illustré, le nœud déséquilibré devient l'enfant droit de son enfant gauche en effectuant une rotation à droite.

Rotation gauche-droite

Les doubles rotations sont une version légèrement complexe des versions déjà expliquées des rotations. Pour mieux les comprendre, il faut prendre note de chaque action effectuée lors de la rotation. Voyons d'abord comment effectuer une rotation gauche-droite. Une rotation gauche-droite est une combinaison de rotation gauche suivie d'une rotation droite.

Etat action
Un nœud a été inséré dans le sous-arbre droit du sous-arbre gauche. Cela faitCun nœud déséquilibré. Ces scénarios amènent l'arborescence AVL à effectuer une rotation gauche-droite.
Nous effectuons d'abord la rotation gauche sur le sous-arbre gauche de C. Cela faitA, le sous-arbre gauche de B.
Nœud C est toujours déséquilibré, mais maintenant, c'est à cause du sous-arbre gauche du sous-arbre gauche.
Nous allons maintenant faire pivoter l'arbre à droite, B le nouveau nœud racine de ce sous-arbre. C devient maintenant le sous-arbre droit de son propre sous-arbre gauche.
L'arbre est maintenant équilibré.

Rotation droite-gauche

Le deuxième type de double rotation est la rotation droite-gauche. C'est une combinaison de rotation droite suivie de rotation gauche.

Etat action
Un nœud a été inséré dans le sous-arbre gauche du sous-arbre droit. Cela faitA, un nœud déséquilibré avec un facteur d'équilibre 2.
Tout d'abord, nous effectuons la bonne rotation le long C nœud, création C le sous-arbre droit de son propre sous-arbre gauche B. Maintenant,B devient le bon sous-arbre de A.
Nœud A est toujours déséquilibré en raison du sous-arbre droit de son sous-arbre droit et nécessite une rotation à gauche.
Une rotation à gauche est effectuée en faisant B le nouveau nœud racine du sous-arbre. A devient le sous-arbre gauche de son sous-arbre droit B.
L'arbre est maintenant équilibré.

Un arbre couvrant est un sous-ensemble du graphe G, qui a tous les sommets couverts avec un nombre minimum d'arêtes possible. Par conséquent, un Spanning Tree n'a pas de cycles et il ne peut pas être déconnecté.

Par cette définition, nous pouvons tirer la conclusion que chaque graphe G connecté et non dirigé a au moins un arbre couvrant. Un graphe déconnecté n'a pas d'arbre couvrant, car il ne peut pas être étendu à tous ses sommets.

Nous avons trouvé trois arbres couvrant un graphique complet. Un graphe non orienté complet peut avoir un maximumnn-2 nombre d'arbres couvrant, où nest le nombre de nœuds. Dans l'exemple abordé ci-dessus,n is 3, Par conséquent 33−2 = 3 couvrant les arbres sont possibles.

Propriétés générales de Spanning Tree

Nous comprenons maintenant qu'un graphique peut avoir plus d'un arbre couvrant. Voici quelques propriétés de l'arbre couvrant connecté au graphe G -

  • Un graphe connexe G peut avoir plus d'un arbre couvrant.

  • Tous les arbres couvrant possibles du graphe G ont le même nombre d'arêtes et de sommets.

  • L'arbre couvrant n'a pas de cycle (boucles).

  • La suppression d'un bord de l'arbre couvrant rendra le graphe déconnecté, c'est-à-dire que l'arbre couvrant est minimally connected.

  • L'ajout d'un bord à l'arbre couvrant créera un circuit ou une boucle, c'est-à-dire que l'arbre couvrant est maximally acyclic.

Propriétés mathématiques de Spanning Tree

  • Spanning Tree a n-1 bords, où n est le nombre de nœuds (sommets).

  • À partir d'un graphique complet, en supprimant le maximum e - n + 1 bords, nous pouvons construire un arbre couvrant.

  • Un graphique complet peut avoir un maximum nn-2 nombre d'arbres couvrant.

Ainsi, nous pouvons conclure que les arbres couvrants sont un sous-ensemble du graphe G connecté et que les graphiques déconnectés n'ont pas d'arbre couvrant.

Application de Spanning Tree

Le Spanning Tree est essentiellement utilisé pour trouver un chemin minimum pour connecter tous les nœuds d'un graphe. Les applications courantes des arbres couvrants sont -

  • Civil Network Planning

  • Computer Network Routing Protocol

  • Cluster Analysis

Comprenons cela à travers un petit exemple. Considérez le réseau de la ville comme un énorme graphe et envisage maintenant de déployer des lignes téléphoniques de telle sorte que, avec un minimum de lignes, nous puissions nous connecter à tous les nœuds de la ville. C'est là que l'arbre couvrant entre en scène.

Arbre couvrant minimum (MST)

Dans un graphique pondéré, un arbre couvrant minimum est un arbre couvrant qui a un poids minimum que tous les autres arbres couvrant du même graphique. Dans des situations du monde réel, ce poids peut être mesuré en tant que distance, encombrement, charge de trafic ou toute valeur arbitraire indiquée sur les bords.

Algorithme Spanning-Tree minimum

Nous allons découvrir ici deux algorithmes d'arbre couvrant les plus importants -

  • Algorithme de Kruskal

  • Algorithme de Prim

Les deux sont des algorithmes gourmands.

Heap est un cas particulier de structure de données d'arborescence binaire équilibrée où la clé du nœud racine est comparée à ses enfants et arrangée en conséquence. Siα a un nœud enfant β puis -

touche (α) ≥ touche (β)

La valeur de parent étant supérieure à celle de child, cette propriété génère Max Heap. Sur la base de ces critères, un tas peut être de deux types -

For Input → 35 33 42 10 14 19 27 44 26 31

Min-Heap - Où la valeur du nœud racine est inférieure ou égale à l'un de ses enfants.

Max-Heap - Où la valeur du nœud racine est supérieure ou égale à l'un de ses enfants.

Les deux arbres sont construits en utilisant la même entrée et le même ordre d'arrivée.

Algorithme de construction de tas maximum

Nous utiliserons le même exemple pour montrer comment un Max Heap est créé. La procédure pour créer un tas min est similaire mais nous optons pour des valeurs min au lieu de valeurs max.

Nous allons dériver un algorithme pour le tas maximum en insérant un élément à la fois. À tout moment, le tas doit conserver sa propriété. Lors de l'insertion, nous supposons également que nous insérons un nœud dans une arborescence déjà entassée.

Step 1 − Create a new node at the end of heap.
Step 2 − Assign new value to the node.
Step 3 − Compare the value of this child node with its parent.
Step 4 − If value of parent is less than child, then swap them.
Step 5 − Repeat step 3 & 4 until Heap property holds.

Note - Dans l'algorithme de construction Min Heap, nous nous attendons à ce que la valeur du nœud parent soit inférieure à celle du nœud enfant.

Comprenons la construction de Max Heap par une illustration animée. Nous considérons le même échantillon d'entrée que nous avons utilisé précédemment.

Algorithme de suppression maximale du tas

Dérivons un algorithme pour supprimer du tas max. La suppression dans le tas Max (ou Min) se produit toujours à la racine pour supprimer la valeur Maximum (ou minimum).

Step 1 − Remove root node.
Step 2 − Move the last element of last level to root.
Step 3 − Compare the value of this child node with its parent.
Step 4 − If value of parent is less than child, then swap them.
Step 5 − Repeat step 3 & 4 until Heap property holds.

Certains langages de programmation informatique permettent à un module ou à une fonction de s'appeler. Cette technique est connue sous le nom de récursivité. En récursivité, une fonctionα s'appelle directement ou appelle une fonction β qui à son tour appelle la fonction d'origine α. La fonctionα est appelée fonction récursive.

Example - une fonction s'appelant elle-même.

int function(int value) {
   if(value < 1)
      return;
   function(value - 1);

   printf("%d ",value);   
}

Example - une fonction qui appelle une autre fonction qui la rappelle à son tour.

int function1(int value1) {
   if(value1 < 1)
      return;
   function2(value1 - 1);
   printf("%d ",value1);   
}
int function2(int value2) {
   function1(value2);
}

Propriétés

Une fonction récursive peut aller à l'infini comme une boucle. Pour éviter l'exécution infinie d'une fonction récursive, il existe deux propriétés qu'une fonction récursive doit avoir -

  • Base criteria - Il doit y avoir au moins un critère de base ou une condition, de sorte que, lorsque cette condition est remplie, la fonction cesse de s'appeler récursivement.

  • Progressive approach - Les appels récursifs doivent progresser de telle sorte que chaque fois qu'un appel récursif est effectué, il se rapproche des critères de base.

la mise en oeuvre

De nombreux langages de programmation implémentent la récursivité au moyen de stacks. Généralement, chaque fois qu'une fonction (caller) appelle une autre fonction (callee) ou elle-même en tant qu'appelé, la fonction appelante transfère le contrôle d'exécution à l'appelé. Ce processus de transfert peut également impliquer certaines données à transmettre de l'appelant à l'appelé.

Cela implique que la fonction appelante doit suspendre temporairement son exécution et la reprendre plus tard lorsque le contrôle d'exécution revient de la fonction appelée. Ici, la fonction appelante doit démarrer exactement à partir du point d'exécution où elle se met en attente. Il a également besoin des mêmes valeurs de données exactes sur lesquelles il travaillait. Pour cela, un enregistrement d'activation (ou frame de pile) est créé pour la fonction appelant.

Cet enregistrement d'activation conserve les informations sur les variables locales, les paramètres formels, l'adresse de retour et toutes les informations transmises à la fonction appelante.

Analyse de la récursivité

On peut se demander pourquoi utiliser la récursivité, car la même tâche peut être effectuée avec l'itération. La première raison est que la récursivité rend un programme plus lisible et en raison des derniers systèmes CPU améliorés, la récursivité est plus efficace que les itérations.

Complexité temporelle

En cas d'itérations, nous prenons le nombre d'itérations pour compter la complexité temporelle. De même, en cas de récursivité, en supposant que tout est constant, nous essayons de déterminer le nombre de fois qu'un appel récursif est effectué. Un appel fait à une fonction est Ο (1), donc le nombre (n) de fois qu'un appel récursif est fait rend la fonction récursive Ο (n).

Complexité spatiale

La complexité de l'espace est comptée comme la quantité d'espace supplémentaire nécessaire à l'exécution d'un module. En cas d'itérations, le compilateur ne nécessite guère d'espace supplémentaire. Le compilateur continue de mettre à jour les valeurs des variables utilisées dans les itérations. Mais en cas de récursivité, le système doit stocker l'enregistrement d'activation à chaque fois qu'un appel récursif est effectué. Par conséquent, on considère que la complexité spatiale d'une fonction récursive peut être supérieure à celle d'une fonction avec itération.

Tour de Hanoi, est un puzzle mathématique qui se compose de trois tours (chevilles) et plus d'un anneaux est comme illustré -

Ces anneaux sont de tailles différentes et empilés dans un ordre croissant, c'est-à-dire que le plus petit se place sur le plus grand. Il existe d'autres variantes du puzzle où le nombre de disques augmente, mais le nombre de tours reste le même.

Règles

La mission est de déplacer tous les disques vers une autre tour sans violer la séquence d'arrangement. Quelques règles à suivre pour la tour de Hanoi sont:

  • Un seul disque peut être déplacé entre les tours à un moment donné.
  • Seul le disque "supérieur" peut être supprimé.
  • Aucun grand disque ne peut reposer sur un petit disque.

Voici une représentation animée de la résolution d'un puzzle de la Tour de Hanoi avec trois disques.

Le puzzle de la tour de Hanoi avec n disques peut être résolu au minimum 2n−1pas. Cette présentation montre qu'un puzzle avec 3 disques a pris23 - 1 = 7 pas.

Algorithme

Pour écrire un algorithme pour la tour de Hanoi, nous devons d'abord apprendre à résoudre ce problème avec moins de disques, disons → 1 ou 2. Nous marquons trois tours avec un nom, source, destination et aux(uniquement pour aider à déplacer les disques). Si nous n'avons qu'un seul disque, il peut facilement être déplacé de la source à la destination.

Si nous avons 2 disques -

  • Tout d'abord, nous déplaçons le plus petit disque (du haut) vers aux peg.
  • Ensuite, nous déplaçons le plus grand disque (du bas) vers la cheville de destination.
  • Et enfin, nous déplaçons le plus petit disque de aux à destination.

Alors maintenant, nous sommes en mesure de concevoir un algorithme pour la tour de Hanoi avec plus de deux disques. Nous divisons la pile de disques en deux parties. Le plus grand disque (n ème disque) est dans une partie et tous les autres (n-1) disques sont dans la seconde partie.

Notre objectif ultime est de déplacer le disque nde la source à la destination, puis placez-y tous les autres disques (n1). On peut imaginer appliquer la même chose de manière récursive pour tout ensemble donné de disques.

Les étapes à suivre sont -

Step 1 − Move n-1 disks from source to aux
Step 2 − Move nth disk from source to dest
Step 3 − Move n-1 disks from aux to dest

Un algorithme récursif pour la tour de Hanoi peut être piloté comme suit -

START
Procedure Hanoi(disk, source, dest, aux)

   IF disk == 1, THEN
      move disk from source to dest             
   ELSE
      Hanoi(disk - 1, source, aux, dest)     // Step 1
      move disk from source to dest          // Step 2
      Hanoi(disk - 1, aux, dest, source)     // Step 3
   END IF
   
END Procedure
STOP

Pour vérifier l'implémentation en programmation C, cliquez ici .

La série de Fibonacci génère le numéro suivant en ajoutant deux numéros précédents. La série de Fibonacci commence à partir de deux nombres -F0 & F1. Les valeurs initiales de F 0 et F 1 peuvent être prises respectivement 0, 1 ou 1, 1.

La série de Fibonacci satisfait aux conditions suivantes -

Fn = Fn-1 + Fn-2

Par conséquent, une série de Fibonacci peut ressembler à ceci -

F 8 = 0 1 1 2 3 5 8 13

ou, ce -

F 8 = 1 1 2 3 5 8 13 21

À des fins d'illustration, Fibonacci de F 8 est affiché comme -

Algorithme itératif de Fibonacci

Nous essayons d'abord de rédiger l'algorithme itératif pour les séries de Fibonacci.

Procedure Fibonacci(n)
   declare f0, f1, fib, loop 
   
   set f0 to 0
   set f1 to 1
   
   display f0, f1
   
   for loop ← 1 to n
   
      fib ← f0 + f1   
      f0 ← f1
      f1 ← fib

      display fib
   end for
	
end procedure

Pour connaître la mise en œuvre de l'algorithme ci-dessus dans le langage de programmation C, cliquez ici .

Algorithme récursif de Fibonacci

Apprenons à créer une série d'algorithmes récursifs de Fibonacci. Les critères de base de la récursivité.

START
Procedure Fibonacci(n)
   declare f0, f1, fib, loop 
   
   set f0 to 0
   set f1 to 1
   
   display f0, f1
   
   for loop ← 1 to n
   
      fib ← f0 + f1   
      f0 ← f1
      f1 ← fib

      display fib
   end for

END

Pour voir l'implémentation de l'algorithme ci-dessus dans le langage de programmation c, cliquez ici .