вернуть Несколько значений из функции в 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#: Возврат значений
Методы, которые мы определяли в предыдущих уроках, заканчивали свою работу тем, что печатали на экран какие-то данные:
class App < public static void Greeting() < Console.WriteLine("Winter is coming"); >>
Пользы от таких методов не очень много, так как результатом их работы невозможно воспользоваться внутри программы. Рассмотрим это на примере.
Возьмем задачу обработки электронной почты. Когда пользователь регистрируется на каком-то сайте, то он может ввести email любым способом:
- Добавив случайно пробелы в начале или в конце _support@hexlet.io__
- Использовав буквы в разном регистре SUPPORT@hexlet.io
Если мы сохраним его в таком виде в базу данных, то пользователь, скорее всего, не сможет войти на сайт, так как будет вбивать адрес без пробелов и используя другой регистр символов. Чтобы этого не произошло, email нужно подготовить к записи в базу, привести его к нижнему регистру и обрезать пробельные символы по краям строки. Вся задача решается в несколько строчек:
class App < public static void Main(string[] args) < // В реальности email приходит из формы var email = " SuppORT@hexlet.IO"; // обрезаем пробельные символы var trimmedEmail = email.Trim(); // приводим к нижнему регистру var preparedEmail = trimmedEmail.ToLower(); Console.WriteLine(preparedEmail); // =>"support@hexlet.io" // здесь будет запись в базу данных > >
Этот код стал возможен только благодаря возврату значения. Методы Trim() и ToLower() ничего не печатают на экран (в консоль), они возвращают результат своей работы и поэтому мы можем записать его в переменные. Если бы они вместо этого печатали на экран, мы бы не могли присвоить результат их работы переменной. Как мы не можем сделать с определенным выше методом greeting() :
// C# будет ругаться что `Greeting()` ничего не возвращает // Код не заработает var message = App.Greeting();
Изменим метод Greeting() таким образом, чтобы он начал возвращать данные, вместо их печати. Для этого нам понадобится выполнить две правки:
- Описать тип возвращаемых данных. В нашем случае это строка string
- Выполнить возврат вместо печати на экран
class App < public static string Greeting() < return "Winter is coming!"; >>
Вместо void теперь написано string , потому что у метода есть возврат. Так мы указали C#, что результатом работы метода будет строка.
return – особая инструкция, которая берет выражение, записанное справа и отдает его наружу, тому коду, который вызвал метод. Как только C# натыкается на return , выполнение метода на этом завершается.
// Теперь этот код работает var message = App.Greeting(); // Мы можем выполнить какие-то действия над результатом Console.WriteLine(message.ToUpper()); // => "WINTER IS COMING!"
Любой код после return не выполняется:
class App < public static string Greeting() < return "Winter is coming!"; // Любой код ниже не выполнится никогда Console.WriteLine("Я никогда не выполнюсь"); >>
Даже если метод возвращает данные, это не ограничивает его в том, что он печатает. Кроме возврата данных мы можем и печатать:
class App < public static string Greeting() < Console.WriteLine("Я появлюсь в консоли"); return "Winter is coming!"; >> // Где-то в другом методе // И напечатает текст на экран и вернет значение var value = App.Greeting();
Возвращать можно не только конкретное значение. Так как return работает с выражениями, то справа от него может появиться почти все что угодно. Здесь нужно руководствоваться принципами читаемости кода:
class App < public static string Greeting() < var message = "Winter is coming!"; return message; >>
Здесь мы не возвращаем переменную, возвращается всегда значение, которое находится в этой переменной. Ниже пример с вычислениями:
class App < public static long DoubleFive() < // или return 5 + 5; var result = 5 + 5; return result; >>
В этом примере в определении метода использовался long так как возвращается целое число.
Вопрос на самопроверку. Что вернёт этот метод?
// Определение class App < public static int Run() < return 5; return 10; >> // Использование App.Run(); // ?
Задание
Реализуйте статический метод GetCurrentYear() в классе App , который возвращает наружу текущий год в виде числа
var year = App.GetCurrentYear(); Console.WriteLine(year); // выведет текущий год
Для выполнения этого задания, вам понадобится выполнить несколько действий:
- Получить текущую дату с помощью свойства DateTime.Now . Это свойство возвращает объект даты. В реальности у даты есть свойство Year , который выполняет нашу задачу. Но здесь мы хотим потренироваться
- Преобразовать дату в строку с помощью метода ToString(«yyyy-MM-dd») . В параметрах нужно указать желаемый формат строки: yyyy-mm-dd, например, 2021-11-05
- Извлечь из нее год с помощью метода Substring()
- Преобразовать год из строки в число с помощью метода Convert.ToInt32() и вернуть наружу
Упражнение не проходит проверку — что делать?
Если вы зашли в тупик, то самое время задать вопрос в «Обсуждениях». Как правильно задать вопрос:
- Обязательно приложите вывод тестов, без него практически невозможно понять что не так, даже если вы покажете свой код. Программисты плохо исполняют код в голове, но по полученной ошибке почти всегда понятно, куда смотреть.
В моей среде код работает, а здесь нет
Тесты устроены таким образом, что они проверяют решение разными способами и на разных данных. Часто решение работает с одними входными данными, но не работает с другими. Чтобы разобраться с этим моментом, изучите вкладку «Тесты» и внимательно посмотрите на вывод ошибок, в котором есть подсказки.
Мой код отличается от решения учителя
Это нормально , в программировании одну задачу можно выполнить множеством способов. Если ваш код прошел проверку, то он соответствует условиям задачи.
В редких случаях бывает, что решение подогнано под тесты, но это видно сразу.
Прочитал урок — ничего не понятно
Создавать обучающие материалы, понятные для всех без исключения, довольно сложно. Мы очень стараемся, но всегда есть что улучшать. Если вы встретили материал, который вам непонятен, опишите проблему в «Обсуждениях». Идеально, если вы сформулируете непонятные моменты в виде вопросов. Обычно нам нужно несколько дней для внесения правок.
Кстати, вы тоже можете участвовать в улучшении курсов: внизу есть ссылка на исходный код уроков, который можно править прямо из браузера.
Как вернуть два значения из метода c
Функция в 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#: Полное руководство
В программировании на C# часто возникает необходимость возвращать из метода не одно, а несколько значений. Это может быть необходимо, например, при работе с координатами в двухмерном пространстве, где метод должен вернуть и x, и y, или при выполнении операции, которая должна возвращать результат и статус операции. В этой статье мы подробно рассмотрим различные способы возврата нескольких значений из методов в C#.
Содержание показать
Использование out параметров
Первый и самый базовый способ вернуть из метода несколько значений — это использование out параметров. Эти параметры позволяют методам возвращать данные через аргументы, переданные в метод.
public void GetCoordinates(out int x, out int y) < x = 10; // предположим, это результат работы метода y = 20; // и это тоже >
Чтобы использовать этот метод, вам нужно объявить переменные, которые будут переданы в метод и в которые метод поместит возвращаемые значения:
int x, y; GetCoordinates(out x, out y); Console.WriteLine($"Coordinates: (, )");
Возврат кортежей
С появлением кортежей (tuples) в C# 7.0, возвращение нескольких значений стало гораздо проще и эстетичнее. Кортежи позволяют объединять несколько элементов в один тип данных.
public (int, int) GetCoordinates() < return (10, 20); >
Использование кортежей делает код более чистым и позволяет возвращать значения сразу:
var (x, y) = GetCoordinates(); Console.WriteLine($"Coordinates: (, )");
Создание структур и классов
Если вам нужно возвращать сложные данные или вы хотите повысить читаемость кода, вы можете создать структуру или класс для хранения возвращаемых значений.
public class Coordinates < public int X < get; set; >public int Y < get; set; >> public Coordinates GetCoordinates() < return new Coordinates < X = 10, Y = 20 >; >
Такой подход не только упрощает понимание того, что возвращает метод, но и позволяет вам добавить в класс или структуру дополнительную логику:
Coordinates coords = GetCoordinates(); Console.WriteLine($"Coordinates: (, )");
Использование словарей и коллекций
Для возврата наборов данных можно использовать словари или другие коллекции. Это особенно полезно, когда количество возвращаемых элементов динамично или их типы различны.
public Dictionary GetData() < var data = new Dictionary< < "Temperature", 23.5 >, < "Humidity", 60 >, < "Pressure", 1013.25 >>; return data; >
Пример использования словаря:
Dictionary weatherData = GetData(); Console.WriteLine($"Temperature: °C");
Использование ValueTuple для именованных значений
ValueTuple — это расширение для кортежей, которое позволяет использовать имена для элементов, делая код еще более понятным.
public (double Temperature, int Humidity, double Pressure) GetWeatherData()
Использование именованных кортежей улучшает читаемость кода:
var weather = GetWeatherData(); Console.WriteLine($"Temperature: °C, Humidity: %, Pressure: hPa");
В этой статье мы рассмотрели основные способы возврата нескольких значений из методов в C#. Выбор способа зависит от конкретной задачи и предпочтений программиста. Использование out параметров подходит для простых сценариев, кортежи и ValueTuple обеспечивают большую гибкость и удобство, а создание классов и структур является оптимальным выбором для сложных данных. Словари и коллекции отлично подойдут для работы с динамическими наборами данных. Понимание всех этих способов позволит вам писать более эффективный и понятный код на C#.