Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
 
 
Рейтинг 4.68/22: Рейтинг темы: голосов - 22, средняя оценка - 4.68
137 / 126 / 14
Регистрация: 03.07.2012
Сообщений: 355
1

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

13.07.2012, 15:26. Просмотров 4177. Ответов 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
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
13.07.2012, 15:26
Ответы с готовыми решениями:

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

Уничтожается переменная по выходу из функции. так не должно быть
Все доброго времени суток! В процессе отладки программки возник один вопрос... Есть функция:...

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

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

21
Модератор
Эксперт Python
26655 / 13901 / 2644
Регистрация: 12.02.2012
Сообщений: 22,783
Записей в блоге: 1
13.07.2012, 15:28 2
Потому, что переменные, создаваемые оператором new - не локальные.
1
137 / 126 / 14
Регистрация: 03.07.2012
Сообщений: 355
13.07.2012, 15:41  [ТС] 3
Цитата Сообщение от Catstail Посмотреть сообщение
Потому, что переменные, создаваемые оператором new - не локальные.
Как же не локальные если она внутри другой функции.? Переменная temp[80] уничтожается после выхода а pn что остается?
В pn копируется строка temp и она не удаляется?
0
Модератор
Эксперт Python
26655 / 13901 / 2644
Регистрация: 12.02.2012
Сообщений: 22,783
Записей в блоге: 1
13.07.2012, 15:54 4
Локальные переменные создаются в стеке и освобождаются после выхода из функции. Переменные, создаваемые оператором new, "живут" не в стеке, а в куче. Куча глобальна. Поэтому указатель на массив из кучи можно вернуть из функции (оператором return) и использовать в другой функции.

Добавлено через 3 минуты
Цитата Сообщение от Intel~lect Посмотреть сообщение
Переменная temp[80] уничтожается после выхода а pn что остается?
- обе уничтожаются... Но не уничтожается блок памяти в куче, на который указывала pn. Этот указатель возвращается оператором return.
1
511 / 463 / 81
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 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
5473 / 4868 / 831
Регистрация: 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
137 / 126 / 14
Регистрация: 03.07.2012
Сообщений: 355
13.07.2012, 16:43  [ТС] 7
Спасибо всем что объяснили Вроде немного стало понятно. А то вот читаю книгу и главное никак не могу разобраться с этой задачей.

Добавлено через 36 минут
Цитата Сообщение от alsav22 Посмотреть сообщение
в область памяти размером strlen(temp) + 1, которая выделяется в динамической памяти
Тогда что находится в этой динамической памяти доступно в любом месте программы? Просто нужно знать ее адрес?
0
Нуб со стажем
36 / 34 / 10
Регистрация: 19.02.2012
Сообщений: 118
13.07.2012, 16:52 8
Цитата Сообщение от Intel~lect Посмотреть сообщение
Тогда что находится в этой динамической памяти доступно в любом месте программы? Просто нужно знать ее адрес?
Да. Если эта память не была возвращена в кучу
1
866 / 764 / 174
Регистрация: 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
5473 / 4868 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
13.07.2012, 17:30 10
Цитата Сообщение от kent Посмотреть сообщение
Да. Если эта память не была возвращена в кучу
Intel~lect, после того, как эта память перестанет быть нужной, необходимо её освободить :
C++
1
delete [] name;
, чтобы не было утечки памяти.

Добавлено через 9 минут
Цитата Сообщение от MrCold Посмотреть сообщение
Catstail, что то вы путаете
Просто не так выразился. Наверное, имелось ввиду, что указатель, который возвращает new, содержит адрес памяти не в области, предназначенной для хранения локальных переменных.
0
137 / 126 / 14
Регистрация: 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
866 / 764 / 174
Регистрация: 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
Модератор
Эксперт Python
26655 / 13901 / 2644
Регистрация: 12.02.2012
Сообщений: 22,783
Записей в блоге: 1
13.07.2012, 17:57 13
Я действительно не вполне удачно выразился. Указатель pn - локальная переменная. Длиной в 4 байта. Содержит адрес, указывающий в кучу. Этот адрес возвращается в вызывающую программу. Переменная pn (4 байта в стеке) уничтожается при выходе из функции.
1
511 / 463 / 81
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
13.07.2012, 18:20 14
Цитата Сообщение от MrCold Посмотреть сообщение
ПОСЛЕ выхода из функции она не доступна
Переменная с именем pn, естественно недоступна в main, потому что имеет локальную область видимости внутри функции getSymbol.

Но сама строка, динамически размещённая в куче, доступна из других функций.
0
5473 / 4868 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
13.07.2012, 18:27 15
Цитата Сообщение от Intel~lect Посмотреть сообщение
А вот еще вопрос есть. Если в getName() используется оператор new для выделения блока памяти а потом эту память нужно удалять. Но где? В main()?
Можно в main(), можно не в main(). В вашем коде в main().
0
866 / 764 / 174
Регистрация: 11.01.2012
Сообщений: 1,942
13.07.2012, 18:28 16
Цитата Сообщение от Schizorb Посмотреть сообщение
Но сама строка, динамически размещённая в куче, доступна из других функций.
Schizorb, пожайлуста, кодом если не трудно
0
511 / 463 / 81
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
13.07.2012, 18:31 17
MrCold, самый первый пост в теме. В функции main создаётся указатель, ему присваивается адрес строки, память под которую выделена в другой функции.
0
137 / 126 / 14
Регистрация: 03.07.2012
Сообщений: 355
13.07.2012, 18:33  [ТС] 18
Цитата Сообщение от alsav22 Посмотреть сообщение
Можно в main(), можно не в main(). В вашем коде в main().
Я это что спросил. Вот если строка динамически создается в одной функции, а удалять в другой. Ведь так и запутаться можно! Неудобно.
Да и забыть также можно про освобождение памяти.
0
866 / 764 / 174
Регистрация: 11.01.2012
Сообщений: 1,942
13.07.2012, 18:34 19
Schizorb, Ага, тогда понятно
0
5473 / 4868 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
13.07.2012, 18:40 20
Цитата Сообщение от Schizorb Посмотреть сообщение
Но сама строка, динамически размещённая в куче, доступна из других функций.
c помощью указателя, возвращённого оператором new,и ,предусмотрительно, сохранённого в указателе name.

Добавлено через 2 минуты
Цитата Сообщение от Intel~lect Посмотреть сообщение
Да и забыть также можно про освобождение памяти.
На совести программиста следить за этим, или использовать классы с автоматическим управлением памятью, которые сами за этим следят.
1
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
13.07.2012, 18:40

Заказываю контрольные, курсовые, дипломные и любые другие студенческие работы здесь.

Локальная переменная масив не создаеться в вызваной функции
Здравствуйте, Не укладывется в голове, после завершения рекурсий (т.е.контроль передаеться на...

Почему не запоминается переменная после первого вызова функции ?
В общем есть кнопки и одна последняя кнопка . На них поставил функции по клику менять значение...

Переменная запоминающаяся после выхода из программы
Подскажите можно ли задать какую то переменную в java чтобы она сохранилась и после перезапуска...

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


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

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

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