プログラミングをしていて、次のような2種類の形式の関数をみることがあります。
void sum(int a, int b, int& c); //aとbの和をcに格納する
int sum(int a, int b); //aとbの和を返す
上の例のint &cは参照渡しされる引数という意味で、CやC++、PHPなどで同様です。
この、計算結果を参照渡しした引数に格納するのと、返り値に返すことの違いがよく分かりません。計算結果が複数ある場合は、前者の方が便利でしょう。しかし計算結果が1つだけなのに前者の方法を採っている関数も多く見かけます。個人的には、計算結果を返り値として返す後者の方法の方が感覚的に自然だと思うのですが、なぜ前者のような関数が存在するのか、教えていただきたく存じます。
No.1ベストアンサー
- 回答日時:
・複数の値を返したい
・↑の応用) 値とエラー等の情報を返したい
if ( MySqrt(a,b) ) { printf( "ERROR" ); } // a>0ならb=√a,戻り値0, a<=0ならbは無効で戻り値1
・戻り値として返すとパフォーマンスが悪い(C/C++の構造体等)
あるいは、関数内で変数に代入→戻ってきて呼出し側の変数に代入、というわずかな処理でも速くしたい。
・↑と関係して、同様の関数と引数の形を揃えた
構造体用が MyFunc1(a,b) / 数値用が b = MyFunc2(a) と見た目が変わらないように。
・戻り値の型を厳密に指定したい
double c = sum(a,b) ;等と暗黙の型変換をしないようにしたい。
・戻り値を返すサブルーチンが書けなかった言語からのなごり
以上、思いついたものを書いてみました。
No.2
- 回答日時:
> 前者のような関数が存在するのか
それはプログラム言語の仕様としてですか?
言語仕様として存在させなきゃ自由度が下がるじゃないですか。
C言語などは、とても自由度の高い言語なので、
関数をどう実現するかは、プログラマーの勝手です。
あまりに自由度の高いのでバグがとっても出やすいです。
なので、バグが少なくなるように、
コーディング規約を会社や部署ごとに決めます。
コーディング規約で前者を禁止すれば、書きませんし、
禁止しなければ書いても良いというだけでしょう。
No.3
- 回答日時:
#1で書かれていること以外でよく使うのは、値の参照と代入を同時に行いたい場合ですね。
例えば、aにbを加算するというような場合や、複数のエラーチェックで最大のエラーレベルを調べたい場合など。
No.4
- 回答日時:
一つのプログラム中で混在していると、気まぐれとしかいいようがないですね。
関数によっては、「結果」と呼ぶべきものと、「処理が正常であったか」を返す必要があります。たとえば、fopen() は「ファイルポインタ」も返さないといけないし、「オープンが正常だったか」も返さないといけない。そこで、ファイルポインタがNULLにならないことを利用して、エラー時にはNULLを返すという決めにすることで、返り値一つで済ませているわけです。でも、これではエラーの理由を返せません。
システム関連の関数では、グローバル変数 errno を使って理由を返すことが多いです。理由を知りたい場合は、#include <errno.h> した上で、errnoを参照します。
ユーザが書くプログラム中で定義する関数のうちの少なからずのものが「正常かどうか」を返す必要がある場合、グローバル変数などは使わず、返り値で「正常かどうか、エラーならどんなエラーか」を返すということに統一して、「結果」は必ず引数で返す。「正常かどうか」を返す必要のない関数でも、0でも返すことにしておいて必ず「結果」を引数で返すことにするのがよく行われるようです。
このルールのメリットとしては、「最初は正常かどうかを返す必要がなかったが、処理の中身が増えて正常かどうかおよびエラー理由を返す必要が出てきた」場合に、修正が楽なことです。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
似たような質問が見つかりました
- C言語・C++・C# C言語 3 2022/10/04 15:07
- C言語・C++・C# このプログラミングの問題を教えてほしいです。 キーボードからデータ数nとn個のデータを入力し、平均値 3 2022/12/19 22:51
- C言語・C++・C# 10人分の生徒の英語の点数{32,34,41,38,40,26,14,46,42,50} と数学の点 2 2022/05/26 21:31
- C言語・C++・C# 関数ポインタの高速化のメリット 7 2023/05/05 20:15
- C言語・C++・C# C言語の課題が出たのですが自力でやっても分かりませんでした。 要素数がnであるint型の配列v2の並 3 2022/11/19 17:41
- C言語・C++・C# c言語の問題です 課題1 (二分探索木とセット) 大きさ size の配列 array を考える。す 2 2023/01/10 21:08
- C言語・C++・C# C言語階乗の総和を求める 2 2023/03/04 23:31
- 大学・短大 C言語線形リストの問題です 3 2022/12/22 00:45
- C言語・C++・C# このプログラミングの問題を教えて欲しいです。 キーボードから整数kを入力し、kが配列aの中に何個存在 2 2022/12/19 22:50
- Java Java 配列<選挙> 4 2023/07/31 15:07
関連するカテゴリからQ&Aを探す
おすすめ情報
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
DWORDの実際の型は何でしょうか
-
long型の定数の末尾にLを付ける...
-
C言語の<ctype.h>のisspace()に...
-
構造体の要素すべてに対する四...
-
C++のfor文について
-
[C++]メンバ関数の仮引数について
-
visualstudio C# テキストボッ...
-
C++の(左辺値)参照を参照渡し
-
C++でboolにintの値を代入する...
-
typedef enumの使い方を教えて...
-
関数の実体定義にヘッダファイ...
-
main.c:7:43: warning: implici...
-
intとINTの違いは?
-
コンパイルすると error C1083 ...
-
ハンドルされていない例外が発...
-
値を返り値に返すのと参照渡し...
-
識別子が定義されていませんと...
-
2重定義って??
-
void func( void )について
-
【#define】 defineで定義した...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
DWORDの実際の型は何でしょうか
-
visualstudio C# テキストボッ...
-
2重定義って??
-
long型の定数の末尾にLを付ける...
-
C++のfor文について
-
typedef enumの使い方を教えて...
-
関数の実体定義にヘッダファイ...
-
変数の型を定義しなかった場合...
-
ハンドルされていない例外が発...
-
C++でboolにintの値を代入する...
-
main.c:7:43: warning: implici...
-
プログラムの中で別のmainを呼...
-
【#define】 defineで定義した...
-
void func( void )について
-
構造体の要素すべてに対する四...
-
C++の(左辺値)参照を参照渡し
-
構造体の宣言でエラーが出ます。
-
main()とint main(void)の違い
-
intとINTの違いは?
-
エラー「invalid conversion fr...
おすすめ情報