Как вернуть из функции несколько значений в с
Функция в C может возвращать только одно значение. Тем не менее, что если нам надо получить из функции сразу несколько значений? В этом случае мы можем использовать разные подходы. Основные из них — возвращение комплексного объекта, который инкапсулирует отдельные значения, либо использование выходных параметров.
Объединение возвращаемых значений
Наиболее простой способ возвратить из функции несколько значений — это определить структуру для хранения этих значений и возвратить ее. Например, нам надо получить из массива минимальное и максимальное значения. Для этого мы могли определить функцию, которая возвращает структуру с двумя значениями:
#include #include typedef struct < int min; int max; >MinMax; MinMax getMinMax(int* array, size_t length) < assert(length >1 && "Array length is invalid"); MinMax result; result.min = array[0]; result.max = array[0]; for(size_t i = 0; i < length; i++) < if(array[i] < result.min) result.min = array[i]; if(array[i] >result.max) result.max = array[i]; > return result; > int main(void) < int array[] = ; size_t length = sizeof(array) / sizeof(int); MinMax data = getMinMax(array, length); printf("min=%d\n", data.min); // min=2 printf("max=%d\n", data.max); // max=9 >
Здесь для возвращения из функции минимального и максимального значения массива определена структура MinMax, которую возвращает функция getMinMax. Это наиболее простой и очевидный способ возвращения нескольких значений. Однако он имеет свои минусы. Прежде всего нам надо специально определять структуру под возвращаемые значения. Во-вторых, при возвращении структуры происходит копирование ее значений в стеке, что увеличивает объем потребляемой памяти.
Выходные параметры
Другой способ возвратить из функции несколько значений представляют выходные параметры (out-параметры). Язык Си как таковой не имеет концепции выходных параметров функции (как например, C#), однако мы можем симулировать выходные параметры с помощью указателей:
#include #include void getMinMax(int* array, size_t length, int* min, int* max) < assert(length >1 && "Array length is invalid"); *min = array[0]; *max = array[0]; for(size_t i = 0; i < length; i++) < if(array[i] < *min) *min = array[i]; if(array[i] >*max) *max = array[i]; > > int main(void) < int array[] = ; size_t length = sizeof(array) / sizeof(int); int minVal = 0; int maxVal = 0; getMinMax(array, length, &minVal, &maxVal); printf("min=%d\n", minVal); printf("max=%d\n", maxVal); >
Общий механизм определения и передачи выходных параметров заключается в следующем. Во-первых, определяем в функции параметры-указатели:
void getMinMax(int* array, size_t length, int* min, int* max)
Здесь параметры-указатели min и max как представляют выходные параметры. Внутри функции нам не важны их начальные значения. Наоборот, функция устанавливает их значения.
При вызове функции определяются переменные, и их адреса передаются выходным параметрам:
int minVal = 0; int maxVal = 0; getMinMax(array, length, &minVal, &maxVal);
В данном случае выходным параметрам min и max передаются соответственно адреса переменных minVal и maxVal.
При использовании выходных параметров нам не обязательно определять дополнительные структуры. Мы избегаем изличнего копирования возвращаемых значений в стеке. Однако выходные параметры рахдувают определение функции, могут снизить ее читабельность, особенно когда таких параметров много.
вернуть Несколько значений из функции в C
Нужно возвратить из функции значения 2х переменных. На ум приходит только идея сделать массив с этими значениями и через return выдать указатель на него. Но может есть другой способ?
vdm ★★
28.12.06 05:20:13 MSK
Не нравится мне твоё «сделать массив», сдаётся мне что ты собрался ссылку на локальный массив возвращать. 🙂 Очень не рекомендую. 🙂
Вообще обычно в таких случаях функции передают указатель на массив или структуру, в которую она пишет результаты. А возвращает она в таком случае просто код завершения (типа успешно/неуспешно). См., например, man 2 fstat.
Teak ★★★★★
( 28.12.06 05:25:30 MSK )
Ответ на: комментарий от Teak 28.12.06 05:25:30 MSK
typedef struct < int one, int two >retval2; retval2 * f(. ) < retval2 * retval = NULL; . if (!error) < retval = malloc(sizeof(retval2)); retval->one = one; retval->two = two; > return retval; > int main(. ) < retval2 * retval; if (retval = f(. )) < printf("one = %i, two = %i\n", retval->one, retval->two); free(retval); > > --- что нить типа такого в голову не приходило. )))
Ex ★★
( 28.12.06 09:01:16 MSK )
> Нужно возвратить из функции значения 2х переменных. На ум приходит > только идея сделать массив с этими значениями и через return выдать > указатель на него. Но может есть другой способ? Так можно делать только если массив статический: int *foo() < static int a[] = ; return a; > Второй способ вернуть структуру: struct S foo(char c, double d) < struct S s; s.ch = c; s.dl = d; return s; >Третий способ, который обычно применяется во всех библиотеках, объявить массив или структуру во внешнем блоке и передавать в функцию их адреса.
shumer ★
( 28.12.06 09:40:31 MSK )
Можно возвращать структуру, но считается, что это не оч. хорошо, т.к. вся структура укладывается в стек: typedef struct < int one; int two; >retval2; retval2 f() < retval2 x; x.one=1; x.two=2; return x; >int main(int argc, char **argv) < retval2 y; y=f(); >Выделять память внутри функции считается не оч. правильным путём, насколько я понимаю. Так что наиболее правильным является передача функции указателя на структуру (или массив): typedef struct < int one; int two; >retval2; int f(reval2 *x) < x->one=1; x->two=2; return 1; //код возврата=всё ок > int main(int argc, char **argv) < retval2 y; if (!f(&y)) //обработка ошибки >
Davidov ★★★★
( 28.12.06 10:17:07 MSK )
Ответ на: комментарий от Davidov 28.12.06 10:17:07 MSK
>> Выделять память внутри функции считается не оч. правильным путём, насколько я понимаю. Так что наиболее правильным является передача функции указателя на структуру (или массив):
не спорю, попросили привести пример, мну привел то что первое в голову пришло )
Ex ★★
( 28.12.06 10:49:02 MSK )
Ответ на: комментарий от Ex 28.12.06 10:49:02 MSK
>не спорю, попросили привести пример, мну привел то что первое в голову пришло )
Да не, никаких претензий. Сам так иногда делаю. Более того, по-моему, в каких-то стандартных вызовах было нечто подобное.
Я так понимаю, что логика состоит в том, что выделением памяти и её освобождением должны быть в логически эквивалентных местах.
То есть, например, в случае наличия функций retval2 *alloc_s1() и int free_s2(retval2 *r), всё уже хорошо. Вот и пример, где это может быть оправдано.
Кстати, всё это уже начинает смахивать на ООП.
Davidov ★★★★
( 28.12.06 10:57:57 MSK )
Ответ на: комментарий от Davidov 28.12.06 10:57:57 MSK
таки ООП это парадигма программирования и не зависит от конкретной реализации в ЯВУ
Ex ★★
( 28.12.06 11:05:00 MSK )
Ответ на: комментарий от shumer 28.12.06 09:40:31 MSK
>Так можно делать только если массив статический: >int *foo() > < >static int a[] = ; > return a; >> Только надо быть очень аккуратным: при повторном вызове будет возвращён тот же указатель.
Davidov ★★★★
( 28.12.06 11:10:25 MSK )
Ответ на: комментарий от Davidov 28.12.06 11:10:25 MSK
>>Так можно делать только если массив статический:
>>int *foo() >>< >> static int a[] = ; >> return a; >>>
>Только надо быть очень аккуратным: при повторном вызове будет возвращён >тот же указатель.
Еще это несколько не thread-safe
anonymous
( 28.12.06 15:08:24 MSK )
void test_retval (int a,int b, int c, int *ret1,int *ret2) < . *ret1=. ; *ret2=. ; >. int r1,r2 test_reatval(1,2,3,&r1,&r2); .
xnix ★★
( 28.12.06 16:25:14 MSK )
Ответ на: комментарий от Ex 28.12.06 09:01:16 MSK
Ну и нахрена звать malloc без необходимости? Если только чтоб выпендриться.
Teak ★★★★★
( 28.12.06 17:38:44 MSK )
Ответ на: комментарий от shumer 28.12.06 09:40:31 MSK
> Так можно делать только если массив статический:
ужоснах, не слушайте его, дети. 🙂 функция станет нереентрабельной, приглашаю всех в гугль на тему чем это грозит.
Teak ★★★★★
( 28.12.06 17:40:11 MSK )
Ответ на: комментарий от Teak 28.12.06 17:40:11 MSK
> ужоснах, не слушайте его, дети. 🙂 функция станет нереентрабельной, приглашаю всех в гугль на тему чем это грозит.
Конечно так делать не следует, хотя и возможно. Вобщем я сказал «can», а ты имел ввиду «may» 🙂
Блин, простой вопрос растянули уже на чертову дюжину постов. Короче, vdm, передавай в функцию адрес структуры с двумя полями и не парься. Все остальное грязный хак.
shumer ★
( 28.12.06 17:52:21 MSK )
Ответ на: комментарий от shumer 28.12.06 17:52:21 MSK
Согласен, все посты в этой теме, кроме первого, моего, — лишние. 🙂
Teak ★★★★★
( 28.12.06 18:02:01 MSK )
Ответ на: комментарий от Teak 28.12.06 17:38:44 MSK
ув. тов. Teak прочтите пост выше.
Ex ★★
( 28.12.06 18:07:45 MSK )
Ответ на: комментарий от Ex 28.12.06 18:07:45 MSK
> ув. тов. Teak прочтите пост выше.
Не, Ex, извини, но ты там фигню написал. Заморачиваться со структурой в функции стоит только если ты ее всю будешь возращать (структурка небольшая). А создавать ее динамически, а потом возвращать указатель, имхо хреновая идея. Кто-то использующий твою функцию должен знать и помнить о том, что ему где-то free надо втыкать, лишний, тяжелый malloc, и ради чего все?
shumer ★
( 28.12.06 18:51:10 MSK )
Ответ на: комментарий от Teak 28.12.06 18:02:01 MSK
> Согласен, все посты в этой теме, кроме первого, моего, — лишние. 🙂
ты не указал решение с возвратом структуры. Оно непопулярно, потому что этого не было в ранних компиляторах и про это не написано в К&Р Но для 2-х полей оно оптимально
dilmah ★★★★★
( 28.12.06 18:59:37 MSK )
Ответ на: комментарий от dilmah 28.12.06 18:59:37 MSK
> Оно непопулярно, потому что этого не было в ранних компиляторах и про это не написано в К&Р
Вот хохма на эту тему, совершенно случайно наткнулся:
2.2: I heard that structures could be assigned to variables and passed to and from functions, but K&R I says not.
K&R I was wrong; they hadn’t actually learned C very well before writing the book. Later, Ritchie got a job at Bell Labs, and worked closely with the authors of C, allowing the 2nd edition of the book to be much more accurate. (Kernighan already worked at Bell Labs, as a video game developer.)
Как вернуть через функцию два значения в C++?
Суть : Допустим у нас есть функция, в результате выполнения действий внутри нее мы получаем значения переменных, которые нам необходимо вернуть. Как это сделать? И как при вызове присвоить эти значения двум переменным одновременно?
Отслеживать
220k 15 15 золотых знаков 120 120 серебряных знаков 233 233 бронзовых знака
задан 11 дек 2017 в 15:55
Vlad Kvochin Vlad Kvochin
614 1 1 золотой знак 6 6 серебряных знаков 15 15 бронзовых знаков
Передать указатель на одно из возвращаемых значений и записать по нему. Второе — через return, например.
11 дек 2017 в 15:56
Как этим указателем воспользоваться вне функции, можно пример?
11 дек 2017 в 15:57
Читайте учебники про указатели и ссылки
11 дек 2017 в 15:58
Используйте structured binding.
11 дек 2017 в 16:00
2 ответа 2
Сортировка: Сброс на вариант по умолчанию
- вернуть структуру или кортеж (пару) с двумя полями
- использовать передачу параметров по ссылке или через указатель.
Например, считать сумму и разность двух переменных:
std::pair sumdif(ina a, int b) < return pair(a+b,a-b); > void sumdif(int a, int b, int& sum, int& dif) < sum = a+b; dif = a-b; >. int a = 5, b = 7; auto p = sumdif(a,b); cout
int a = 5, b = 7, s, d; sumdif(a,b,s,d); cout
Дополнение через 3 года 🙂 - теперь можно использовать новый вариант синтаксиса структурного связывания:
std::pair sumdif(int a, int b) < return pair(a+b,a-b); > . int a = 5, b = 7; auto [s,d] = sumdif(a,b); cout
Отслеживать
ответ дан 11 дек 2017 в 15:57
220k 15 15 золотых знаков 120 120 серебряных знаков 233 233 бронзовых знака
Дополню ответ @Harry
Если под рукой компилятор с поддержкой 11 ого (и выше) стандарта, то можно воспользоваться std::tuple и вернуть его из функции, эдакое обобщение на std::pair для произвольноно числа элементов с разными типами
std::tuple f() < std::tuplet = < -1, 123 >; // . return t; > std::tuple g() < std::tuplet = < -1, false, 123 >; // . return t; >
Отслеживать
ответ дан 11 дек 2017 в 18:59
3,701 1 1 золотой знак 17 17 серебряных знаков 31 31 бронзовый знак
- c++
- функции
- структурное-связывание
-
Важное на Мете
Связанные
Похожие
Подписаться на ленту
Лента вопроса
Для подписки на ленту скопируйте и вставьте эту ссылку в вашу программу для чтения RSS.
Дизайн сайта / логотип © 2024 Stack Exchange Inc; пользовательские материалы лицензированы в соответствии с CC BY-SA . rev 2024.3.8.5973
Как вернуть несколько значений из функции в php?
В PHP нельзя вернуть несколько значений из функции, как это делается в некоторых других языках. return всегда возвращает только одно значение. Вы можете использовать массив для возврата нескольких значений из функции.
function myFunction() $value1 = 1; $value2 = 2; $value3 = 3; $value4 = 'four'; $value5 = [10, 11, 12]; return [ 'value1' => $value1, 'value2' => $value2, 'value3' => $value3, 'value4' => $value4, 'value5' => $value5, ]; > print_r(myFunction()); // => [ // => 'value1' => 1, // => 'value2' => 2, // => 'value3' => 3, // => 'value4' => 'four', // => 'value5' => [10, 11, 12], // => ];