Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.65/40: Рейтинг темы: голосов - 40, средняя оценка - 4.65
1 / 1 / 0
Регистрация: 25.08.2012
Сообщений: 47
1

Передача двумерного массива в функцию. Ошибка "invalid types for array subscript"

16.11.2014, 03:19. Показов 8079. Ответов 10
Метки нет (Все метки)

Author24 — интернет-сервис помощи студентам
Заранее прошу прощения, если тема уже поднималась, но я ничего дельного не нашел.
При попытке обратиться к элементам массива, указатель на который передается в функцию, выдает invalid types float[int] for array subscript
Что характерно, при вызове swap() из main, ошибки нет, значит я неправильно передаю массив?
Где ошибка?
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 <iostream>
using namespace std;
 
void swapel (float* Alloc,int Size)
{
    swap(Alloc[0][0], Alloc[Size-1][Size-1]);
}
 
 
int main()
{
    int Size=4;
    float A[Size];
    float B[Size][Size];
    cout<<"Enter Size of allocators: ";
    cin>>Size;
    cout<<"Enter A matrix (1-Size): ";
    for(int i=0;i<Size;i++)
        cin>>A[i];
 
    for(int i=0;i<Size;i++)
        for(int j=0;j<Size;j++)
            {
                if(i==j) B[i][j]=A[i]-j;
                else B[i][j]=i-A[j];
            }
 
    swapel(&B[0][0],Size);
    swap(B[0][0], B[Size-1][Size-1]);
    return 0;
    }
0
Лучшие ответы (1)
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
16.11.2014, 03:19
Ответы с готовыми решениями:

Ошибка error: invalid types 'int[int]' for array subscript
Дана строка,содержащая английский текст.Найти количество слов,начинающихся с буквы b. Выдает ошибку в 29 строке:error: invalid types...

Ошибка компиляции invalid types ‘int[int]’ for array subscript
Вот сам код. Не понимаю в чем может быть ошибка. #include &lt;iostream&gt; using namespace std; int main() { int N, M; ...

Ошибка: error: invalid types ‘int[int]’ for array subscript
Выпрыгивает ошибка: &quot;main.cpp: In function ‘int main()’: main.cpp:33:12: error: invalid types ‘int’ for array subscript if (a...

10
Диванный эксперт
 Аватар для Max Dark
2335 / 2006 / 961
Регистрация: 09.10.2013
Сообщений: 4,585
Записей в блоге: 4
16.11.2014, 03:34 2
Вы передаете в функцию swapel 1-мерный массив и пытаетесь внутри нее вызвать swap для элементов 2хмерного массива...

Добавлено через 5 минут
C++ Скопировано
1
2
3
4
void swapel (float** Alloc,int Size)
{
    swap(Alloc[0][0], Alloc[Size-1][Size-1]);
}
Добавлено через 1 минуту
C++ Скопировано
1
swapel(B,Size);
0
Вездепух
Эксперт CЭксперт С++
 Аватар для TheCalligrapher
12866 / 6731 / 1809
Регистрация: 18.10.2014
Сообщений: 17,035
16.11.2014, 03:50 3
Цитата Сообщение от Cra3y Посмотреть сообщение
C++ Скопировано
1
2
3
4
void swapel (float** Alloc,int Size)
{
  swap(Alloc[0][0], Alloc[Size-1][Size-1]);
}
Для двумерного массива объявленного как 'float B[Size][Size]' это, разумеется, работать не будет. Обычный двумерный массив несовместим с типом указатель-на-указатель.

Цитата Сообщение от Cra3y Посмотреть сообщение
swapel(B,Size);
Это даже не скомпилируется.

Добавлено через 2 минуты
Цитата Сообщение от mersinvald Посмотреть сообщение
Где ошибка?
Ошибка в том, что вы передаете в функцию указатель и почему-то полагаете, что вы с этим указателем сможете работать, как с двумерным массивом. Это бессмыслица.

Тут надо заметить, что у вас в программе используется нестандартное расширение компилятора. Вот такой код

C++ Скопировано
1
2
3
int Size=4;
float A[Size];
float B[Size][Size];
является нелегальным в языке С++. В С++ запрещается объявлять массивы неконстантного размера.

Однако, если ваш компилятор поддерживает такие массивы и вы не против воспользоваться этим нестандартным расширением, то передавать массив в вашу функцию можно так

C++ Скопировано
1
2
3
4
void swapel (int Size, float Alloc[Size][Size])
{
    swap(Alloc[0][0], Alloc[Size-1][Size-1]);
}
Вызывать эту функцию следует так

C++ Скопировано
1
swapel(Size, B);
Однако если вы хотите избавиться от этого нестанадартного расширения, то все придется делать по-другому...
0
1 / 1 / 0
Регистрация: 25.08.2012
Сообщений: 47
16.11.2014, 03:50  [ТС] 4
Цитата Сообщение от TheCalligrapher Посмотреть сообщение
является нелегальным в языке С++. В С++ запрещается объявлять массивы неконстантного размера.
А как правильно объявнять динамические массивы? И, соответственно, потом передавать?
Только учусь, лучше уж привыкать к хорошему сразу)

Компилятор gcc
0
Эксперт С++
 Аватар для hoggy
8972 / 4318 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
16.11.2014, 04:06 5
Цитата Сообщение от mersinvald Посмотреть сообщение
А как правильно объявнять динамические массивы?
Никак.
На языке с++ не существует динамических массивов.

Вы можете воспользоваться контейнерами, которые умеют динамику.
Либо через указатель работать с куском памяти, выделенном в куче.
0
1 / 1 / 0
Регистрация: 25.08.2012
Сообщений: 47
16.11.2014, 04:08  [ТС] 6
Цитата Сообщение от hoggy Посмотреть сообщение
контейнерами
Я б с радостью, но лаба по массивам.
Про второе можно поподробнее?
0
Вездепух
Эксперт CЭксперт С++
 Аватар для TheCalligrapher
12866 / 6731 / 1809
Регистрация: 18.10.2014
Сообщений: 17,035
16.11.2014, 04:32 7
Лучший ответ Сообщение было отмечено mersinvald как решение

Решение

Цитата Сообщение от mersinvald Посмотреть сообщение
А как правильно объявлять динамические массивы?
Как создать двумерный массив неконстантного размера в С++

Существует несколько способов. Рассмотрим их на примере двумерного массива 'array' размера N x M с элементами типа 'int'. При этом ни N, ни M не являются константами времени компиляции.

1. Простейший способ: вектор векторов

C++ Скопировано
1
vector<vector<int>> array(N, vector<int>(M));
В результате будет определен вектор векторов, представляющий двумерный массив N x M, заполненный изначально нулями.

Синтаксис доступа - привычный 'array[i][j]'.

Передача в функции - как 'vector<vector<int>> &'

C++ Скопировано
1
2
3
4
5
6
void swapel(vector<vector<int>> &array)
{
  int N = array.size();
  int M = array[0].size();
  swap(array[0][0], array[N - 1][M - 1]);
}
В этом случае размеры массива в функцию передавать не надо, так как вектор уже хранит свой размер внутри себя.

2. Моделирование двумерного массива через одномерный с пересчетом индексов

C++ Скопировано
1
int *array = new int[N * M]();
Это создаст одномерный массив размера N*M, заполненный нулями.

Чтобы доступиться к элементу по двумерному индексу [i][j] просто выполняем пересчет индексов по формуле 'i * M + j', т.е. доступаемся к элементу 'array[i * M + j]'. (На самом деле именно так работают встроенный тип "массив" в языках С и С++ для любой размерности массива.)

Передача в функцию - очевидным образом

C++ Скопировано
1
2
3
4
5
void swapel(int *array, int N, int M)
{
  swap(array[0], array[(N - 1) * M + (M - 1)]);
  // или просто swap(array[0], array[N * M - 1]);
}
Освобождение памяти

C++ Скопировано
1
delete[] array;
Понятно, что в этом способе вместо голого массива можно использовать 'std::vector'.

3а. "Рваный" массив с индивидуальным выделением памяти

C++ Скопировано
1
2
3
int **array = new int*[N];
for (int i = 0; i < N; ++i)
  array[i] = new int[M]();
Синтаксис доступа - привычный 'array[i][j]'.

Передача в функции

C++ Скопировано
1
2
3
4
void swapel(int **array, int N, int M)
{
  swap(array[0][0], array[N - 1][M - 1]);
}
Освобождение памяти

C++ Скопировано
1
2
3
4
for (int i = 0; i < N; ++i)
  delete[] array[i];
 
delete[] array;
Несложно догадаться, что этот способ - это практически то же самое, что и способ 1, только с голыми массивами вместо векторов.

3б. "Рваный" массив с общим выделением памяти

C++ Скопировано
1
2
3
4
int **array = new int*[N];
int *data = new int[N * M]();
for (int i = 0; i < N; ++i, data += M)
  array[i] = data;
Синтаксис доступа и передача в функции - как и в предыдущем способе.

Освобождение памяти

C++ Скопировано
1
2
delete[] array[0];
delete[] array;
Если внимательно посмотреть на способ 2 и способ 3б, то можно заметить, что они очень похожи. В обоих случаях сами данные хранятся в одномерном массиве размера N*M. Разница только в том, что в способе 2 вычисление финального индекса делается через полную формулу, а в способе 3б это вычисление частично уже выполнено для начала каждой строки двумерного массива и указатели на эти начала сохранены в дополнительном массиве указателей.

Добавлено через 11 минут
Цитата Сообщение от mersinvald Посмотреть сообщение
А как правильно объявнять динамические массивы?
Разумеется, вышеприведенные разглагольствования на тему динамических массивов вам нужны только в том случае, если вам действительно нужен массив неконстантного размера в программе. Т.е. если величина 'Size', например, вводится пользователем или читается из файла, т.е. заранее не известна.

Если же величина 'Size' жестко задана на стадии компиляции, то все эти усилия становятся ненужными. Просто объявите ее глобально как

C++ Скопировано
1
const int Size=4;
и смело пользуйтесь обычными встроенными массивами размера 'Size'.
8
3 / 3 / 0
Регистрация: 16.01.2014
Сообщений: 55
14.12.2014, 20:23 8
@TheCalligrapher, подскажите пожалуйста, какую функцию выполняют / что означают пустые скобки "()" в конце строки
C++ Скопировано
1
int *data = new int[N * M]();
Предполагаю, что это пустой список параметра конструктора объекта, типом которого является int[N * M]. Верно?
Если эти круглые скобки не указывать, не изменится ничего?
Спасибо.
0
19411 / 10030 / 2443
Регистрация: 30.01.2014
Сообщений: 17,695
14.12.2014, 22:08 9
Цитата Сообщение от SaShka K Посмотреть сообщение
Если эти круглые скобки не указывать, не изменится ничего?
Изменится. С ними память инициализируется значением по-умолчанию для данного типа. В случае int - это будет 0. Если их не писать, то инициализации не будет, а элементы будут содержать случайные мусорные значения.
2
3 / 3 / 0
Регистрация: 16.01.2014
Сообщений: 55
16.12.2014, 16:41 10
@DrOffset, я правильно понимаю - такая инициализация гарантируется стандартом?
0
19411 / 10030 / 2443
Регистрация: 30.01.2014
Сообщений: 17,695
16.12.2014, 17:07 11
Цитата Сообщение от SaShka K Посмотреть сообщение
я правильно понимаю - такая инициализация гарантируется стандартом?
Да. Нужна ссылка на параграф?

Добавлено через 17 минут
8.5/5
To zero-initialize an object or reference of type T means:
— if T is a scalar type (3.9), the object is set to the value 0 (zero), taken as an integral constant expression,
converted to T;
.........
— if T is an array type, each element is zero-initialized;
1
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
16.12.2014, 17:07
Помогаю со студенческими работами здесь

Ошибка в структуре: Invalid types `int[int]' for array subscript
Ошибка при работе со структурой:607 D:\Ø\ïðàêòèêà\8\lpr81111.cpp invalid types `int' for array subscript #include &lt;cstdlib&gt; ...

invalid types ‘double[int]’ for array subscript
Что компилятор от меня хочет? :cry: /mpi/5laba/main.cpp||In function ‘int main(int, char**)’:| /mpi/5laba/main.cpp|19|error: invalid...

Invalid types 'float[int]' for array subscript
Ошибка в строках cin&gt;&gt;x; p2*=x; p2*=x; invalid types 'float' for array subscript,как исправить? #include &lt;iostream&gt; #include...

Invalid types 'float[int]' for array subscript
Выдает ошибку invalid types 'float' for array subscript в 44 и 48 строке. Не пойму что не так. Подскажите пожалуйста #include...

Invalid types `float[int]' for array subscript
вот такой код, вроде всё написал правильно, но компилятор ругается на 19 строку, мол &quot;Invalid types `float' for array subscript&quot;....


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

Или воспользуйтесь поиском по форуму:
11
Ответ Создать тему
Новые блоги и статьи
Циклы for в Python
py-thonny 17.03.2025
Существует множество ситуаций, когда нам нужно выполнить одно и то же действие несколько раз. Цикл for в Python — настоящий рабочий конь для большинства программистов. Если вам нужно пройтись по всем. . .
Предсказание ветвлений - путь к высокопроизводи­тельному C++
NullReferenced 17.03.2025
В высокопроизводительном программировании на C++ каждый такт процессора на счету. Когда речь заходит о разработке систем с низкой задержкой — будь то высокочастотная торговля, обработка потоковых. . .
Паттерн CQRS в C#
UnmanagedCoder 17.03.2025
Создание сложных корпоративных приложений часто требует нестандартных подходов к архитектуре. Один из таких подходов — паттерн CQRS (Command Query Responsibility Segregation), предлагающий простую,. . .
Паттерн Цепочка ответственности в C#
UnmanagedCoder 17.03.2025
Цепочка ответственности — это поведенческий паттерн проектирования, который позволяет передавать запросы последовательно по цепочке потенциальных обработчиков, пока один из них не обработает запрос. . . .
Создаем микросервисы с NestJS, TCP и Typescript
run.dev 17.03.2025
NestJS — фреймворк, который значительно упрощает создание серверных приложений на Node. js. Его прелесть в том, что он комбинирует концепции ООП, функционального программирования и предлагает. . .
Гексагональная архитектура со Spring Boot
Javaican 17.03.2025
Если вы когда-нибудь сталкивались с ситуацией, когда внесение простых изменений в базу данных или пользовательский интерфейс заставляло вас переписывать весь код, то вы точно оцените элегантность. . .
Позиционировани­е Kafka Consumer и Seek-операции
Javaican 17.03.2025
Что же такое Consumer Seek в Kafka? По сути, это API-метод, который позволяет программно указать, с какой позиции (offset) Consumer должен начать или продолжить чтение данных из партиции. Без этого. . .
Python NumPy: Лучшие практики и примеры
py-thonny 17.03.2025
NumPy (Numerical Python) — одна из ключевых библиотек для научных вычислений в Python. Она превращает Python из просто удобного языка общего назначения в среду для проведения сложных математических. . .
Java Micronaut в Docker: контейнеризация с Maven и Jib
Javaican 16.03.2025
Когда речь заходит о микросервисной архитектуре на Java, фреймворк Micronaut выделяется среди конкурентов. Он создан с учётом особенностей облачных сред и контейнеров, что делает его идеальным. . .
Управление зависимостями в Java: Сравнение Spring, Guice и Dagger 2
Javaican 16.03.2025
Инъекция зависимостей (Dependency Injection, DI) — один из фундаментальных паттернов проектирования, который радикально меняет подход к созданию гибких и тестируемых Java-приложений. Суть этого. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru
Выделить код Копировать код Сохранить код Нормальный размер Увеличенный размер