14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
|||||||||||
1 | |||||||||||
Преобразования указателей27.04.2021, 18:35. Показов 1280. Ответов 17
Метки нет (Все метки)
0
|
27.04.2021, 18:35 | |
Ответы с готовыми решениями:
17
Правила преобразования указателей Почему в сортировке указателей на объекты в вызове функции используются адреса объектов, а не указателей? Объяснить различия в работе указателей на целое число и указателей на const char (строки в стиле Си) Создать специфицированный шаблон функции, принимающей массив указателей на char и количество самих указателей |
"C with Classes"
|
|
27.04.2021, 18:56 | 2 |
DWORD** это своего рода хак, который из m имеющий тип Mammal * , делает тип DWORD ** , что бы потом можно было применить разыменование * и получит указатель на таблицу виртуальных функций.То есть в реализации класса Mammal в начале объекта есть указатель на таблицу виртуальных функций, язык C++ не дает стандартных способов доступа к нему, по этому нужно сделать из m указатель на указатель, что бы получить из него указатель на таблицу.
1
|
14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
|
27.04.2021, 19:26 [ТС] | 3 |
Спасибо за ответ. А m это в нем и хранится адрес таблицы виртуальных функций? То есть *m указывает на таблицу виртуальных функций, да? Что хранится в m- адрес памяти по которому лежит адрес таблицы вирт. функций, или сразу адрес таблицы вирт. функций?
0
|
зомбяк
1584 / 1218 / 345
Регистрация: 14.05.2017
Сообщений: 3,940
|
|||||||||||
27.04.2021, 19:35 | 4 | ||||||||||
kurlyak, https://ru.wikipedia.org/wiki/... ых_методов
Оттуда же Добавлено через 4 минуты В нормальном применении нужно вместо
1
|
14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
|||||||||||
27.04.2021, 20:10 [ТС] | 5 | ||||||||||
Да, я как раз и разбираюсь как работает компилятор, это у меня учебный пример. А можно по- подробнее что значит каждая звездочка в выражении:
То есть по идее вот так должно быть
0
|
зомбяк
1584 / 1218 / 345
Регистрация: 14.05.2017
Сообщений: 3,940
|
|
27.04.2021, 20:28 | 6 |
Нет, это значит несколько другое.
(DWORD*)m - просто меняем тип адреса с Mammal * на DWORD* *(DWORD**)m - представляем, что по адресу m лежит значение с адресом, по которому нужно перейти. Соответственно на (DWORD**) говорим о том, что у нас по адресу лежит адрес, а * в начале говорим "перейти по этому адресу, который там лежит"Добавлено через 3 минуты Если не понятно, повтори реализацию односвязного списка, там как раз иллюстрируются "прыжки по адресам"
1
|
14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
|
27.04.2021, 20:34 [ТС] | 7 |
А! Вот оно что! Я понял, спасибо. Значит m это указатель, те ячейка памяти, она хранит адрес другой ячейки памяти, а в этой другой ячейке- адрес таблицы вирт.функций. И две звездочки в скобках- это значит мы по цепочке указателей обращаемся ко второй ячейке памяти (которая хранит адрес таблицы вирт.функ.) и разыменовываем эту вторую ячейку памяти (то есть звездочка перед скобками) что бы получить из нее адрес таблицы вирт. функ. Вроде так.
0
|
"C with Classes"
|
||||||
27.04.2021, 20:43 | 8 | |||||
оно было бы так если бы
m не был указателем.
хорошая интерпретация!
1
|
14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
||||||
27.04.2021, 22:34 [ТС] | 9 | |||||
Я для наглядности сделал чуть чуть по другому, тоже работает, но теперь не ясно DWORD p1 это адрес второй ячейки памяти? Если да, то что значит (DWORD*)p1 - это значит что мы берем значение из второй ячейки памяти (то есть таблица виртуальных функций), или что? Что значит это преобразование из DWORD p1 в указатель (DWORD*)p1? Или это значит, что мы преобразовываем адрес p1 в указатель и извлекаем из него значение вроде разыменовываем?
0
|
19409 / 10028 / 2443
Регистрация: 30.01.2014
Сообщений: 17,678
|
|||||||||||
27.04.2021, 23:52 | 10 | ||||||||||
kurlyak,
Вообще же использование DWORD для хранения значения указателя здесь не нужно. Зачем, если можно сразу использовать тип указателя, как сделано было в стартовом посте. Добавлено через 13 минут
Код
объект типа Mammal --------------------------------------------- | DWORD* vptr | ... | --------------------------------------------- | | V m - указатель на Mammal (Mammal*) | | V p - указатель на DWORD* (DWORD**) | | V *p - значение vptr (DWORD*)
1
|
14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
|||||||||||
28.04.2021, 12:31 [ТС] | 11 | ||||||||||
0
|
зомбяк
1584 / 1218 / 345
Регистрация: 14.05.2017
Сообщений: 3,940
|
|
28.04.2021, 12:39 | 12 |
1
|
14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
|||||||||||
28.04.2021, 13:38 [ТС] | 13 | ||||||||||
А почему эти два кода работают одинаково, несмотря что разное количество звездочек в указателях?
Так вы мне скажите, m хранит адрес таблицы вирт.функ. или m хранит адрес другой ячейки памяти, и эта другая ячейка памяти хранит адрес табл. вирт. функ.?
0
|
зомбяк
1584 / 1218 / 345
Регистрация: 14.05.2017
Сообщений: 3,940
|
|
28.04.2021, 13:40 | 14 |
Не совсем. Адрес может не поместиться в DWORD при компиляции в 64-битной системе, в результате при выполнении
*p у тебя часть адреса потеряется из-за переполнения. И первый вариант не сработает.
0
|
19409 / 10028 / 2443
Регистрация: 30.01.2014
Сообщений: 17,678
|
|
28.04.2021, 14:06 | 15 |
Как я уже выше сказал, m хранит адрес объекта типа Mammal. У этого объекта в составе есть указатель (на таблицу виртуальных функций), он располагается первым относительно начала объекта. Таким образом получается, что адрес всего объекта и адрес этого указателя численно совпадают. Поэтому мы можем сделать приведение m к адресу указателя (адрес указателя - это указатель на указатель, т.е. две звездочки). Когда мы разыменуем получившийся двойной указатель после этого приведения, мы получим значение адреса таблицы виртуальных функций и сможем использовать его для доступа к функциям.
Если вы внимательно перечитаете что выше было написано, то вы увидите, что вся эта информация там уже содержится.
1
|
14 / 14 / 8
Регистрация: 26.09.2007
Сообщений: 919
|
|||||||||||
28.04.2021, 18:01 [ТС] | 16 | ||||||||||
То есть мы указатель m приводим к (DWORD*) потом разыменовываем и еще раз приводим к (DWORD*) и получаем указатель на таблицу вирт.функ. как строчка кода ниже:
0
|
фрилансер
5848 / 5379 / 1103
Регистрация: 11.10.2019
Сообщений: 14,380
|
|
28.04.2021, 18:05 | 17 |
kurlyak, * разыменовывает то, что справа, с учётом приоритета операторов
если приоритеты не помнишь (я вот не всегда помню), его всегда можно точно задать при помощи скобок
1
|
зомбяк
1584 / 1218 / 345
Регистрация: 14.05.2017
Сообщений: 3,940
|
|
28.04.2021, 18:07 | 18 |
К разыменовыванию указателя на указатель. По смыслу ровно то же самое что первое, только после разыменования адрес помещается не в переменную типа
DWORD , а сразу в переменную типа DWORD *
1
|
28.04.2021, 18:07 | |
28.04.2021, 18:07 | |
Помогаю со студенческими работами здесь
18
Создать специализацию для шаблона, которая принимает массив указателей на строки и количество этих указателей Как обойтись без указателей и указателей на указатель? Различия указателей char* от указателей других типов Создание массивов указателей на массивы указателей Добавление нового указателя в конец массива указателей, удаление указанного элемента, добавление указателей Массив указателей на массив строк и сортировка массива указателей Искать еще темы с ответами Или воспользуйтесь поиском по форуму: |