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

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

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 200, средняя оценка - 4.94
nill
9 / 9 / 0
Регистрация: 16.08.2009
Сообщений: 417
#1

Возвращение массива из функции - C++

23.08.2009, 09:20. Просмотров 26511. Ответов 37
Метки нет (Все метки)

Нужно вернуть массив типа int из функции


Как я понимаю сделать это можно только с помощью указателей
У меня это получилось но так как везде пишут что указатели очень сложная тема решил узнать правильно ли я все сделал
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
void straight(int*); // прототип функции
 
void straight(int* pmas_outs_straight){
 
pmas_outs_straight[0]=1;
pmas_outs_straight[1]=2;
 
}
 
main{
int straight_outs[]={0,0};
straight(straight_outs);
}
два вопроса

1)Обязательно ли присваивать значения каждому элементу массива который передаеться в функцию и потом изменяеться с помощью указателей ?
2) Как я понимаю ссылки вообще никак нельзя использовать для возвращения массива из функции ?
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
M128K145
Эксперт С++
8282 / 3501 / 143
Регистрация: 03.07.2009
Сообщений: 10,707
23.08.2009, 09:35     Возвращение массива из функции #2
Если я понял правильно, то вот посмотри этот
код
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
32
33
34
35
36
37
38
int* Fill(int* m, int n)
{
    std::cout<<"Введите элементы массива:\n";
    for(int i =0; i < n; ++i)
        std::cin>>m[i];
    return m;
}
int* Sort(int *m, int n)
{
    for(int j = 0; j < n - 1; ++j)
        for(int i = 0; i < n - 1; ++i)
            if(m[i] > m[i + 1])
            {
                int rab = m[i];
                m[i] = m[i + 1];
                m[i + 1] = rab;
            }
    return m;
}
void Out(int* m, int n)
{
    std::cout<<"Результат:\n";
    for(int i =0; i < n; ++i)
        std::cout<<m[i]<<' ';
}
int main()
{
    setlocale(LC_ALL,"Rus");
    int n;
    std::cout<<"Введите n:\n",std::cin>>n;
    int * mas = new int [n];
    Fill(mas, n);
    Sort(mas, n);
    Out(mas, n);
    delete []mas;
    system("pause");
    return 0;
}
XuTPbIu_MuHTAu
Эксперт С++
2222 / 737 / 10
Регистрация: 27.05.2008
Сообщений: 1,508
23.08.2009, 09:44     Возвращение массива из функции #3
Цитата Сообщение от nill Посмотреть сообщение
1)Обязательно ли присваивать значения каждому элементу массива который передаеться в функцию и потом изменяеться с помощью указателей ?
2) Как я понимаю ссылки вообще никак нельзя использовать для возвращения массива из функции ?
При работе с массивом опирайся на здравый смысл. В общем, с указателями сложного ничего нет это треп все. Указатель - это переменная,в которой хранится адрес.Еще указателя есть тип - с помощью него ты говоришь себе,компилятору и тому,кто читает код,что именно по этому адресу лежит.Массив - это кусок памяти,который разбит для нескольких элементов одного типа. Или несколько элементов одного типа идущих в памяти подряд. Поэтому полное описание массива включает в себя

1. адрес его начала
2. размер одного элемента
3. кол-во элементов.

В Си :
1.Адрес его начала ассоциируется с именем массива и является по своей сути указателем.
2.Размер одного элемента определяется исходя из типа,который ты объявил.
3.Кол-во элементов остается на усмотрение пользователя. Именно поэтому есть ошибки выхода за пределы.
По твоим вопросам.

Передавая в функцию указатель ты сообщаешь ей адрес начала массива и размер одного элемента одновременно. Размер - потому что сообщаешь тип, адрес начала - потому что указатель. Часто дополнительно передается еще одна переменная - размер массива,чтобы функция обладала всей информацией.

Для обработки массивов используют обычно циклы. Что тут еще сказать. Пробегаешь циклом от начала до конца и делаешь с каждым что надо.
zim22
depict1
276 / 141 / 2
Регистрация: 11.07.2009
Сообщений: 606
23.08.2009, 09:47     Возвращение массива из функции #4
если массив завернуть в структуру, то можно передавать/возвращать его по значению.
skvor
640KB мне хватило на всё.
118 / 49 / 2
Регистрация: 07.06.2009
Сообщений: 442
23.08.2009, 09:54     Возвращение массива из функции #5
В общем всё нормально, кроме использования термина "Функция возвращает"
Обычно под "возвращением" подразумевают данные которые "возвращаются" через имя функции оператором return.
a=f(); //функция f взвращает после своей работы некоторое занчение записываемое в переменную a

И, мне кажется, лучше в заголовке указывать не ссылку на тип элементов массива, а указывать то, что это и есть массив, и желательно указать длинну.
void f(int a[], const int length);
M128K145
Эксперт С++
8282 / 3501 / 143
Регистрация: 03.07.2009
Сообщений: 10,707
23.08.2009, 10:08     Возвращение массива из функции #6
действительно, я кое-что упустил. Вот теперь поменял, то что пропустил, вот исправленный
код
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
32
33
34
35
36
37
38
39
int* Fill(int n)
{
    int *m = new int [n];
    std::cout<<"Введите элементы массива:\n";
    for(int i =0; i < n; ++i)
        std::cin>>m[i];
    return m;
}
void Sort(int m[], int n)
{
    for(int j = 0; j < n - 1; ++j)
        for(int i = 0; i < n - 1; ++i)
            if(m[i] > m[i + 1])
            {
                int rab = m[i];
                m[i] = m[i + 1];
                m[i + 1] = rab;
            }
}
void Out(int* m, int n)
{
    std::cout<<"\nРезультат:\n";
    for(int i =0; i < n; ++i)
        std::cout<<m[i]<<' ';
}
int main()
{
    setlocale(LC_ALL,"Rus");
    int n;
    std::cout<<"Введите n:\n",std::cin>>n;
    int * mas = new int [n];
    mas = Fill(n);
    Out(mas, n);
    Sort(mas, n);
    Out(mas, n);
    delete []mas;
    system("pause");
    return 0;
}
valeriikozlov
Эксперт C++
4663 / 2489 / 321
Регистрация: 18.08.2009
Сообщений: 4,550
23.08.2009, 11:00     Возвращение массива из функции #7
Ссылки можно использовать для передачи массива параметром в функцию, но это немного сложнее чем использовать указатели.
Как вариант:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
 void straight(int&);//прототип функции
 
void straight(int& pmas_outs_straight){
 
(&pmas_outs_straight)[0]=1;
(&pmas_outs_straight)[1]=2;
 
}
 
main{
int straight_outs[]={0,0};
straight(*straight_outs);
}
nill
9 / 9 / 0
Регистрация: 16.08.2009
Сообщений: 417
23.08.2009, 11:47  [ТС]     Возвращение массива из функции #8
спасибо всем за ответы

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

skvor
А зачем это на что то может повлиять или просто для наглядности ?
skvor
640KB мне хватило на всё.
118 / 49 / 2
Регистрация: 07.06.2009
Сообщений: 442
23.08.2009, 12:01     Возвращение массива из функции #9
Цитата Сообщение от nill Посмотреть сообщение
skvor
А зачем это на что то может повлиять или просто для наглядности ?
Называйте вещи своими именами - меньше вероятность ошибки.
valeriikozlov
Эксперт C++
4663 / 2489 / 321
Регистрация: 18.08.2009
Сообщений: 4,550
23.08.2009, 12:12     Возвращение массива из функции #10
nill
В этом случае книга не права. В данном случае лучше пользоваться указателем (как в твоем коде).
Поясняю подробнее:
В моем коде сначала объявлен массив (наименование массива является указателем на сам массив, а если точнее на его первый элемент. Т.е. в переменной straight_outs находится адрес первого элемента массива). Затем я передаю в функцию straight() ссылку на первый элемент массива. Затем в самой функции straight() я беру адрес первого элемента массива (он является указателем на массив) и с его помощью присваиваю значения переменным массива.
В твоем коде делается то же самое, но проще:
Объявляешь массив. Указатель массива передаешь в функцию straight(), а в самой функции с помощью этого указателя на массив присваиваешь значения переменным массива.
Ну а какой вариант использовать, конечно решать только тебе.
Evg
Эксперт CАвтор FAQ
17387 / 5625 / 351
Регистрация: 30.03.2009
Сообщений: 15,407
Записей в блоге: 26
26.08.2009, 20:22     Возвращение массива из функции #11
valeriikozlov указал на нетривиальный для начинающих момент, но явно его не обозначил. Попробую выделить это место.

Код
C
1
straight(straight_outs);
визуально выглядит так, как будто бы весь массив целиком передаётся по значению. Но это не так. Имя переменной-массива без индексов означает адрес на самый первый (т.е. нулевой) элемент. Т.е. эта запись фактически означает

C
1
straight(&(straight_outs[0]));
mustardandrew
Человек
96 / 80 / 3
Регистрация: 16.06.2009
Сообщений: 229
26.08.2009, 20:51     Возвращение массива из функции #12
У меня наверное глупий вопрос.
Скажите пожалуста, а нет какой то функции в язике с++ для нахаждения размера масива,
как например в Delphi ( Length(arr) )?
Monte-Cristo
2786 / 1372 / 30
Регистрация: 07.03.2009
Сообщений: 4,446
26.08.2009, 21:02     Возвращение массива из функции #13
DreamWolf, если массив статический (не создан через new или malloc/calloc)
можно так:

C++
1
2
int Arr[10];
int size = sizeof(Arr)/sizeof(int);
другого варианта нет.
Evg
Эксперт CАвтор FAQ
17387 / 5625 / 351
Регистрация: 30.03.2009
Сообщений: 15,407
Записей в блоге: 26
26.08.2009, 21:08     Возвращение массива из функции #14
Monte-Cristo, лучше всё-таки

C
1
sizeof(Arr)/sizeof(Arr[0])
обычно пишут макрос

C
1
2
3
#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
...
int size = ARRAY_SIZE (Arr);
Monte-Cristo
2786 / 1372 / 30
Регистрация: 07.03.2009
Сообщений: 4,446
26.08.2009, 21:13     Возвращение массива из функции #15
Цитата Сообщение от Evg Посмотреть сообщение
Monte-Cristo, лучше всё-таки
ну да. Просто хотел показать что по-сути расчет считается исходня из типа.
mirso
524 / 342 / 17
Регистрация: 05.04.2009
Сообщений: 709
26.08.2009, 21:27     Возвращение массива из функции #16
Evg,
Цитата Сообщение от Evg Посмотреть сообщение
обычно пишут макрос
Цитата Сообщение от Evg Посмотреть сообщение
#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
а чего ни так?
C++
1
2
3
4
5
6
7
int s[5];
 
    for( size_t i = 0; i < sizeof(s)/sizeof(*s); i++ )
    {
        s[i] = i*3 + 2;
        std::cout << *(s + i) << std::endl;
    }
skvor
640KB мне хватило на всё.
118 / 49 / 2
Регистрация: 07.06.2009
Сообщений: 442
26.08.2009, 21:40     Возвращение массива из функции #17
Цитата Сообщение от DreamWolf Посмотреть сообщение
У меня наверное глупий вопрос.
Скажите пожалуста, а нет какой то функции в язике с++ для нахаждения размера масива,
как например в Delphi ( Length(arr) )?
Да/нет.
1. Строго, определить размер массива невозможно или не нужно.
2. В C++ есть стандартная библиотека, в ней есть всяки-разны шаблоны.
В т.ч. шаблоны vector и map. Для них есть "стандартный" метод size().
novi4ok
550 / 503 / 8
Регистрация: 23.07.2009
Сообщений: 2,359
Записей в блоге: 1
26.08.2009, 21:48     Возвращение массива из функции #18
не морочьте человеку голову.
все зависит от того, что предполагается делать с массивом в этой ф-ии. это нужно четко понимать.
самый надежный способ - это как делает майкрософт в своем api: память выделается в клиенте, и в ф-ю передается адрес этой памяти (первого элемента массива), максимальное количество элеметнов, на которое массив рассчитан, и указатель на счетчик, который вернет число действительных элементов массива. ф-я возвращает булево значение, означающее "получилось-неполучилось". как например:

C++
1
2
3
4
5
6
7
BOOL ReadFile(
  HANDLE hFile,                // handle to file
  LPVOID lpBuffer,             // data buffer
  DWORD nNumberOfBytesToRead,  // number of bytes to read
  LPDWORD lpNumberOfBytesRead, // number of bytes read
  LPOVERLAPPED lpOverlapped    // overlapped buffer
);
первый и последний параметр игнорируйте. остальное подходит как иллюстрация к случаю.

как альтернативу можно использовать vector (что я чаще всего и делаю). клиент создает вектор, резервирует памяти достаточно для предположительного количества элементов (иначе будут затраты ресурсов на выделение памяти под буфер вектора) и передает указатель на него в ф-ю. как:

C++
1
2
3
bool func (std::vector<int> *pDataVect){
...
}
обращение к ф-ии:

C++
1
2
3
std::vector<int> dataVect;
dataVect.reserve (ESTIMATED_SIZE);
bool ok = func (&dataVect);
если функция - в dll, только первый интерфейс подходит. второй может при определенных обстоятельствах создать проблемы, которые долго не поймете откуда происходят
Evg
Эксперт CАвтор FAQ
17387 / 5625 / 351
Регистрация: 30.03.2009
Сообщений: 15,407
Записей в блоге: 26
26.08.2009, 23:34     Возвращение массива из функции #19
Цитата Сообщение от mirso Посмотреть сообщение
а чего ни так?
Всё так, но макрос нагляднее
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
26.08.2009, 23:45     Возвращение массива из функции
Еще ссылки по теме:

C++ Как реализовать передачу массива в функцию и возвращение массива из функции?
C++ Возвращение из функции массива указателей
Возвращение массива из функции C++
Возвращение массива из функции C++
C++ Возвращение результата функции bool и использование его в функции main()

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

Или воспользуйтесь поиском по форуму:
Rififi
2336 / 1051 / 44
Регистрация: 03.05.2009
Сообщений: 2,656
26.08.2009, 23:45     Возвращение массива из функции #20
Цитата Сообщение от Monte-Cristo Посмотреть сообщение
DreamWolf, если массив статический (не создан через new или malloc/calloc)можно так:
А других и не бывает


Цитата Сообщение от Monte-Cristo Посмотреть сообщение
C++
1
2
int Arr[10];
int size = sizeof(Arr)/sizeof(int);
другого варианта нет.
А теперь - другой вариант (и, кстати, правильно работающий для всех типов, а не только для int) (:
C++
1
2
template <typename T, size_t N>
size_t array_size(const (&)[N]) { return N; }
Yandex
Объявления
26.08.2009, 23:45     Возвращение массива из функции
Ответ Создать тему
Опции темы

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