以下のようなプログラム(DLLとEXE)を書いたのですが、変数の値がおかしくなる(●参照)箇所があります。原因がお分かりになりましたら、ご回答をよろしくお願い致します。
●mkdll.cppの、sub1()の(※1)までは、input[]が正しい値で入っているが、(※2)で値がおかしくなる。(※1)から(※2)までで、input[]は参照するだけです。
---test.cpp(EXE)---
…
main(){
int input[10],output[10];
int err;
CDLL DLL;
err = DLL.func(input,output);
…
return(0);
}
---test.cpp End---
---mkdll.h(DLL)---
…
class __declspec(dllexport) CDLL{
public:
int func(int *input, int *output);
private:
int sub1(int *input, int *output);
int sub2(int in, int out);
…
};
---mkdll.h End---
---mkdll.cpp(DLL)---
#include "mkdll.h"
…
int CDLL::func(int *input, int*output){
int i;
int error;
error = sub1(input, output);
return(error);
}
int CDLL::sub1(int *input, int *outout){
int i;
int in1, in2, out1, out2;
int err;
//(※1)
for(i=0; i<5; i++){
err = 0;
in1 = input[i*2];
in2 = input[i*2+1];
//(※2)
err = sub2(in1, out1);
if(err != 0)
return(err);
…
}
}
…
---mkdll.cpp End---
No.2ベストアンサー
- 回答日時:
ざっと見たところでは変数の値をおかしくするような要因はないようです。
回答No.1の指摘にある永続期間にしても、ローカル変数の値はスコープから外れるまで有効ですからDLL.func()の実行中にinput[]がなくなってしまうことはありませんし、DLLが実行されるのはDLLを呼び出しているプロセス内以外あり得ないのでローカル変数へのポインタが無効になってしまうこともありません。
質問の内容のコードはそのままでは実行できませんよね。これでは問題究明にも限度があります。いま手元にある問題が発生するコードから、問題とは関係ないと思われるコードをどんどん取り除いていって、実際に実行できて、なおかつ問題が発生するような最小限のコードを作成してみてください。
たいてはこの最小限のコードを作成する途上でうっかりミスの原因が見つかるものですが、どうしても見つからなければ、その「最小限のコード」を改めてここに掲示してみてください。
ご回答、ありがとうございました。そして、ソースを省略しすぎて大変申し訳ありませんでした。
結論から言いますと、おっしゃる通り、切り分け作業でコードをどんどん削除している中で、問題は解決しました。ただ、思いもよらない、というか、今でも何故それで解決したのか不明なのですが。
どのようなものかというと、EXEを作成するソースに、DLLからインポートするクラスのprivate変数を記述すると、問題が解決しました。インポートする側からはpublicだけ書いておけば良いと思っていたのですが…。
多分に、"たまたま"解決しただけの感もあるのですが(汗)、とりあえず動作するものを作るのが第一優先だったので、とても助かりました。ありがとうございました。
No.5
- 回答日時:
No4 ngsvx さんが回答されているように、output[] が添え字の上限を越えてしまって、input[] の領域を破壊しているような気がします。
「…」で省略されているあたりの処理で、output[] に対して添え字を越えてしまって、for 文をグルグル回してませんか?
# たとえば、こんな感じとか。。。は、さすがにないですかねぇ。。。
------------------------------------------------
for ( i = 0; i < 10; i++ )
{
output[i*2] = out1;
output[i*2+1] = out2;
}
------------------------------------------------
関係ないですけど、out1 って設定せずに sub2() に渡してしまっているようですけど、これは大丈夫なのでしょうか?
ローカル変数ですし、不定値が入っていそうです。
# sub2() コール前に、実は out1 に何か値が設定されていて、その部分が
# 省略されているだけなら良いのですけど。。。
【参考】
(以下、私が良く使う手です。ご参考までに。。。)
私の場合、配列 (の先頭要素へのポインタ) を関数に渡す場合には、一緒に要素数も渡すようにしています。
たとえばこんな感じです。
------------------------------------------------
……
int func
(
int *input,
int input_num, /* input[] の要素数 */
int *output,
int output_num /* output[] の要素数 */
);
……
int sub1
(
int *input,
int input_num, /* input[] の要素数 */
int *output,
int output_num /* output[] の要素数 */
);
……
------------------------------------------------
# fgets() のイメージですね。
そして、関数側では配列 (の先頭要素へのポインタ) と一緒に渡された要素数を使って for 文でグルグル回すと。。。
あと、今回の原因には直接関係なさそうですが、入力用にポインタを渡す場合には const をつけてあげると、万が一のときに役立ちますよ。
間違って、そのポインタの参照先を書き換えようとした場合には、コンパイラがエラーを出してくれますから。。。
たとえば、こんな感じかな。
------------------------------------------------
……
int func
(
const int *input, /* 関数内で値を更新しないことを保障 */
int *output
);
……
int sub1
(
const int *input, /* 関数内で値を更新しないことを保障 */
int *output
);
……
------------------------------------------------
# strcpy() とかのイメージですね。
ま、面倒なことはコンパイルに任せてしまえ!という私の性格のなせるところですかねぇ。。。
ご回答、ありがとうございました。
そして、テクニックのご教授、ありがとうございます。是非、参考にさせていただきます。
>関係ないですけど、out1 って設定せずに sub2() に渡してしまっているようですけど、これは大丈夫なのでしょうか?
>ローカル変数ですし、不定値が入っていそうです。
省略しすぎで申し訳ありません。
sub2()の中で、out1には値が代入されるだけなので、設定していませんでした。まずいでしょうか…。
No.4
- 回答日時:
>(※1)から(※2)までで、input[]は参照するだけです。
(※1)はループの外、(※2)はループの中にありますから、これだけのソースではわからないと思います。
(※2)以降のループ内で壊している可能性もあります。
想像ですが、配列outputの添え字上限を越えて書き込みをしたためinputの内容が破壊されているような気がします。
ご回答、ありがとうございました。
そして、ソースを省略しすぎで、申し訳ありませんでした。
ご指摘いただいた点について、outputをはじめとして配列の要素数以上に書き込んだり参照したりしていないかを確認してみました。
結論としては大丈夫でしたが、しっかり調査していなかったのは確かでしたので、再度新たな目で点検しなおすというきっかけを与えていただき、感謝いたします。
No.1
- 回答日時:
ローカル変数で宣言した変数を、プロセスの明確な永続期間指定せずにDLLで呼び出したって、何処のプロセスでDLLが実行されるかは、OS任せになるって事だけです。
プロセス(タスク)がスタックに積まれた時点でローカルの変数なんてどうなってるか分らないのでは?
しかも、このプログラムCの文法だからオブジェクトの永続性も無いです。
input[10],output[10];をオブジェクトの考え方でどこまで存続するか明確にすれば、値は保障されると思いますが。別解なら、宣言にstatic付けるとか。
早速のご回答、ありがとうございました。
関数の中で宣言したローカル変数の値は、その関数を抜
けたら保持されないというのはわかっていたのですが、
関数の中にいる時点でもそのようなことは起こり得るの
ですね。勉強になりました。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
似たような質問が見つかりました
- C言語・C++・C# c言語の問題です 3 2023/01/10 16:15
- C言語・C++・C# 宣言する関数の形が決まっている状態で、 str1とstr2の文字列をこの順に引っ付けてstrに保存し 2 2022/05/30 18:21
- Ruby パイソンプログラミング 2 2022/12/03 18:44
- FX・外国為替取引 mql4のコンパイルエラー箇所の修正お願いします。 1 2023/03/15 16:14
- C言語・C++・C# C言語のエラーについて 2 2022/07/11 13:56
- C言語・C++・C# c言語の問題の説明、各所ごとに 5 2023/07/26 11:03
- Java javaでのプログラム(配列)について質問です. 2 2022/10/14 22:27
- MySQL MYSQL エラー 2 2022/10/18 11:37
- C言語・C++・C# c言語の問題です 課題1 (二分探索木とセット) 大きさ size の配列 array を考える。す 2 2023/01/10 21:08
- Ruby 【JAVA】数字をひし形に出力するプログラムについて 2 2022/07/11 23:32
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・「みんな教えて! 選手権!!」開催のお知らせ
- ・漫画をレンタルでお得に読める!
- ・【選手権お題その2】この漫画の2コマ目を考えてください
- ・2024年に成し遂げたこと
- ・3分あったら何をしますか?
- ・何歳が一番楽しかった?
- ・治せない「クセ」を教えてください
- ・【大喜利】看板の文字を埋めてください
- ・【大喜利】【投稿~12/17】 ありそうだけど絶対に無いことわざ
- ・【選手権お題その1】これってもしかして自分だけかもしれないな…と思うあるあるを教えてください
- ・【穴埋めお題】恐竜の新説
- ・我がまちの「給食」自慢を聞かせてっ!
- ・冬の健康法を教えて!
- ・一番好きな「クリスマスソング」は?
- ・集合写真、どこに映る?
- ・自分の通っていた小学校のあるある
- ・フォントについて教えてください!
- ・これが怖いの自分だけ?というものありますか?
- ・スマホに会話を聞かれているな!?と思ったことありますか?
- ・それもChatGPT!?と驚いた使用方法を教えてください
- ・見学に行くとしたら【天国】と【地獄】どっち?
- ・これまでで一番「情けなかったとき」はいつですか?
- ・この人頭いいなと思ったエピソード
- ・あなたの「必」の書き順を教えてください
- ・10代と話して驚いたこと
- ・14歳の自分に衝撃の事実を告げてください
- ・人生最悪の忘れ物
- ・あなたの習慣について教えてください!!
- ・都道府県穴埋めゲーム
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
「指定されたキャストは有効で...
-
複数桁10進数の*桁目だけを抽出...
-
C言語での引数の省略方法
-
C言語 エラーの原因がわからな...
-
足して100になるような乱数のア...
-
実数の整数部,小数部の取得
-
#define _CRT_SECURE_NO_WARNIN...
-
数字列を3桁ごとにカンマで区切...
-
c言語の配列を使ってサイコロを...
-
入力を待たずにstdinの監視をし...
-
構造体の勉強中です 合計点の高...
-
C#のコンパイルエラーCS0120に...
-
ColorをRGBで指定する方法
-
c言語の問題です 課題1 (二分探...
-
四捨五入方法
-
C言語の配列をC++のvectorに高...
-
C言語の関数で戻り値を返す必要...
-
Win32APIで作るコンボボックス...
-
配列v2を逆順にしてv1に格納す...
-
CStringの配列要素を関数で受け...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
「指定されたキャストは有効で...
-
C言語での引数の省略方法
-
複数桁10進数の*桁目だけを抽出...
-
C言語 エラーの原因がわからな...
-
#define _CRT_SECURE_NO_WARNIN...
-
ラップ関数とはどんなものですか?
-
【C++】関数ポインタの使い方
-
if と配列の組み合わせ
-
(int *)の意味
-
実数の整数部,小数部の取得
-
acceptをalarmでタイムアウトさ...
-
C言語
-
std::set<int> で、ある値が何...
-
因数分解を行うプログラムについて
-
数字列を3桁ごとにカンマで区切...
-
(マルチスレッド)_beginthrea...
-
比較回数と交換回数表示について
-
式は定数値が必要です」という...
-
int型の変数値をバイト列として...
-
c言語の配列を使ってサイコロを...
おすすめ情報