Аватар для Lindemann66
4 / 4 / 5
Регистрация: 28.04.2010
Сообщений: 162

Отличие операции с выделением памяти и без

09.08.2011, 16:05. Показов 6464. Ответов 60
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Всем привет!

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

Вот нужно, допустим, создать массив из n элементов

Сделать это можно 2-мя способами


C++
1
int arr[n];

и
C++
1
int *arr = new int[n];
Но в одном случае создание массива сопровождается выделением памяти, а в другом - без
В чём принципиальная разница?
Я понимаю, в Qt сразу написано - объекты QObject необходимо создавать с выделением памяти, чтобы не было проблем
Понятно, так и делаем. А тут как правильней?
0
Лучшие ответы (1)
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
09.08.2011, 16:05
Ответы с готовыми решениями:

Стек, куча, хранение в памяти, динамическое выделение памяти, указатели в чем отличие?
Здравствуйте. Прочитал кучу определений но никак не пойму вообще что к чему. 1)Стек - это якобы кусок оперативной памяти который...

Ошибка с выделением памяти
Помогите с прогой. Какая то ошибка с выделением памяти. В free_result пишет, что вызвано исключение. Подскажите в чем ошибка. Заранее...

Конструктор с выделением памяти
Среди всех данных есть указатель, память для которого выделяется в конструкторе, необходимый размер передается в параметрах конструктора....

60
Эксперт С++
 Аватар для grizlik78
2383 / 1667 / 279
Регистрация: 29.05.2011
Сообщений: 3,402
09.08.2011, 17:16
Лучший ответ Сообщение было отмечено как решение

Решение

Студворк — интернет-сервис помощи студентам
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include <iostream>
#include <iomanip>
 
using namespace std;
 
int main()
{
    int const N = 20;
    int M = 10;
    unsigned char (*p)[N] = new unsigned char[M][N];
    unsigned char *p1 = (unsigned char *)p;
    for (int i = 0; i < M*N; ++i)
        p1[i] = i;
    for (int i = 0; i < M; ++i)
    {
        for (int j = 0; j < N; ++j)
            cout << setw(4) << (unsigned)p[i][j];
        cout << endl;
    }
    cout << endl << p[1]-p[0] << endl;
    return 0;
}
Добавлено через 3 минуты
Ну и в конце ещё
C++
1
delete [] p;
3
Заблокирован
09.08.2011, 17:19
Цитата Сообщение от grizlik78 Посмотреть сообщение
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include <iostream>
#include <iomanip>
 
using namespace std;
 
int main()
{
    int const N = 20;
    int M = 10;
    unsigned char (*p)[N] = new unsigned char[M][N];
    unsigned char *p1 = (unsigned char *)p;
    for (int i = 0; i < M*N; ++i)
        p1[i] = i;
    for (int i = 0; i < M; ++i)
    {
        for (int j = 0; j < N; ++j)
            cout << setw(4) << (unsigned)p[i][j];
        cout << endl;
    }
    cout << endl << p[1]-p[0] << endl;
    return 0;
}
Добавлено через 3 минуты
Ну и в конце ещё
C++
1
delete [] p;
Надо же! А кто-то говорил, что в С++ двумерных массивов нет!
0
 Аватар для oxotnik
1665 / 1134 / 80
Регистрация: 21.08.2008
Сообщений: 4,734
Записей в блоге: 1
09.08.2011, 17:21
Цитата Сообщение от Сыроежка Посмотреть сообщение
Как-то странно звучит, что в С++ нет двумерных массивов, а в С они есть.
А в голых сях они откуда взялись?

Добавлено через 1 минуту
Цитата Сообщение от Сыроежка Посмотреть сообщение
Надо же! А кто-то говорил, что в С++ двумерных массивов нет!
Любой массив в памяти располагается линейно, и не важно сколько у него измерений, хоть сто будет, все равно это одномерный.
0
Эксперт С++
1069 / 848 / 60
Регистрация: 30.04.2011
Сообщений: 1,659
09.08.2011, 17:25
Сыроежка, читайте стандарт... В С++ НЕТ многомерных массивов. Только одномерные. Именно поэтому при объявлении массива с многими размерностями, разрешается ТОЛЬКО ОДНУ - самую старшую - не указывать...
1
Заблокирован
09.08.2011, 17:25
Цитата Сообщение от oxotnik Посмотреть сообщение
А в голых сях они откуда взялись?

Добавлено через 1 минуту
Любой массив в памяти располагается линейно, и не важно сколько у него измерений, хоть сто будет, все равно это одномерный.
А в С они взялись из стандарта! Там даже есть константа, задающая максимальное количество размерностей массива! Вы очевидно путаете определение n-мерных массивов и то, как они располагаются в памяти. Как бы массивы не располагались в памяти, они остаются массивами!
0
Эксперт С++
 Аватар для grizlik78
2383 / 1667 / 279
Регистрация: 29.05.2011
Сообщений: 3,402
09.08.2011, 17:25
Цитата Сообщение от Сыроежка Посмотреть сообщение
Надо же! А кто-то говорил, что в С++ двумерных массивов нет!
Сыроежка, я говорил? Хотя рассматривать ли это как двухмерный массив или как массив массивов — это какой-то схоластический спор получится. Я в таком не участвую
Хотя здесь, очевидно, получился полудинамический двухмерный массив, так как второе измерение константного размера.
1
Заблокирован
09.08.2011, 17:27
Цитата Сообщение от ValeryLaptev Посмотреть сообщение
Сыроежка, читайте стандарт... В С++ НЕТ многомерных массивов. Только одномерные. Именно поэтому при объявлении массива с многими размерностями, разрешается ТОЛЬКО ОДНУ - самую старшую - не указывать...
Да, вот, я как раз стандарт читал! И как-то нигде не увидел, что в С или в С++ нет двумерных массивов. При этом ваш аргумент не состоятелен! Он не имеет никакого отношения к тому, есть массивы или нет. А не указывают самую старшую размерность потому, что массив в выражениях, за некоторыми исключениями, преобразуется в указатель на массив меньшей размерности! Только и всего!
0
Эксперт С++
1069 / 848 / 60
Регистрация: 30.04.2011
Сообщений: 1,659
09.08.2011, 17:29
Цитата Сообщение от Сыроежка Посмотреть сообщение
Надо же! А кто-то говорил, что в С++ двумерных массивов нет!
Нет.
У вас размерности - константные.
У вас массив из М элементов, каждый элемент которого - массив из N элементов...
0
Заблокирован
09.08.2011, 17:29
Цитата Сообщение от grizlik78 Посмотреть сообщение
Сыроежка, я говорил? Хотя рассматривать ли это как двухмерный массив или как массив массивов — это какой-то схоластический спор получится. Я в таком не участвую
Хотя здесь, очевидно, получился полудинамический двухмерный массив, так как второе измерение константного размера.
Да нет. Вы как раз в выражении new указали тип двумерного массива! То есть когда выделяется память с помощью new, то указывается тип того объекта, для которого выделяется память.

В противном случае не могли бы вы мне сказать, какой это тип char[ m ][ n ]?!

И где в этом выражении знак '*', который обязателен при задании типов указателей?! Куда звездочка-то пропала?! Я щас заплачу!
0
Эксперт С++
 Аватар для grizlik78
2383 / 1667 / 279
Регистрация: 29.05.2011
Сообщений: 3,402
09.08.2011, 17:31
Цитата Сообщение от ValeryLaptev Посмотреть сообщение
У вас размерности - константные.
У вас массив из М элементов, каждый элемент которого - массив из N элементов...
Только одна. В программе же видно.
0
Эксперт С++
1069 / 848 / 60
Регистрация: 30.04.2011
Сообщений: 1,659
09.08.2011, 17:34
Народ! Двумерный массив имеет РАВНОПРАВНЫЕ измерения. То есть в двумерном массиве я могу одинаковым образом работать и со строкой и со столбцом! Ну-ка попробуйте такое в С++.
Нет двумерных массивов в С++. Есть массив массивов. А это - не то же самое, что двумерный массив.

Добавлено через 1 минуту
Цитата Сообщение от grizlik78 Посмотреть сообщение
Только одна. В программе же видно.
Попробуйте сделать две...
В том-то и фокус...
Для массива n размерностей n-1 размерность должна быть константной...
0
09.08.2011, 17:34

Не по теме:

Я ещё раз обращаю внимание, что отказываюсь участвовать здесь в схоластических спорах. Место им в холиварах.

2
Заблокирован
09.08.2011, 17:48
Цитата Сообщение от ValeryLaptev Посмотреть сообщение
Народ! Двумерный массив имеет РАВНОПРАВНЫЕ измерения. То есть в двумерном массиве я могу одинаковым образом работать и со строкой и со столбцом! Ну-ка попробуйте такое в С++.
Нет двумерных массивов в С++. Есть массив массивов. А это - не то же самое, что двумерный массив.

Добавлено через 1 минуту


Попробуйте сделать две...
В том-то и фокус...
Для массива n размерностей n-1 размерность должна быть константной...
Вы не занимайтесь словоблудием! Ваше словоблудие к стандарту С и С++ никакого отношения не имеет!

Я вас уже спросил, что это за тип в выражении new

C++
1
new char[ m ][ n ]
ВЫ знаете, как определяются типы указателей, которые являются производными от других типов? Синтаксис объявления указателей знаете? Ну, и что скажите про тип, заданный в выражении выше?

Как раз в выражении

char ( *p )[ m ] = new char[ n ][ m ];

слева задан указатель (звездочку видите в определе6нии?! ), а справа задан массив. Какой массив? Двумерный!

Если вы еще не понимаете, то я это выражение могу упростить

C++
1
char *p = new char[ m ];
Видите закономерность? Слева указатель, указывающий на первый элемент выделеннйо памяти под массив, а справа в выражении указан тип массива. Вот так по индукции, вы можете добраться до массива n-размерности., которая, если память не изменяет, обычно ограничена в Стандарте числом 16.


А ваши спекуляции по поводу того, что массив в памяти располагается линейно (а иначе его и расположить по другому нельзя, так как память линейна), никакого отношения к типу массив не имеют!

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

Добавлено через 8 минут
Вы говорили, что книги по С/С++ писали? Теперь я знаю, какие книги ни в коем случае рекомендовать нельзя!
0
09.08.2011, 17:52

Не по теме:

Цитата Сообщение от Сыроежка Посмотреть сообщение
Вы говорили, что книги по С/С++ писали? Теперь я знаю, какие книги ни в коем случае рекомендовать нельзя!
А Вы писали? От Вас даже кода здесь фиг дождёшься. Ну да, настоящие профессионалы забесплатно код не пишут.

1
Заблокирован
09.08.2011, 18:03
Я еще хотел бы добавить. Когда мы имеем выражение вида

char ( *p )[ m ] = new char[ n ][ m ];

то значением *p является массив меньшей размерности! То есть если вы зададите конструкцию большей размерности, то разыменованный указатель будет представлять собой тип массива размерностью на единицу меньше, чем исходный тип массива, на который указывает указатель.

То есть значением разыменованного указателя всегда является массив, если исходная размерность его была не меньше 2. Если исходный массив имел размерность 16, то после разыменования указателя тип выражения *p будет массив размерностью 15.

Добавлено через 4 минуты
Цитата Сообщение от grizlik78 Посмотреть сообщение

Не по теме:


А Вы писали? От Вас даже кода здесь фиг дождёшься. Ну да, настоящие профессионалы забесплатно код не пишут.

Я не хотел обидеть автора, просто пошутил. Это довольнот распространенное заблуждение, что, якобы, в С и С++ нет многомерных массивов. Только когда задаешь вопрос, а какой тип у выражения, являющего разыменованным указателем *p (когда исходный массив, например, имел размерность не меньше 3), то наступает тишина, или неудержимый полет фантазии.
0
Эксперт С++
1069 / 848 / 60
Регистрация: 30.04.2011
Сообщений: 1,659
09.08.2011, 19:05
Сыроежка, ссылки на пункты стандарта - в студию! Особенно там, где написано об ограничении количества размерностей...
C++
1
int ttt[2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2][2];
Количество размерностей = 20.
Компилятор в студии не возникает.
0
Делаю внезапно и красиво
Эксперт С++
 Аватар для Deviaphan
1313 / 1228 / 72
Регистрация: 22.03.2011
Сообщений: 3,744
09.08.2011, 19:38
Цитата Сообщение от Сыроежка Посмотреть сообщение
Если исходный массив имел размерность 16, то после разыменования указателя тип выражения *p будет массив размерностью 15.
Это ты вообще о чём??? Индексация с нуля. Поэтому размерность 16, но максимальный индекс - 15.
0
Эксперт С++
1069 / 848 / 60
Регистрация: 30.04.2011
Сообщений: 1,659
09.08.2011, 19:43
Сыроежка, ошибочка вышла.
то значением *p является массив меньшей размерности! То есть если вы зададите конструкцию большей размерности, то разыменованный указатель будет представлять собой тип массива размерностью на единицу меньше, чем исходный тип массива, на который указывает указатель.
Тип разыменованного указателя - не массив, а указатель на массив...

Добавлено через 1 минуту
Сыроежка, пока ссылки из стандарта по своим вопросам не покпажешь - не о чем разговаривать.
Напоминаю, что про массивы написано в пункте 8.3.4 Стандарта...
0
Делаю внезапно и красиво
Эксперт С++
 Аватар для Deviaphan
1313 / 1228 / 72
Регистрация: 22.03.2011
Сообщений: 3,744
09.08.2011, 19:47
Цитата Сообщение от Сыроежка Посмотреть сообщение
char ( *p )[ m ] = new char[ n ][ m ];
Я просто в шоке!
Я не могу понять, как происходит для этого выделение памяти. Но работает! Уже проверил.
Объясните плиз.)
0
Эксперт С++
 Аватар для grizlik78
2383 / 1667 / 279
Регистрация: 29.05.2011
Сообщений: 3,402
09.08.2011, 19:56
Цитата Сообщение от Deviaphan Посмотреть сообщение
Я не могу понять, как происходит для этого выделение памяти. Но работает! Уже проверил.
Ну как... Раз существует указатель на массив (то есть именно массив целиком является элементом), то должно быть и выделение памяти для массива из таких массивов.
Вот new char[n][m] и выделяет память под n элементов с типом char[m].
А сам указатель имеет тип
C++
1
char (*)[m];
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
09.08.2011, 19:56
Помогаю со студенческими работами здесь

Проблема с выделением памяти
Пишу вычислительную задачу thread'ы+OpenGL, при загрузке программы контролирую переменную AllocMemSize. Начиная с некоторого момента...

Проблема с выделением памяти
Всем привет! Нужна помощь. Создаю класс и при компиляции возникает ошибка. Не пойму причину. Помогите пожалуйста. Конструктор должен...

Ошибка с выделением памяти
выдает ошибку в выделении памяти. никто не подскажет почему? если что -задача звучит так примерно вводим веественные числа пока не...

Проблемы с выделением памяти
Здравствуйте. Есть контейнер QList &lt;Node *&gt; В Node свой QList &lt;Node *&gt;. Строю программное дерево разбора выражение. На небольших...

Ошибка в коде с выделением памяти
Не могу понять, в чем проблема? double** FuzzySet::operator&amp;&amp;(FuzzySet* _set) { int k = _set-&gt;set.size(); double** R =...


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

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

Новые блоги и статьи
Транскрипция 55-минутного видео через Whisper: WhisperDesktop облажался, спас Google Colab[
anaschu 01.06.2026
Понадобилось получить текст из свежезагруженного видео на YouTube. Казалось бы, задача на пять минут. Заняла полтора часа. Делюсь опытом — может кому пригодится последовательность решений. . . .
21 мат мед. Планы на развитие модели здравоСохранения
anaschu 01.06.2026
AnyLogic: план развития симуляционной модели рабочего коллектива — динамический абсентеизм, реальные данные, три сценария сравнения Продолжаю серию постов о дискретно-событийной модели рабочего. . .
20. Мат мед. Абсентеизм как отдельный тип простоя
anaschu 29.05.2026
Апдейт модели: исправленные баги, абсентеизм и новые механизмы Продолжаю развивать ранее описанную модель рабочего коллектива на AnyLogic. За последние несколько дней был проведён серьёзный. . .
19. здоровье, усталость и психотип работника влияют на производительность предприятия, и наоборот, производительность на здоровье, усталось и психотип
anaschu 28.05.2026
Дискретно-событийная модель рабочего коллектива на AnyLogic: здоровье, выгорание, психотипы и микростимуляция Привет, коллеги. Хочу поделиться итогами нескольких недель работы над симуляционной. . .
"Прокси" для последовательного порта
Eddy_Em 28.05.2026
Эту штуку написал я достаточно давно. Но сейчас вот понадобилось настроить датчик грозы, но при этом не отключать его от "метеодемона". Соответственно, надо запустить этот "прокси": метеодемон будет. . .
Рефакторинг программы уравнивания.
Massaraksh7 26.05.2026
Пример по предыдущей записи в блоге. Но, надо заметить, что, во-первых, там оптимизация не только математики, но и работы с базой данных, и с графами, а во-вторых, это ещё не всё.
Использование TThread в Lazarus для математических вычислений.
Massaraksh7 25.05.2026
Производя рефакторинг своих программ на предмет ускорения их работы, обратил внимание на такой аспект, как сокращение времени матвычислений. Дело в том, что приходится работать с большими матрицами. . .
Модель здравосохранения 18. Чем здоровее работник, тем быстрее выгорает
anaschu 24.05.2026
Имитационная модель корпоративного здравоохранения: что показывает математика Сегодня в модели рабочего коллектива на AnyLogic появились три новые механики — выгорание через накопленную усталость,. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru