Qu'est-ce que le déroulement de la pile? J'ai cherché mais je n'ai pas trouvé de réponse éclairante!
Qu'est-ce que le déroulement de la pile? J'ai cherché mais je n'ai pas trouvé de réponse éclairante!
Le déroulement de la pile est généralement abordé en relation avec la gestion des exceptions. Voici un exemple:
void func( int x )
{
char* pleak = new char[1024]; // might be lost => memory leak
std::string s( "hello world" ); // will be properly destructed
if ( x ) throw std::runtime_error( "boom" );
delete [] pleak; // will only get here if x == 0. if x!=0, throw exception
}
int main()
{
try
{
func( 10 );
}
catch ( const std::exception& e )
{
return 1;
}
return 0;
}
Ici, la mémoire allouée pour pleak
sera perdue si une exception est levée, tandis que la mémoire allouée à s
sera correctement libérée par le std::string
destructeur dans tous les cas. Les objets alloués sur la pile sont "déroulés" lorsque la portée est quittée (ici, la portée est de la fonction func
.) Ceci est fait par le compilateur insérant des appels aux destructeurs de variables automatiques (de pile).
Maintenant, c'est un concept très puissant menant à la technique appelée RAII , qui est l' acquisition de ressources est l'initialisation , qui nous aide à gérer des ressources comme la mémoire, les connexions de base de données, les descripteurs de fichiers ouverts, etc. en C ++.
Cela nous permet maintenant de fournir des garanties de sécurité exceptionnelles .
delete [] pleak;
n'est atteint que si x == 0.
Tout cela concerne le C ++:
Définition : Lorsque vous créez des objets de manière statique (sur la pile au lieu de les allouer dans la mémoire du tas) et que vous effectuez des appels de fonction, ils sont «empilés».
Lorsqu'une portée (tout ce qui est délimité par {
et }
) est quittée (en utilisant return XXX;
, en atteignant la fin de la portée ou en lançant une exception), tout ce qui se trouve dans cette portée est détruit (les destructeurs sont appelés pour tout). Ce processus de destruction d'objets locaux et d'appel de destructeurs est appelé déroulement de pile.
Vous rencontrez les problèmes suivants liés au déroulement de la pile:
éviter les fuites de mémoire (tout ce qui est alloué dynamiquement qui n'est pas géré par un objet local et nettoyé dans le destructeur sera divulgué) - voir RAII référencé par Nikolai, et la documentation pour boost :: scoped_ptr ou cet exemple d'utilisation de boost :: mutex :: scoped_lock .
cohérence du programme: les spécifications C ++ indiquent que vous ne devez jamais lever d'exception avant qu'une exception existante n'ait été gérée. Cela signifie que le processus de déroulement de la pile ne doit jamais lever d'exception (soit utiliser uniquement du code garanti de ne pas lancer de destructeurs, soit tout entourer de destructeurs avec try {
et } catch(...) {}
).
Si un destructeur lève une exception pendant le déroulement de la pile, vous vous retrouvez au pays d'un comportement indéfini qui pourrait entraîner la fin inattendue de votre programme (comportement le plus courant) ou la fin de l'univers (théoriquement possible mais n'a pas encore été observé dans la pratique).
Dans un sens général, une pile "dérouler" est à peu près synonyme de la fin d'un appel de fonction et de l'éclatement ultérieur de la pile.
Cependant, spécifiquement dans le cas de C ++, le déroulement de la pile a à voir avec la façon dont C ++ appelle les destructeurs pour les objets alloués depuis le début de tout bloc de code. Les objets créés dans le bloc sont désalloués dans l'ordre inverse de leur allocation.
try
blocs. Les objets de pile alloués dans n'importe quel bloc (que ce soit try
ou non) sont sujets au déroulement lorsque le bloc se termine.
Le déroulement de la pile est un concept principalement C ++, traitant de la façon dont les objets alloués à la pile sont détruits lorsque sa portée est quittée (soit normalement, soit via une exception).
Disons que vous avez ce fragment de code:
void hw() {
string hello("Hello, ");
string world("world!\n");
cout << hello << world;
} // at this point, "world" is destroyed, followed by "hello"
Je ne sais pas si vous avez encore lu ceci, mais l'article de Wikipedia sur la pile d'appels a une explication décente.
Détente:
Le retour de la fonction appelée fera sortir le cadre supérieur de la pile, laissant peut-être une valeur de retour. L'action plus générale de faire sauter une ou plusieurs images de la pile pour reprendre l'exécution ailleurs dans le programme est appelée déroulement de pile et doit être effectuée lorsque des structures de contrôle non locales sont utilisées, telles que celles utilisées pour la gestion des exceptions. Dans ce cas, le cadre de pile d'une fonction contient une ou plusieurs entrées spécifiant des gestionnaires d'exceptions. Lorsqu'une exception est levée, la pile est déroulée jusqu'à ce qu'un gestionnaire soit trouvé, prêt à gérer (intercepter) le type de l'exception levée.
Certaines langues ont d'autres structures de contrôle qui nécessitent un déroulement général. Pascal permet à une instruction goto globale de transférer le contrôle d'une fonction imbriquée vers une fonction externe précédemment invoquée. Cette opération nécessite le déroulement de la pile, en supprimant autant de cadres de pile que nécessaire pour restaurer le contexte approprié pour transférer le contrôle à l'instruction cible dans la fonction externe englobante. De même, C a les fonctions setjmp et longjmp qui agissent comme des gotos non locaux. Common Lisp permet de contrôler ce qui se passe lorsque la pile est déroulée à l'aide de l'opérateur spécial dérouler-protéger.
Lors de l'application d'une continuation, la pile est (logiquement) déroulée puis rembobinée avec la pile de la continuation. Ce n'est pas la seule façon de mettre en œuvre des continuations; par exemple, en utilisant plusieurs piles explicites, l'application d'une continuation peut simplement activer sa pile et enrouler une valeur à passer. Le langage de programmation Scheme permet d'exécuter des thunks arbitraires à des points spécifiés lors du "déroulement" ou du "rembobinage" de la pile de contrôle lorsqu'une continuation est appelée.
Inspection [modifier |
J'ai lu un article de blog qui m'a aidé à comprendre.
Qu'est-ce que le déroulement de la pile?
Dans tous les langages qui supportent les fonctions récursives (c'est-à-dire à peu près tout sauf Fortran 77 et Brainf * ck), le runtime du langage garde une pile des fonctions en cours d'exécution. Le déroulement de la pile est un moyen d'inspecter, et éventuellement de modifier, cette pile.
Pourquoi voudriez-vous faire ça?
La réponse peut sembler évidente, mais il existe plusieurs situations liées, mais subtilement différentes, où le déroulement est utile ou nécessaire:
- En tant que mécanisme de flux de contrôle d'exécution (exceptions C ++, C longjmp (), etc.).
- Dans un débogueur, pour montrer à l'utilisateur la pile.
- Dans un profileur, pour prélever un échantillon de la pile.
- Depuis le programme lui-même (comme à partir d'un gestionnaire de crash pour afficher la pile).
Ceux-ci ont des exigences légèrement différentes. Certains d'entre eux sont essentiels aux performances, d'autres non. Certains nécessitent la capacité de reconstruire des registres à partir d'une trame externe, d'autres non. Mais nous aborderons tout cela dans une seconde.
Vous pouvez trouver le message complet ici .
Tout le monde a parlé de la gestion des exceptions en C ++. Mais, je pense qu'il y a une autre connotation pour le déroulement de la pile et qui est liée au débogage. Un débogueur doit effectuer le déroulement de la pile chaque fois qu'il est censé aller à une image antérieure à l'image actuelle. Cependant, il s'agit d'une sorte de déroulement virtuel car il doit revenir en arrière lorsqu'il revient à l'image actuelle. L'exemple pour cela pourrait être les commandes up / down / bt dans gdb.
OMI, le diagramme ci-dessous dans cet article explique magnifiquement l'effet du déroulement de la pile sur l'itinéraire de l'instruction suivante (à exécuter une fois qu'une exception est levée qui n'est pas interceptée):
Dans la photo:
Dans le second cas, lorsqu'une exception se produit, la pile d'appels de fonction est recherchée linéairement pour le gestionnaire d'exceptions. La recherche se termine à la fonction avec le gestionnaire d'exceptions, c'est- main()
à- dire avec le try-catch
bloc englobant , mais pas avant de supprimer toutes les entrées avant celle-ci de la pile des appels de fonction.
Le runtime C ++ détruit toutes les variables automatiques créées entre throw et catch. Dans cet exemple simple ci-dessous, les lancers f1 () et les captures main (), entre les objets de type B et A sont créés sur la pile dans cet ordre. Lorsque f1 () lance, les destructeurs de B et A sont appelés.
#include <iostream>
using namespace std;
class A
{
public:
~A() { cout << "A's dtor" << endl; }
};
class B
{
public:
~B() { cout << "B's dtor" << endl; }
};
void f1()
{
B b;
throw (100);
}
void f()
{
A a;
f1();
}
int main()
{
try
{
f();
}
catch (int num)
{
cout << "Caught exception: " << num << endl;
}
return 0;
}
La sortie de ce programme sera
B's dtor
A's dtor
C'est parce que la pile d'appels du programme lorsque f1 () lance ressemble à
f1()
f()
main()
Ainsi, quand f1 () est sautée, la variable automatique b est détruite, puis quand f () est sautée, la variable automatique a est détruite.
J'espère que cela aide, bon codage!
Lorsqu'une exception est levée et que le contrôle passe d'un bloc try à un gestionnaire, le runtime C ++ appelle des destructeurs pour tous les objets automatiques construits depuis le début du bloc try. Ce processus est appelé déroulement de la pile. Les objets automatiques sont détruits dans l'ordre inverse de leur construction. (Les objets automatiques sont des objets locaux qui ont été déclarés auto ou register, ou non déclarés static ou extern. Un objet automatique x est supprimé chaque fois que le programme quitte le bloc dans lequel x est déclaré.)
Si une exception est levée lors de la construction d'un objet constitué de sous-objets ou d'éléments de tableau, les destructeurs ne sont appelés que pour les sous-objets ou éléments de tableau construits avec succès avant que l'exception ne soit levée. Un destructeur pour un objet statique local ne sera appelé que si l'objet a été construit avec succès.
Dans la pile Java, le déroulement ou le déroulement n'est pas très important (avec garbage collector). Dans de nombreux articles sur la gestion des exceptions, j'ai vu ce concept (déroulement de pile), en particulier ces rédacteurs traitent de la gestion des exceptions en C ou C ++. avec des try catch
blocs, nous ne devons pas oublier: pile libre de tous les objets après les blocs locaux .