Форум программистов, компьютерный форум, киберфорум
Наши страницы

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 13, средняя оценка - 4.92
Intel~lect
135 / 124 / 2
Регистрация: 03.07.2012
Сообщений: 355
#1

Почему локальная переменная не уничтожается после выхода из функции - C++

13.07.2012, 15:26. Просмотров 2077. Ответов 21
Метки нет (Все метки)

В функции создается строка char *pn = new char[ strlen(temp) + 1 ];. Мне непонятно почему при выходе из функции эта строка доступна в main()? Почему эта строка не удаляется после выхода из функции?
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
char *getName(void)
{
    char temp[80];
    
    cout << "Enter last name: ";
    cin >> temp;
 
    char *pn = new char[ strlen(temp) + 1 ];
    strcpy(pn, temp);
 
    return pn;
}
 
int main()
{
   char *name;
 
   name = getName();
}
Объясните, пожалуйста, что вообще там происходит.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
13.07.2012, 15:26
Я подобрал для вас темы с готовыми решениями и ответами на вопрос Почему локальная переменная не уничтожается после выхода из функции (C++):

Почему переменная "d" не была удалена из стека после выхода из функции? - C++
Почему переменная &quot;d&quot; не была удалена из стека после выхода из функции? #include&quot;stdafx.h&quot; #include&quot;iostream&quot; using namespace std; ...

Уничтожается переменная по выходу из функции. так не должно быть - C++
Все доброго времени суток! В процессе отладки программки возник один вопрос... Есть функция: void bignum_fromhex(bignum_digit_t* num,...

Почему выводит ошибку C4700, что локальная переменная неинициализированна, если она инициализированна - C++
Все это более чем странно. #include &lt;iostream&gt; #include &lt;locale.h&gt; #include &lt;time.h&gt; #include &lt;cstdlib&gt; using namespace std; ...

Локальная переменная внутри статической функции - C++
Всем доброго дня! Из данного примера выяснилось, что при обьявлении функции func_1() статической, переменная x внутри этой ...

Локальная переменная масив не создаеться в вызваной функции - C++
Здравствуйте, Не укладывется в голове, после завершения рекурсий (т.е.контроль передаеться на if(ii==1){return 0;}) наступает цикл for...

Почему в выводе выдаются предупреждения "warning C4101: sm: неиспользованная локальная переменная" - C++
// gd.cpp: определяет точку входа для консольного приложения. // #include &quot;stdafx.h&quot; #include &quot;stdlib.h&quot; #include &quot;math.h&quot; ...

21
Catstail
Модератор
23498 / 11606 / 1893
Регистрация: 12.02.2012
Сообщений: 18,945
13.07.2012, 15:28 #2
Потому, что переменные, создаваемые оператором new - не локальные.
1
Intel~lect
135 / 124 / 2
Регистрация: 03.07.2012
Сообщений: 355
13.07.2012, 15:41  [ТС] #3
Цитата Сообщение от Catstail Посмотреть сообщение
Потому, что переменные, создаваемые оператором new - не локальные.
Как же не локальные если она внутри другой функции.? Переменная temp[80] уничтожается после выхода а pn что остается?
В pn копируется строка temp и она не удаляется?
0
Catstail
Модератор
23498 / 11606 / 1893
Регистрация: 12.02.2012
Сообщений: 18,945
13.07.2012, 15:54 #4
Локальные переменные создаются в стеке и освобождаются после выхода из функции. Переменные, создаваемые оператором new, "живут" не в стеке, а в куче. Куча глобальна. Поэтому указатель на массив из кучи можно вернуть из функции (оператором return) и использовать в другой функции.

Добавлено через 3 минуты
Цитата Сообщение от Intel~lect Посмотреть сообщение
Переменная temp[80] уничтожается после выхода а pn что остается?
- обе уничтожаются... Но не уничтожается блок памяти в куче, на который указывала pn. Этот указатель возвращается оператором return.
1
Schizorb
510 / 462 / 16
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
Завершенные тесты: 1
13.07.2012, 15:55 #5
Память под строку выделяется в куче, совместно используемой всеми функциями. Такая память не освобождается по завершению функции, для этого надо вызывать delete.

А сам по себе указатель pn - локальная переменная, содержащий адрес первого элемента этой строки. return pn - просто возвращает значение этого адреса. Как в случае

C++
1
2
3
4
5
int foo()
{
    int x = 8;
    return x;
}
Тоже возвращается значение локальной переменной, только типа int.
1
alsav22
5437 / 4832 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
13.07.2012, 16:07 #6
Цитата Сообщение от Intel~lect Посмотреть сообщение
В pn копируется строка temp и она не удаляется?
Не в pn, а в область памяти размером strlen(temp) + 1, которая выделяется в динамической памяти, и адрес которой помещается в указатель pn. Этот адрес, после выхода из функции getName() нужно сохранить (так как pn - локальная переменная). Что и делается в строке: name = getName(); Теперь указатель name хранит адрес области памяти, куда скопировали temp.
1
Intel~lect
135 / 124 / 2
Регистрация: 03.07.2012
Сообщений: 355
13.07.2012, 16:43  [ТС] #7
Спасибо всем что объяснили Вроде немного стало понятно. А то вот читаю книгу и главное никак не могу разобраться с этой задачей.

Добавлено через 36 минут
Цитата Сообщение от alsav22 Посмотреть сообщение
в область памяти размером strlen(temp) + 1, которая выделяется в динамической памяти
Тогда что находится в этой динамической памяти доступно в любом месте программы? Просто нужно знать ее адрес?
0
kent
Нуб со стажем
36 / 34 / 2
Регистрация: 19.02.2012
Сообщений: 118
13.07.2012, 16:52 #8
Цитата Сообщение от Intel~lect Посмотреть сообщение
Тогда что находится в этой динамической памяти доступно в любом месте программы? Просто нужно знать ее адрес?
Да. Если эта память не была возвращена в кучу
1
MrCold
859 / 757 / 71
Регистрация: 11.01.2012
Сообщений: 1,942
13.07.2012, 17:18 #9
Цитата Сообщение от Catstail Посмотреть сообщение
Потому, что переменные, создаваемые оператором new - не локальные.
Catstail, что то вы путаете

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
#include <iostream> 
 
char getSymbol(void)
{
    char p;
 
    char *arr = new char [ 20 ];
    
   std::cout << "Enter a symbol :  ";
 
    std::cin >> p;
 
    return ( p );
}
 
int main()
{
 
   std::cout << getSymbol();
 
   std::cout << arr;   //   НЕОБЪЯВЛЕННЫЙ ИДЕНТИФИКАТОР
 
   return ( 0 );
}
Цитата Сообщение от Intel~lect Посмотреть сообщение
В функции создается строка char *pn = new char[ strlen(temp) + 1 ];. Мне непонятно почему при выходе из функции эта строка доступна в main()?
ПОСЛЕ выхода из функции она не доступна
1
alsav22
5437 / 4832 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
13.07.2012, 17:30 #10
Цитата Сообщение от kent Посмотреть сообщение
Да. Если эта память не была возвращена в кучу
Intel~lect, после того, как эта память перестанет быть нужной, необходимо её освободить :
C++
1
delete [] name;
, чтобы не было утечки памяти.

Добавлено через 9 минут
Цитата Сообщение от MrCold Посмотреть сообщение
Catstail, что то вы путаете
Просто не так выразился. Наверное, имелось ввиду, что указатель, который возвращает new, содержит адрес памяти не в области, предназначенной для хранения локальных переменных.
0
Intel~lect
135 / 124 / 2
Регистрация: 03.07.2012
Сообщений: 355
13.07.2012, 17:30  [ТС] #11
Цитата Сообщение от MrCold Посмотреть сообщение
ПОСЛЕ выхода из функции она не доступна
Нет. Она все таки доступна Правда другим способом.
C++
1
2
3
4
5
6
int main()
{
   char *name;
 
   name = getName();  // в getName() создается строка а в main() ее видно
}
А вот еще вопрос есть. Если в getName() используется оператор new для выделения блока памяти а потом эту память нужно удалять. Но где? В main()?
0
MrCold
859 / 757 / 71
Регистрация: 11.01.2012
Сообщений: 1,942
13.07.2012, 17:40 #12
Intel~lect,

после оперетора return функция завершает свою работу

C++
1
2
3
4
5
6
7
8
9
10
int main()
{
   char *name, *name2;
 
   name = getName();  // здесь работает 
.....................................  // здесь не работает 
 
name2 = getName();  // здесь СНОВА работает 
 
}
0
Catstail
Модератор
23498 / 11606 / 1893
Регистрация: 12.02.2012
Сообщений: 18,945
13.07.2012, 17:57 #13
Я действительно не вполне удачно выразился. Указатель pn - локальная переменная. Длиной в 4 байта. Содержит адрес, указывающий в кучу. Этот адрес возвращается в вызывающую программу. Переменная pn (4 байта в стеке) уничтожается при выходе из функции.
1
Schizorb
510 / 462 / 16
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
Завершенные тесты: 1
13.07.2012, 18:20 #14
Цитата Сообщение от MrCold Посмотреть сообщение
ПОСЛЕ выхода из функции она не доступна
Переменная с именем pn, естественно недоступна в main, потому что имеет локальную область видимости внутри функции getSymbol.

Но сама строка, динамически размещённая в куче, доступна из других функций.
0
alsav22
5437 / 4832 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
13.07.2012, 18:27 #15
Цитата Сообщение от Intel~lect Посмотреть сообщение
А вот еще вопрос есть. Если в getName() используется оператор new для выделения блока памяти а потом эту память нужно удалять. Но где? В main()?
Можно в main(), можно не в main(). В вашем коде в main().
0
13.07.2012, 18:27
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
13.07.2012, 18:27
Привет! Вот еще темы с ответами:

Почему указатель теряет свои значения после выхода с метода, память выделена динамически - C++
#include &lt;iostream&gt; using namespace std; void sqr( int* a, int length ); int main() { int *a = new int; int length =...

После выхода из функции, чтобы матрица не изменялась - C++
Как сделать так, чтобы передать матрицу в функцию, в функции матрицу изменить (поменять элементы матрицы симметрично относительно побочной...

Предотвращение продолжения работы потока после выхода из функции - C++
https://books.google.ru/books?id=1UXRAAAAQBAJ&amp;pg=PA47&amp;lpg=PA47&amp;dq=%D0%B0%D1%81%D1%81%D0%BE%D1%86%D0%B8%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D...

Локальная переменная - C++
Если в функции объявляется символ,например так: char f(void){ char ch; ch = getchar(); return ch; }


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
15
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2018, vBulletin Solutions, Inc.
Рейтинг@Mail.ru