Ответ 1
В обоих случаях, поскольку вы поймаете по ссылке, вы эффективно изменяете состояние исходного объекта исключения (который вы можете представить как находящийся в магический место памяти, которое останется действительным во время последующего размотки - 0x98e7058
в примере ниже). Тем не менее,
- В первом случае, поскольку вы ревертируете с помощью
throw;
(который, в отличие отthrow err;
, сохраняет исходный объект исключения с вашими изменениями в указанном "магическом месте" в0x98e7058
) будет отражать вызов для добавления() - Во втором случае, поскольку вы явно бросаете что-то, копировать из
err
будет создаваться, а затем забрасываться заново (в другом "волшебном месте"0x98e70b0
- потому что для всех компилятор знает, чтоerr
может быть объектом в стеке, который должен быть развязан, например,e
находился в0xbfbce430
, а не в "магическом месте" в0x98e7058
), поэтому вы потеряете производный класс -специфические данные во время копирования-сборки экземпляра базового класса.
Простая программа, иллюстрирующая, что происходит:
#include <stdio.h>
struct MyErr {
MyErr() {
printf(" Base default constructor, this=%p\n", this);
}
MyErr(const MyErr& other) {
printf(" Base copy-constructor, this=%p from that=%p\n", this, &other);
}
virtual ~MyErr() {
printf(" Base destructor, this=%p\n", this);
}
};
struct MyErrDerived : public MyErr {
MyErrDerived() {
printf(" Derived default constructor, this=%p\n", this);
}
MyErrDerived(const MyErrDerived& other) {
printf(" Derived copy-constructor, this=%p from that=%p\n", this, &other);
}
virtual ~MyErrDerived() {
printf(" Derived destructor, this=%p\n", this);
}
};
int main() {
try {
try {
MyErrDerived e;
throw e;
} catch (MyErr& err) {
printf("A Inner catch, &err=%p\n", &err);
throw;
}
} catch (MyErr& err) {
printf("A Outer catch, &err=%p\n", &err);
}
printf("---\n");
try {
try {
MyErrDerived e;
throw e;
} catch (MyErr& err) {
printf("B Inner catch, &err=%p\n", &err);
throw err;
}
} catch (MyErr& err) {
printf("B Outer catch, &err=%p\n", &err);
}
return 0;
}
Результат:
Base default constructor, this=0xbfbce430
Derived default constructor, this=0xbfbce430
Base default constructor, this=0x98e7058
Derived copy-constructor, this=0x98e7058 from that=0xbfbce430
Derived destructor, this=0xbfbce430
Base destructor, this=0xbfbce430
A Inner catch, &err=0x98e7058
A Outer catch, &err=0x98e7058
Derived destructor, this=0x98e7058
Base destructor, this=0x98e7058
---
Base default constructor, this=0xbfbce430
Derived default constructor, this=0xbfbce430
Base default constructor, this=0x98e7058
Derived copy-constructor, this=0x98e7058 from that=0xbfbce430
Derived destructor, this=0xbfbce430
Base destructor, this=0xbfbce430
B Inner catch, &err=0x98e7058
Base copy-constructor, this=0x98e70b0 from that=0x98e7058
Derived destructor, this=0x98e7058
Base destructor, this=0x98e7058
B Outer catch, &err=0x98e70b0
Base destructor, this=0x98e70b0
Также смотрите: