【初心者歓迎】C/C++室 Ver.103【環境依存OK】

■ このスレッドは過去ログ倉庫に格納されています
2018/07/01(日) 03:44:53.53ID:4MMF8cDN
エスケープシーケンスやWin32APIなどの環境依存なものもOK
そのような質問は必ず環境を書きましょう
半角空白やタブでのインデントはスレに貼ると無くなります

コードを貼れる所
http://codepad.org/
https://ideone.com/

前スレ
【初心者歓迎】C/C++室 Ver.102【環境依存OK】
http://mevius.5ch.net/test/read.cgi/tech/1509780815/
2018/09/02(日) 00:24:09.91ID:fnBlw2CS
>>334
それって単にプログラミングをする人の増加の問題では
337デフォルトの名無しさん
垢版 |
2018/09/02(日) 00:26:27.09ID:Tb3tt8fk
バカは自分で引用してる文章の意味がわかってない
ホントなかわいぞうなぐらい頭悪い
338デフォルトの名無しさん
垢版 |
2018/09/02(日) 00:27:46.09ID:Tb3tt8fk
バカはなんでとてつもなく自分がバカであるか
分かることは永遠にない
2018/09/02(日) 00:34:24.38ID:7WdK6EV6
>>333
mallocしないとできんのじゃないの?
Cだし
340335
垢版 |
2018/09/02(日) 00:36:08.13ID:PYGLVvbR
>>330
ごめん引用元のURL貼り忘れてた。
http://kikakurui.com/x3/X3010-2003-01.html
2018/09/02(日) 00:42:46.41ID:eH7Ohm4T
>>333
ヒープというのは動的に確保されたメモリのことでしょうか
ネストしたstd::arrayというのは、std::array<std::array<int, 2>, 3>というような意味で使っていました(リンク先のコードにある通り)
その場合メモリは動的確保されないと思いますが…

>>335
Cのその「配列オブジェクト」というのは多次元配列を一つのオブジェクトとして見るのではないということですか
ポインタと要素数を引数にとる関数に多次元配列を渡すコードをこれまで少なからず見てきた(教えられた)から勘違いしてたかも
342デフォルトの名無しさん
垢版 |
2018/09/02(日) 00:50:28.37ID:Tb3tt8fk
ヒープといのはオレがウソ書いてた
そのまま書いたら固定で配列がスタックにできる

アホが引用してるのは
あきらかに全然関係ない頭悪いのを引用してる
343デフォルトの名無しさん
垢版 |
2018/09/02(日) 00:54:46.07ID:Tb3tt8fk
array
┣array
┃┣ int
┃┗ int
┣array
┃┣ int
┃┗ int
┗array
 ┣ int
 ┗ int

こうなる
344デフォルトの名無しさん
垢版 |
2018/09/02(日) 01:03:53.61ID:Tb3tt8fk
アホが引用してる部分は

たとえばこんなソースがあった場合、

int* ai_aho[3] = {1, 2, 3}
int* pi_aho = ai_aho;

for (int i = 0; i < 3; ++i, ++pi_aho) {
 *pi_aho = 1;
}

pi_ahoはループを抜けたあと
pi_ahoはソース上適切でないアドレスをさしてるが
こいつを参照しなければ問題ないということが書いてある
2018/09/02(日) 01:18:27.35ID:PYGLVvbR
>>341
int a[N][M] に対して a + i するときの「配列オブジェクト」は要素型 int [M] で要素数 N の配列のこと。
N を超えて加算すると未定義動作になる。
a[0] + i するときの「配列オブジェクト」は要素型 int で要素数 M の配列のこと。
同じく M を超えて加算すると未定義動作になる。

ポインタを受け取る関数に a[0] あるいは &a[0][0] を渡した場合も M を超えて加算した場合は同じ理由で未定義動作になる。
期待した動作をすることも多いだろうけど、信頼性や移植性は損なわれる。
2018/09/02(日) 01:18:43.68ID:7WdK6EV6
おまえのたとえばは前科があるから信用できん
コピペしろコピペ
347デフォルトの名無しさん
垢版 |
2018/09/02(日) 01:20:32.56ID:Tb3tt8fk
(正)int ai_aho[3] = {1, 2, 3};
(誤)ai_aho[3] = {1, 2, 3}

目視で確認した
コレでいける
348デフォルトの名無しさん
垢版 |
2018/09/02(日) 01:23:43.36ID:Tb3tt8fk
な、低学歴知恵遅れはなにも分かってない
まずなにも分かってないのに規格読んで分かったふりしてるのがよく分かる
349デフォルトの名無しさん
垢版 |
2018/09/02(日) 01:25:42.40ID:Tb3tt8fk
低学歴知恵遅れが規格読んでも
規格なんか分かるわけがないからな

そもそも基本的にポインタがどう加算されるかすら分かってないのに
350デフォルトの名無しさん
垢版 |
2018/09/02(日) 01:43:11.24ID:Tb3tt8fk
uint32_t aho[4];
uint32_t* p_aho32 = aho;
uint8_t* p_aho_8 = (uint8_t*)aho;
p_aho32++;
p_aho_8+=4;

どっちのアドレスも同じになる

配列の場合どうなるか
考え方はまったく同じだな
あとはもう分かるハズだ
2018/09/02(日) 04:27:28.97ID:xXZGLywM
ロベール05-09の関数内に引きこもりって章のサンプルコードが長すぎて読めません(3ページ使用)
<algorithm>ってファイルを始めてincludeしてるにも関わらず其れに関して何ら説明ないし…
この章に関しては静的変数はグローバル変数と同じように使えてプログラムが終わるまでずっと同じメモリを参照している
と覚えられれば必要十分ですか?
何か簡単なコードを書いて教えてくれると助かります

よろしくお願いたします。
2018/09/02(日) 06:48:14.86ID:UYvvvhTW
あってるしコードは気にしなくていい
2018/09/02(日) 06:49:15.39ID:p4iMlD/t
例によってロベール著を読んでないし、下のコードはCだけど、
関数内static変数の説明なら、CもC++も変わらんだろう。
(行数節約のため詰め込み・省略してるので良いスタイルじゃない)

#include <stdio.h>

static void in_static(void) {
 static int cnt = 0; /* staticな変数 */
 cnt++;
 printf(" static %d 回目!\n", cnt);
}

static void not_static(void) {
 int cnt = 0; /* staticじゃない変数 */
 cnt++;
 printf("not_static %d 回目?\n", cnt);
}

int main(void) {
 in_static(); not_static();
 in_static(); not_static();
 in_static(); not_static();
 return 0;
}

こんな感じかな。
このコードには盛り込んでないけど、グローバル変数と違って
「関数内static変数には、基本的には関数の外からはアクセスできない」
という性質もある。
2018/09/02(日) 07:47:30.33ID:xXZGLywM
>>353
違う言語にも関わらず書いていただきありがとうございます
実際に実行してみてかなりスッキリしました
自動変数と静的変数の違い、しかと理解できました
関数についてるstaticも気になるところですが読み進めたいと思います
2018/09/04(火) 08:18:27.05ID:bj+tb6HS
char str[]={"aaaaaa","iiiiiiiii","uuuuuuu"};
は出来ないのに
char* str[]={"aaaaaaa",iiiiiii","uuuuuuu"};
が出来るのはなぜですか?
ポインタと配列は密接な関係にあるんじゃないんですか

よろしくお願い致します
2018/09/04(火) 11:38:18.11ID:ClV8CEAe
"aaaaaa" とか "iiiiiiiii" とか "uuuuuuu" がchar*型だから
char型の配列に入れようとしても無理な話
2018/09/04(火) 12:36:40.35ID:f5HJ/2BD
仮に出来たとしてどういう結果になって欲しいのだろう?
2018/09/04(火) 12:51:15.55ID:bz6n3SY3
char str[]={"aaaaaa"};はできる
char str[]={"aaaaaa","iiiiiiiii","uuuuuuu"};はできない
char *str[] = {"aaaaaa","iiiiiiiii","uuuuuuu"};はできる
char str[][9] = {"aaaaaa","iiiiiiiii","uuuuuuu"};ならできる

一番目は長さを指定しない配列変数の宣言で初期化により長さが指定されている。
二番目は{"aaaaaa","iiiiiiiii","uuuuuuu"}は文字列の配列なので変数の型と一致しないためエラーになる
三番目はポインタの配列なので配列の各要素を対応する文字列で初期化できる
四番目は配列の配列なので文字列の配列で初期化できる(ただし要素数の指定は必要)
2018/09/04(火) 14:22:07.00ID:JAXadswE
>>355
前者の[]は、文字の「列」を表すから
後者の[]は、文字列ポインタの「配列」を表すから

charはcharacterの略称、characterは文字、それが複数個集まった列が文字列
列となっているものが根本的に違う
2018/09/04(火) 17:31:45.30ID:bj+tb6HS
>>357
なんかこの違いを仕様で片付けて覚えるのはあまりよろしくないみたいなこと本に書いてあったので
違いが知りたかったです
>>356,358,359
ありがとうございます
文字列リテラルはポインタ型だったんですね
それなら代入出来ない理由も納得出来ます。スッキリしました
2018/09/04(火) 17:48:16.60ID:f5HJ/2BD
>>360
厳密にいえば文字列リテラルの型は配列だよ。 具体的に言えば const char[] ね。
暗黙の型変換で const char* として解釈される場合も多いってだけ。
362デフォルトの名無しさん
垢版 |
2018/09/04(火) 18:01:39.16ID:gZnu4stV
そのリテラル文字列はNULでターミネートされる
配列の長さが+1される

べつにNULターミネートしなくても
文字列は扱える

つまり長さが保存されないかわりに
NULでターミネートされてる
それをcでは文字列と呼称している
363デフォルトの名無しさん
垢版 |
2018/09/04(火) 18:04:32.71ID:gZnu4stV
そして、charの配列要素が
1つの文字を表してるとは当然限らない

MBCのように連続する複数の配列の要素が
1つの文字を表すことが多い

で、strlen()で文字の数が返却されないのは当然
2018/09/04(火) 18:52:35.55ID:UJ5UaD/4
>>360
曜日の出力もこれで簡単になる

int wday = 4;
printf("%c", "SMTWTFS"[wday]);
2018/09/04(火) 19:07:47.99ID:bz6n3SY3
>>364
それだと土曜日と日曜日の区別がつかないw
2018/09/04(火) 20:21:12.61ID:i0TPcXrC
月月火水木金金なので問題ないです
2018/09/04(火) 20:25:08.61ID:bz6n3SY3
>>366
それより毎日が日曜日の方が良いなあw
368デフォルトの名無しさん
垢版 |
2018/09/04(火) 20:43:24.10ID:gZnu4stV
ユリウス日に変換してから計算するから
問題ない
2018/09/05(水) 08:41:31.77ID:w5sEnOXo
>>365
火木「」
2018/09/05(水) 10:25:09.13ID:ZjqSDQmE
そいつをlongのポインタにする知恵があれば誉めてあげるんだが
2018/09/06(木) 23:16:14.27ID:xdo6cDUj
 int wday=5;
 char week[] = "SunMonTueWedThuFriSat";
 printf("%.3s",&week[wday*3]);

これでどうだw
2018/09/07(金) 06:40:53.97ID:Pk3Mmzkj
悪い例 (キャストの悪用、データサイズの勝手な仮定)
int wday = 5;
printf("%.4s\n", &((int*)"Sun.Mon.Tue.Wed.Thu.Fri.Sat.")[wday]);

軽いジョークのつもりで書いてみたけど、セグメンテーション・フォールトが出て
動くまでに予想外に手間取ってしもうた。
原因はアドレス演算子 & の付け忘れだった。
2018/09/07(金) 06:59:23.46ID:itoB4vJx
それを.3sで表示すれば371と同じ結果にならなかね?
2018/09/07(金) 07:46:30.50ID:Pk3Mmzkj
表示幅の指定で .3 なら >>371 と同じになるよ。
英単語の省略形にピリオドをつけたいのは好みの問題なの。

今は曜日も月名もピリオドを付けないのが一般的なのかな。
<time.h> の asctime() も3文字3文字だね。
375372
垢版 |
2018/09/07(金) 17:37:51.74ID:Pk3Mmzkj
たびたび失礼、そしてまるっきり実用の役に立たない話ですまぬ。

&((int*)"Sun.Mon.Tue.Wed.Thu.Fri.Sat.")[wday]
「いったん[]で配列形式でアクセスしてから要素のアドレスを得る」
なんて回り道をしないでも
(int*)"Sun.Mon.Tue.Wed.Thu.Fri.Sat." + wday
と素直にポインタの足し算だけでいけたね。


…難解Cプログラムに凝る人の気持ちがちょっと分かった気がする。
376デフォルトの名無しさん
垢版 |
2018/09/07(金) 22:40:13.74ID:B/yxkRYZ
intのサイズがcharのサイズの4倍でないと動かない
難解ではなくただの知恵遅れのコード
2018/09/08(土) 01:17:02.50ID:ejo+9snz
sizeofがイマイチ分からないです
型や変数のサイズを出す演算子ってのは分かったんですが
例えばこの例↓どういう風に化けてるんですか?
https://ideone.com/szVAAP
sizeof studentで要素数3だから3に化けると思うんですが、右の*studentは何に化けてるんですか?
2018/09/08(土) 01:27:45.49ID:VmsJpbI+
sizeof が返すサイズってのはバイト数だというのがよくわかってないのかな。
もう C++ では type_traits の std::extent を使って欲しい気持ち。
379デフォルトの名無しさん
垢版 |
2018/09/08(土) 01:39:35.14ID:j/6nk0eH
https://ideone.com/OduQG8

こんな風になる
2018/09/08(土) 01:56:15.85ID:ejo+9snz
>>378
分かってないです
ロベールの入門書読んでるんですが説明が少なすぎてこの27行目の式が分かりません。
構造型のバイト数も分かりませんし…int3個とchar一個だから13バイト?
>>379
すいません、分かりません
sizeof(student) / sizeof(Student);が何に化けてるのか検討も付きません

ありがとうございます。
381デフォルトの名無しさん
垢版 |
2018/09/08(土) 02:11:10.26ID:j/6nk0eH
https://ideone.com/aPMXLA
コレでなにが分からないか自分で説明できないなら
すべてを諦めたほうがいい
2018/09/08(土) 02:24:06.10ID:VmsJpbI+
>>380
int のサイズは処理系依存なので、具体的に何バイトとは言えない。
でもまあ現代的なパソコンなら 4 バイトのことがほとんどだろうね。
さらに言えば仕様上は構造体の要素の間にスキマがあいててもかまわない。

> int3個とchar一個だから13バイト?

この構造体中の char は MAX_NAME+1 個じゃん。

まあそれはそれとして、
sizeof student というのは、 student という配列の大きさを「バイト数で」返してきて、
sizeof *student というのは student 配列の要素ひとつの大きさをバイト数で返してくる。
つまり、配列全体の大きさを配列の要素の大きさで割ったら配列に含まれる要素の個数になる。
この場合は 3 ってこと。

sizeof *student が配列の要素ひとつを表現しているっていうのがちょっと分かり難いかなぁ。
2018/09/08(土) 02:52:28.11ID:OyJ8jP1h
>>377, >>380
前にも質問してた人だな
まず、sizeofはメモリ上に確保された対象のバイト単位のサイズを返す(charは1バイト、intは4バイト)
sizeof studentは、配列全体(要素数 x Student型のサイズ)のバイト単位のサイズを返す、配列の要素数は返さない
sizeof *studentは、Student型のバイト単位のサイズを返す
それらの割り算をしてるのだから、結果は要素数になる
2018/09/08(土) 02:56:57.12ID:OyJ8jP1h
>>382
>sizeof *student が配列の要素ひとつを表現しているっていうのがちょっと分かり難いかなぁ。

個人的にそういう表記は嫌だわw
変数のサイズを求めるなら、その型名でって思う
2018/09/08(土) 03:03:57.30ID:VmsJpbI+
>>384
へー、私は逆に型名よりも変数名で書きたい派。
これは変数 student について調べたいのであって、
型 Student について調べたいわけでは無いので。
2018/09/08(土) 03:07:29.94ID:ejo+9snz
>>382
すいません、おかげさまでどういう計算してるのか理解できました
確かにポインタにすると何で要素一つのサイズなのか理屈づけて説明できません…sizeof student/sizeof student[0]とかだと分かりやすいです
ありがとうございます。意味が理解できたので応用して使えそうです
2018/09/08(土) 03:08:42.91ID:VmsJpbI+
>>384-385
私が sizeof *student が分かり難いと述べたのは
配列名に * を付けたものが配列の先頭要素になるルールの背景にある暗黙の型変換が分かり難いって話。
C/C++ を使ってると空気みたいにやっちゃうんだけど、これ実際のところだいぶんアレなルールですよ。
2018/09/08(土) 03:13:32.57ID:OyJ8jP1h
>>385
型が表すサイズの定数だから、変数から取得するってのが気持ち悪いよ
sizeofなんて何か細かいことワチャワチャしたい時に型のサイズ欲しいよねーって感じの演算子だものw
2018/09/08(土) 03:17:21.63ID:OyJ8jP1h
>>387
それも含めてね
2018/09/08(土) 03:22:37.80ID:VmsJpbI+
>>388
この場合くらいの小さなサンプルだとあまり気にならないけど、
その変数が何の型であるか知っていなければならないっていうのが、
読むときの負荷が高いと感じてしまう。

自分を信用してないというか、自分がアホだということを信用しているので、
色々忘れて読んでも良いように書きたいと思ってる。
2018/09/08(土) 03:26:28.66ID:VmsJpbI+
C++ ではもうそんな古代のイディオムなんて忘れて std::extent を使いたい
2018/09/08(土) 03:28:24.66ID:VmsJpbI+
もう C++17 の時代なんだから std::size の方がいいな
2018/09/08(土) 03:48:16.76ID:VmsJpbI+
そもそも

for (int i = 0; i < size; ++i) {
Show(student[i]);
}

のところはモダンな C++ では (つまり C++11 以降では)

for (auto& e: student) Show(e);

と書けるので配列の要素数を事前に計算する必要なかったわ。
2018/09/08(土) 03:53:09.60ID:t7GfMYxV
>>380

>>382
>sizeof *student が配列の要素ひとつを表現しているっていうのがちょっと分かり難いかなぁ。

私は、これはかなり紛らわしいとおもっているので、こういうときには必ず
struct FOO { int n; ..... } array[N];
int n = sizeof(array) / sizeof(array[0]);
としています、こっちの方が好きです

構造体の配列の第一番目を指すポインタ
struct FOO *p;
p = &array[0];
のときに *p は array[0] に入っている「構造体の一つの要素全体」を表します、すなわち *p イコール array[0]

しかし、この表現はすぐに直感的にイメージとして頭に入ってくるわけではなく、何度となく書いたり読んだりしてゆっくり染みこんでくるものだ、と私は経験しています
同様に p->n とか、これと等価な (*p).n とかの表現など
書き続ければ、あるいは読み続ければ、そういうのは、だんだんわかってくるのではないかな、と。
2018/09/08(土) 03:59:06.82ID:VmsJpbI+
C を使ってるときは感覚的に理解してて全く不都合なかったけど、
C++ のテンプレートやなんやかやを利用するようになると、
やっぱりちゃんとルールを把握せないかんなと思った。
2018/09/08(土) 08:56:41.97ID:RilhlE4U
_countof()
環境考えなくていい(趣味etc)ならこれもある
2018/09/08(土) 09:31:57.86ID:obhERXW7
初心者に古い文法を辞めろってよろしくないな
2018/09/08(土) 10:37:06.40ID:ZUEeKRTR
>>388
何かの都合でその変数の型を変えた時に変更し忘れたらバグになる元
型で書く奴はアホだと言い切ってもいいレベル
2018/09/08(土) 11:59:22.19ID:OyJ8jP1h
>>398
言われると思った
2018/09/10(月) 18:57:43.66ID:70yMIW0+
c#から使うクラスライブラリをc++/cliで作っています。
クラス変数をoutで渡してメソッド内で値を入れたいのですが、c++側はどんな風に書けばよいでしょうか?
2018/09/10(月) 22:20:18.13ID:7zqNUn7X
>>400
https://msdn.microsoft.com/ja-jp/library/h9t2463y.aspx
2018/09/11(火) 11:36:04.20ID:EScF+7wZ
>>401
ありがとうございました。
解決しました。
2018/09/17(月) 21:49:35.29ID:t753Z4pL
ロベール入門の内部リンケージ外部リンケージ云々の下りがちょっとよくわからないんですが
externを付けると別のcppの変数を弄れるってことであってますか?
2018/09/17(月) 22:34:28.19ID:t753Z4pL
externをつけると別のファイルのグローバル変数、関数を使えてローカル変数は使えないんですか?
2018/09/17(月) 22:40:39.04ID:yTXbMS5S
>>403
はい、そうです。
>>404
staticではないローカル変数は外側から参照できない。static付きの変数は静的に確保されるので、アドレスを使えば参照できる。externを付けるのはグローバル変数の宣言。
2018/09/17(月) 22:43:11.40ID:yTXbMS5S
関数プロトタイプにexternを付けるのは無意味。
2018/09/17(月) 23:32:43.22ID:t753Z4pL
>>405
分かりました。
ありがとうございます
staticを付けると内部リンケージになって別のcppから使えない云々書いてありますが
アドレスを使えば弄れるんですか。よく分かりませんね(@@;
2018/09/17(月) 23:41:37.07ID:yTXbMS5S
static int a = 0;
...
f(&a);
もしくは、
int& g(void)
{
static int a = 3;
return a;
}
みたいな感じで。
2018/09/18(火) 01:22:31.81ID:OdQP9+YI
>>407
リンクってのはシンボル(名前)とアドレスを付き合わせるってこと
プログラマが意識してアドレスを使うなら何でもあり
2018/09/18(火) 01:45:06.12ID:XTp+U7qP
そして次のページにて
ヘッダファイルに#ifndef#define#endifを定義すると2重定義防止ができると説明があるのですが
これを使えばexternとか関数プロトタイプを使わないで住むということでしょうか。
だとするんであればexternとかプロトタイプ宣言とはどう使い分ければいいんでしょうか;
2018/09/18(火) 07:22:02.89ID:XTp+U7qP
すいません自己解決しました…多分;;
412デフォルトの名無しさん
垢版 |
2018/09/23(日) 16:56:38.50ID:dZPE20Nd
現在利用出来る無償のメモリリーク解析ツールってありますか?プログラム初心者でメモリリークの対策がよくわからないのでどういうところに対策する必要があるのかを知るのに利用したいです
2018/09/23(日) 17:17:02.47ID:oc2N8BDI
>>412
スマートポインタ使えばいいと思うよ
414デフォルトの名無しさん
垢版 |
2018/09/23(日) 18:38:22.45ID:cRG95Xcq
vs使えばexpressエディションでも標準装備されてる
415デフォルトの名無しさん
垢版 |
2018/09/23(日) 23:53:10.56ID:nKMJRhW5
>>413>>414
ありがとうございます
すみません。言語はCなんですが、スマートポインタなるものは使えるのでしょうか
VSのほう先ずは試してみます
416デフォルトの名無しさん
垢版 |
2018/09/24(月) 00:23:01.63ID:Kxio7RVg
とりあえずこんな感じのコードで軽くお試しできる
デバッグモードで実行するとデバッグコンソールにリークしてるログが出力される
コメントはずしてfree()で解放するとリークしてるログがなくなる

#define _CRTDBG_MAP_ALLOC
#include <stdlib.h>
#include <crtdbg.h>

void aho(void) {
  void* p = malloc(100);
  // free(p);
}

int main(void) {
  _CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF);
  aho();
  return 0;
}

↓詳細はココみなさい
https://msdn.microsoft.com/ja-jp/library/x98tx3cf.aspx

あとはもうできるハズだ
2018/09/24(月) 00:32:39.68ID:twbzBaDz
https://github.com/KindDragon/vld/wiki/Using-Visual-Leak-Detector
visual leak detector
cで使えるかは試したことないけど、crtdebugより使い勝手はいい。
2018/09/24(月) 00:58:58.99ID:qTTELbCs
>>417
全然ビジュアルじゃねーなw
2018/09/24(月) 07:47:25.22ID:A22Aay1A
入門の+operator演算子で訳わからず詰んでしまったのですが
簡単に分かりやすく解説してるサイトないですか?
2018/09/24(月) 09:09:54.16ID:tUW1+gfS
>>419
基本的なことがまとめられているのはこのあたりだと思う。
http://stlalv.la.coocan.jp/Operator.html
421デフォルトの名無しさん
垢版 |
2018/09/24(月) 10:37:05.34ID:nAcFE6zY
>>416>>417
ありがとうございます
頑張ってみます!
2018/09/24(月) 18:28:47.09ID:A22Aay1A
>>420
ありがとうございます
ひとまず扱えるようにはなりましたm(__)m
2018/09/25(火) 00:10:59.63ID:TBXAVCzN
>>418
visual (studioで使える) leak detectorかな?
重いけどスタックトレースがでるので便利。
424デフォルトの名無しさん
垢版 |
2018/09/27(木) 01:59:28.90ID:HwqzTUAF
http://codepad.org/gwutTCie
AIUEOを何回も呼び出したいとき、引数のa, b, c, d, eの部分が共通なので
何回もa, b, c, d, eと書かなくても良いようにしようと色々試しましたが中々上手く行きません
別にコピペすればいいのでそんなに気にすることでもないと思うのですが、
こんなときはどのように書いたら良いのでしょうか?
2018/09/27(木) 02:11:57.91ID:Oj9x/TA+
>>424
意図を把握できてるか自信はないが↓のbinded_AIUEOみたいなことでいいのん?

void MACRO(int a, int b, int c, int d, int e, int s, int t) {
 auto binded_AIUEO = [=](int x, int y, int z){ AIUEO(a, b, c, d, e, x, y, z); };
 if (s < 0) {
  if (t < 0) binded_AIUEO(-1, -2, -3);
  else binded_AIUEO(3, 0, 7);
 }
 else {
  if (s < t) binded_AIUEO(4, 99, 8);
  else binded_AIUEO(333, 33, 3);
 }
}
2018/09/27(木) 02:36:54.70ID:nKTHsEzL
std::bind を使う方法もあるな。
2018/09/27(木) 02:41:01.72ID:/q4lnTfs
>>424
struct Args { int a,b,c,d,e;};を引数に使う
2018/09/27(木) 12:00:19.82ID:QfV3dtaO
void foo(int argc, const char *argv[])
{
}

int main(int argc, char *argv[])
{
  foo(argc, argv);
  return 0;
}

VC++2008でコンパイルすると、
「error C2664: 'foo' : 2 番目の引数を 'char *[]' から 'const char *{}' に変換できません。」
ってなるんだけどなんで?
constから非constならわかるんだけど。
2018/09/27(木) 12:27:29.59ID:3irDnMHM
type ミスなのかコピペなのか判断に迷うとこだけど

'char *[]' から 'const char *{}' 後者が { } になってるあたりにヒントは無い?
2018/09/27(木) 12:42:33.07ID:pmYsr/E+
>>428
[]のタイプミスだろうけど、下のページが参考に。
https://stackoverflow.com/questions/45011978/const-causing-incompatible-pointer-type-why-only-for-double-pointers

ようは、fooで argv[0]="abcd"; みたいなことができちゃうから。
でも、const char * const argv[] でもだめな理由は知らない。
431428
垢版 |
2018/09/27(木) 13:06:59.87ID:QfV3dtaO
>>429,430
失礼。428の書き込みがタイプミスで、実際のソースコードは [] なのですが
コンパイルエラーになります。

>>430
なるほど・・・↓にも同じようなことが書いてありますね。
https://social.msdn.microsoft.com/Forums/ja-JP/24cd88de-2159-4e8b-aee7-3fa30d8647db/12509124521253112479123982144229031124343830625968123952819312?forum=vcgeneralja

ちなみにVC2003では通ってたらしいです。
2018/09/27(木) 14:04:18.04ID:rwW0e8CF
>>431
2010ではエラーにならなかったな
2008固有の問題なのかも知れない

foo(argc, (const char **)argv);
こうしてみたらどうだろう
2018/09/27(木) 17:43:51.14ID:HwqzTUAF
>>425 >>426 >>427
ありがとうございます
これのマクロ版を書くときはどのようにすれば良いのかと思いました
例えば #define BINDED_AIUEO(x, y, z) AIUEO(a, b, c, d, e, x, y, z)
とすると1, 2, 3, 4, 5ではなくa, b, c, d, eとそのままの文字で展開されてしまいます
2018/09/27(木) 18:37:57.80ID:nKTHsEzL
>>433
こんな感じかな?

#define MACRO(a, b, c, d, e, s, t) {\
#define BINDED_AIUEO(x, y, z) AIUEO(a, b, c, d, e, x, y, z) \
if (s < 0) {\
if (t < 0) BINDED_AIUEO(-1, -2, -3) \
else BINDED_AIUEO(3, 0, 7) \
}\
else {\
if (s < t) BINDED_AIUEO(4, 99, 8) \
else BINDED_AIUEO(333, 33, 3) \
}\
#undef BINDED_AIUEO \
}

この事例ではどうしてもマクロじゃなければ駄目という理由はなさそうだけど、
実際の利用シーンでは何か意味があるのかな?
2018/09/27(木) 19:16:32.64ID:Oj9x/TA+
>>434
それできたっけ?と思って試してみたらエラーになった
マクロの中では#は識別子の文字列化になる

可変長マクロを悪用してこんな感じにはできるけれども……おすすめはできない

#define packed_MACRO(s, t, ...) {\
 if (s < 0) {\
  if (t < 0) AIUEO(__VA_ARGS__, -1, -2, -3) \
  else AIUEO(__VA_ARGS__, 3, 0, 7) \
 }\
 else {\
  if (s < t) AIUEO(__VA_ARGS__, 4, 99, 8) \
  else AIUEO(__VA_ARGS__, 333, 33, 3) \
 }\
}
#define MACRO(a, b, c, d, e, s, t) {\
 packed_MACRO(s, t, a, b, c, d, e) \
}
■ このスレッドは過去ログ倉庫に格納されています
5ちゃんねるの広告が気に入らない場合は、こちらをクリックしてください。

ニューススポーツなんでも実況