Форум программистов, компьютерный форум CyberForum.ru

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

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 36, средняя оценка - 4.69
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
#1

Понимание адреса, ссылки и указателя - C++

30.05.2012, 11:31. Просмотров 4890. Ответов 103
Метки нет (Все метки)

Гм... Конечно, стремно создавать подобную тему, однако, уж очень захотелось понять все это дело. Правда сколько раз ни пытался - не удавалось ни разу, возможно, потому что пытался разобраться самостоятельно (хотя, наверное, многие это постигают именно так). Но вот моя очередная попытка, которую решил так сказать обнародовать Поэтому очень рассчитываю на вашу помощь и поддержку.

Итак, начал практически с самого начала и уже сразу же зашел в тупик.
Ниже приведен легенький код, демонстрирующий расположение переменных в памяти компа.

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
25
26
27
28
29
30
31
// расположение переменных в памяти
#include <cstdio.h>
#include <cstlib.h>
#include <iostream.h>
 
using namespace std;
 
int main(int intArgc, char* pszArgs[])
{
int end;
int n;
long l;
float f;
double d;
 
// Вывод в шестнадцатеричном виде
cout.setf (ios::hex);
cout.unsetf (ios::hex);
 
// выводить адреса переменных
// по очереди, чтобы показать размер
// каждой переменной
cout << "--- = " << &end << "\n";
cout << "&n = " << &n << "\n";
cout << "&l = " << &l << "\n";
cout << "&f = " << &f << "\n";
cout << "&d = " << &d << "\n";
 
system ("PAUSE");
return 0;
}
Результат:

C++
1
2
3
4
5
--- = 0x22ff6c
&n  = 0x22ff68
&l  = 0x22ff64
&f  = 0x22ff60
&d  = 0x22ff58
А теперь цитата из книги:
Обратите внимание на то, что переменная n располагается ровно в 4 байтах от переменной end (---). Переменная l располагается еще на 4 байта ниже, а переменная типа double занимает 8 байт. Для каждой переменной выделяется память, необходимая для ее типа.
Для каждой переменной выделяется память, необходимая для ее типа.
Это мне известно. Но вот что меня занесло в тупик:

Разница, скажем, между 0x22ff6c и 0x22ff68 действительно 4. Но 4 чего - байта? А разве 4 в десятичной системе не 4? А в двоичной это разве не 100? Откуда тут 4 байта? Я понимаю, что там должно быть 4 байта, но не получается осознать, что 1 hex-единица у нас вдруг байт. Или это не так?

В общем, на таком банальном я запутался

Спасайте...
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
30.05.2012, 11:31
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Понимание адреса, ссылки и указателя (C++):

Изменение адреса указателя - C++
Здравствуйте. Обращаюсь к форуму по следующему вопросу: как можно изменить значение указателя? То есть, есть следующий код: ...

Динамическое указание адреса указателя - C++
мне нужно сделать int * a=(int*)0x123ff; cout&lt;&lt;a; динамически. Кто нибудь знает как это сделать?

Использование ссылки и указателя - C++
Доброго времени суток! Возник такой вопрос, есть код: #include &lt;iostream&gt; #include &lt;cmath&gt; using namespace std; int main() ...

Получение адреса объекта после обнуления указателя на этот объект - C++
Добрый вечер уважаемые программисты. Интересует следующий вопрос. Есть ли способ получить адрес объекта, после того, как указатель на него...

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

Возвращение ссылки или указателя на закрытый элемент класса. - C++
Всех уважаемых форуман. поздравляю с наступившим новым годом и прошу ответить на такой вопрос. Почему до сих пор современный C++ не...

Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Borkot
4 / 4 / 0
Регистрация: 05.05.2012
Сообщений: 183
Записей в блоге: 1
31.05.2012, 15:27 #46
Я почему спросил по поводу указателей
Укзатель
C
1
2
3
4
5
6
7
8
int px,*p;
p=&px//тут указателю *p присваивается адресс &px
//Это понятно
//Но во многих книгах указывается еще такое
*p=&px//причем без **P
//ИЛи оба тождевства верны но толко если?
p=&px;//Либо
*p=px;
Ладно пойду опять читать мануалы пока не пойму.Спасибо за ваши ответы.
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
31.05.2012, 15:32  [ТС] #47
alkagolik, большое спасибо за пример! Буду разбирать его.


Borkot,
Цитата Сообщение от Borkot Посмотреть сообщение
*p=&px//причем без **P
Да вот это по-моему бред. Что за книга такая?
alkagolik
Заблокирован
31.05.2012, 15:38 #48
Цитата Сообщение от The_Immortal Посмотреть сообщение
Да вот это по-моему бред. Что за книга такая?
да собственно ничего страшного. Правда этого просто не нужно. В данном случае p хранит адрес px, по которому позже записывает ее же адрес, т.е. значение px равно адресу px. Подвох в том что адреса не имеют отрицательных значений и следовательно представлены беззнаковым типом, а px объявлена как знаковое целое.
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
31.05.2012, 15:44  [ТС] #49
alkagolik,
Цитата Сообщение от alkagolik Посмотреть сообщение
Подвох в том что адреса не имеют отрицательных значений и следовательно представлены беззнаковым типом, а px объявлена как знаковое целое.
Именно!

То есть тут я был прав?
Цитата Сообщение от The_Immortal Посмотреть сообщение
хм, а тут ты по этому адресу (т.е. переменной p) присваиваешь значение этого адреса что ли
Deviaphan
Делаю внезапно и красиво
Эксперт C++
1286 / 1220 / 50
Регистрация: 22.03.2011
Сообщений: 3,744
31.05.2012, 15:47 #50
Цитата Сообщение от The_Immortal Посмотреть сообщение
Да вот это по-моему бред
Ты абсолютно прав. Вариант, когда ты был бы не прав, я чуть выше нарисовал, это когда int **p;
Поэтому шняжные книжки лучше не читать.)

Цитата Сообщение от The_Immortal Посмотреть сообщение
У меня просто в голове не укладывается, как две переменные могут сидеть по одному адресу... Или это чисто логический прием?
Щяс разжую.

есть код
C++
1
2
3
int v = 5; // адрес, допустим 0x0022ff08
int * p = &v; // адрес, допустим 0x0022ff04
int & r = v; // ВНИМАНИЕ , адрес r РАВЕН адресу v, т.е. 0x0022ff08
Т.е. в коде между использованием v и r нет никакой разницы, это одно и то же. Это не две переменные, расположенные по одному адресу, а одна и та же переменная. Ты сразу спросишь, зачем это нужно и я не менее сразу же отвечу кодом

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
struct A
{
    int v;
};
 
A a;
int & r = a.v;
// Теперь вместо a.v = 10 можно написать r = 10
 
 
void Func( int & r )
{
    r = r * r;
}
 
int b = 10;
Func(b);
 int c= b; // c = 100

По указателям.
р расположен по адресу 0x0022ff04 и по этому адресу записан адрес переменной v 0x0022ff08.

Добавлено через 1 минуту
Цитата Сообщение от alkagolik Посмотреть сообщение
В данном случае p хранит адрес px
Только нужно сделать оговорку, что это действительно только для 32 битной системы и, что компилятор будет страшно ругаться.
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
31.05.2012, 16:01  [ТС] #51
Deviaphan, по первому мини-кодику усе ясно, второй же код мой мозг пока не осилил - чуть позже еще попробую

Цитата Сообщение от Deviaphan Посмотреть сообщение
Ты сразу спросишь, зачем это нужно и я не менее сразу же отвечу кодом
Просто я, если честно, ответа не уловил И с трудом пока понимаю необходимость использования и ссылок и указателей (да и вообще их вместе взятых ). Ведь ссылка - частный случай указателя. Или наоборот...

Ну как бы это разве не равносильно:

C++
1
2
3
4
5
6
7
8
void Func( int r )
{
    r = r * r;
}
 
int b = 10;
Func(b);
 int c= b; // c = 100
?



Вот еще такой вопрос созрел: как можно вывести на экран то, что в данный момент хранит указатель?
К примеру:
C++
1
2
3
4
5
6
7
8
int x;
int *y;
 
y=&x;
 
//пытался так:
cout << "y= " <<&y << "\n"; //но так я получаю адрес, по которому расположен 
//этот указатель, а не то, какой адрес он хранит, т.е. адрес переменной x
alkagolik
Заблокирован
31.05.2012, 16:08 #52
Deviaphan, да что-то даже без варнингов.
indicator@indicator:~$ cat test.cc
#include <iostream>
#include <iomanip>

using namespace std;
typedef unsigned uint32;

int main() {

uint32 *p , x = 2;
p = &x;
*p = (uint32)&x;

cout << hex << *p << ' ' << &x << ' ' << x << endl;

return 0;
}
indicator@indicator:~$ g++ test.cc
indicator@indicator:~$ ./a.out
bf910e48 0xbf910e48 bf910e48
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
31.05.2012, 16:13  [ТС] #53
alkagolik, у меня ругается:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
#include <cstdlib>
#include <iostream>
 
using namespace std;
 
int main(int intArgc, char* pszArgs[])
{
int end;
int * pntr;
*pntr=&end; // main.cpp invalid conversion from `int*' to `int' 
 
system ("PAUSE");
return 0;
}
Deviaphan
Делаю внезапно и красиво
Эксперт C++
1286 / 1220 / 50
Регистрация: 22.03.2011
Сообщений: 3,744
31.05.2012, 16:15 #54
Цитата Сообщение от alkagolik Посмотреть сообщение
да что-то даже без варнингов
В исходной задаче используются указатели на знаковый int. То что без ворнингов, это косяк компилятора. Попробуй х64 версию собрать и будут тебе ворнинги. И не правильно работающая программа.

Добавлено через 50 секунд
Цитата Сообщение от alkagolik Посмотреть сообщение
*p = (uint32)&x;
И вообще, ты явное преобразование сделал. По условию вопроса там неявное преобразование.
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
31.05.2012, 16:16  [ТС] #55
С uint32 тоже ругается:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <cstdlib>
#include <iostream>
 
using namespace std;
typedef unsigned uint32;
 
int main(int intArgc, char* pszArgs[])
{
uint32 end=5;
uint32 * pntr;
*pntr=&end; // main.cpp invalid conversion from `uint32*' to `unsigned int' 
 
system ("PAUSE");
return 0;
}
alkagolik
Заблокирован
31.05.2012, 16:17 #56
The_Immortal, конечно ругается. у тебя изначально указатель указывает в неизвестность, хранит в себе мусор. ты его проинициализируй каким - то адресом, а уж потом используй. Ты пытаешься разыменовать мусор.

Добавлено через 59 секунд
Deviaphan, конечно. без преобразования и не скомпиляется. Грязные методы.
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
31.05.2012, 16:19  [ТС] #57
Цитата Сообщение от Deviaphan Посмотреть сообщение
И вообще, ты явное преобразование сделал. По условию вопроса там неявное преобразование.
Да, с явным ничего не говорит (не ругается), но прога у меня вылетает при запуске.

Добавлено через 58 секунд
alkagolik,
Цитата Сообщение от alkagolik Посмотреть сообщение
ты его проинициализируй каким - то адресом
Сделал
Deviaphan
Делаю внезапно и красиво
Эксперт C++
1286 / 1220 / 50
Регистрация: 22.03.2011
Сообщений: 3,744
31.05.2012, 16:19 #58
Цитата Сообщение от The_Immortal Посмотреть сообщение
Ну как бы это разве не равносильно:
Нет. Результатом твоего примера r не изменится.


Цитата Сообщение от The_Immortal Посмотреть сообщение
как можно вывести на экран то, что в данный момент хранит указатель
Так же, как и присвоить ему значение. cout << *у.


Цитата Сообщение от The_Immortal Посмотреть сообщение
Ведь ссылка - частный случай указателя. Или наоборот...
Между ними нет ничего общего. Это тебя запутали фразами типа "разыменованный указатель".

Фактически, основная разница в том, что ссылку нужно инициализировать при создании. Т.е. вместо указателя ты можешь написать 0, а вот с ссылкой это не пройдёт. А что использовать, ссылку или указатель, это уже дело вкуса. О тонкостях, связанных с оптимизацией кода, лучше не волнуйся.
The_Immortal
1551 / 487 / 8
Регистрация: 04.04.2009
Сообщений: 1,891
31.05.2012, 16:20  [ТС] #59
Цитата Сообщение от The_Immortal Посмотреть сообщение
Вот еще такой вопрос созрел: как можно вывести на экран то, что в данный момент хранит указатель?
Ребят, так а это как сделать?
alkagolik
Заблокирован
31.05.2012, 16:26 #60
Цитата Сообщение от The_Immortal Посмотреть сообщение
С uint32 тоже ругается:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <cstdlib>
#include <iostream>
 
using namespace std;
typedef unsigned uint32;
 
int main(int intArgc, char* pszArgs[])
{
uint32 end=5;
uint32 * pntr = &end; /************************* тут я дописал **************************/
*pntr=&end; // main.cpp invalid conversion from `uint32*' to `unsigned int' 
 
system ("PAUSE");
return 0;
}
5 букв

Добавлено через 1 минуту
Цитата Сообщение от The_Immortal Посмотреть сообщение
Ребят, так а это как сделать?
C++
1
2
3
4
5
6
7
8
9
int main() {
 
    uint32 *p , x = 2;
    p = &x;
 
    cout << p << ' ' << *p << endl;
    
    return 0;
}
Добавлено через 3 минуты
Цитата Сообщение от Deviaphan Посмотреть сообщение
Между ними нет ничего общего. Это тебя запутали фразами типа "разыменованный указатель".
разве? а не так?
int &x == int* const point;
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
31.05.2012, 16:26
Привет! Вот еще темы с ответами:

Ссылки и адреса - C++
Вот, где лучше всего использовать адреса и ссылки? Просто не много не понятня для чего это все. Вот например эту запись int mas; int*...

Адресное пространство, адреса, ссылки - C++
Добрый вечер. Я новичок в программировании и стал недавно читать туториалы по ссылкам. В общем, прилагаю цитату с одного сайта: ...

Как снять константность ссылки для передачи в функцию адреса - C++
Здравствуйте. Есть функция с сигнатурой: void func(const tm &amp; _tm); В теле функции надо вызвать: time_t _mkgmtime( struct tm*...

При передачи указателя на обьект ошибка,а при передаче ссылки на указатель нет. Почему? - C++
Hi All! class SomeObj { public: int x; }


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

Или воспользуйтесь поиском по форуму:
Yandex
Объявления
31.05.2012, 16:26
Закрытая тема Создать тему
Опции темы

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