C言語なら俺に聞け 145

■ このスレッドは過去ログ倉庫に格納されています
2018/02/19(月) 22:13:58.98ID:9/te2eSJ0
C言語の話題のみ取り扱います C++の話題はC++スレへ
質問には最低限の情報(ソース/コンパイラ/OS)を付ける
数行で収まらないソースは以下を適当に使ってURLを晒す
https://paiza.io/
https://ideone.com/
http://codepad.org/

C11
http://www.open-std.org/jtc1/sc22/WG14/www/docs/n1570.pdf

C99
http://www.open-std.org/jtc1/sc22/WG14/www/docs/n1256.pdf
http://kikakurui.com/x3/X3010-2003-01.html

C FAQ 日本語訳
http://www.kouno.jp/home/c_faq/

JPCERT C コーディングスタンダード
https://www.jpcert.or.jp/sc-rules/


C言語なら俺に聞け 144
https://mevius.5ch.net/test/read.cgi/tech/1514025223/

次スレを立てる時は本文の1行目に以下を追加して下さい
!extend:on:vvvvv:1000:512
VIPQ2_EXTDAT: default:vvvvv:1000:512:----: EXT was configured
2018/02/23(金) 16:43:48.45ID:lIelkvwO0
Bの頃からあったprintfの実装テクニックだし
だいいちCは簡潔さを売りにしてたんで
理想論に走って複雑さを増すようなことを嫌ったんだよ
2018/02/23(金) 16:48:19.38ID:yeKpu5OYM
>>72
せやな。
で、func(int,....) な関数ポインタは作れるよね。
それを汎用的に使うことは出来る。ってのが56への回答じゃね?
2018/02/23(金) 17:03:04.90ID:lIelkvwO0
void func(int, double);
void (*ptr)() = func;
ptr(1, 1); //1.0で渡すべきところ1が渡される
void func(int top, double arg) //スタック構造の不一致が起きる
{
va_list ap;
va_start(ap, top);
assert(va_arg(ap, double) == arg); //通るわけねえだろ
va_end(ap);
}
2018/02/23(金) 17:11:03.17ID:V7qyvSbE0
>>75 いいや ポインタだけじゃ片手落ち
(キャストじゃだめで)関数の実体側の引数も同じ引数で実装しなければならない
2018/02/23(金) 17:26:46.79ID:UgOkn/Bq0
スタートアップは argc,argvをスタックに積んで _mainを呼ぶ
_mainが int main(void)だとしても問題ない
2018/02/23(金) 17:49:49.55ID:Ska5zxPe0
>>67 はエラーなくコンパイル・実行できてるようだから、もう少し検証。
ideone.com でなくローカルのgccで実験。

どうやら func() 定義の引数リストに制限があるみたい。
void f(double x) ... エラーなし ( >>67 の例と同じ)
void f(int x) ... エラーなし
void f(void* x) ... エラーなし
void f(char x) ... 警告:「互換性のないポインタ型」
void f(short x) ... 警告:「互換性のないポインタ型」
void f(float x) ... 警告:「互換性のないポインタ型」

警告は void (*pfunc)() = f; の行で発生。
実引数の自動拡張よりも狭い幅の仮引数の型を使おうとすると警告かな?
コンパイラによって動作が異なる部分で、深追いする価値は無いのかも。
2018/02/23(金) 18:00:16.52ID:yeKpu5OYM
>>77
78が正解。更に言えばmainの引数には envp もある
2018/02/23(金) 18:25:51.38ID:UgOkn/Bq0
envpはANSI C 標準に対する Microsoft の拡張機能
https://msdn.microsoft.com/ja-jp/library/k104fy6h.aspx
2018/02/23(金) 18:42:59.00ID:mpvgXBL20
>>81
posix にもある、推奨されているわけではないが
2018/02/23(金) 19:10:50.73ID:Ska5zxPe0
補足というか訂正。
>>55 で「long(*pfunc)() = func; の行がエラーになる」
と書いたけど警告だった。よって >>25 のソースもコンパイル可能。

ちなみに結果は期待の 6 とは桁違いのデカい数値だった。
2018/02/23(金) 20:24:30.86ID:mFkXyQ34M
void*でええやん
2018/02/23(金) 21:15:05.64ID:JKrDz8Gza
 ヘ_ヘ
ミ・・ ミ
(   )〜
2018/02/23(金) 21:32:39.13ID:my0767Fs0
void から ヘミ猫 て fj かよ
2018/02/24(土) 00:57:58.46ID:ar5NnKGja
>>83
>>25の書き方そのままじゃlong aの部分しか渡されてないからb, cは不定値になってるのでは?
2018/02/24(土) 06:40:32.39ID:agv5rOmv0
>>87
param には「long幅の共用体3要素の配列」が含まれるから、
値渡しすればスタックに (sizeof(long) * 3) byteのデータが積まれるはず。

というか、main()でparamの各要素をlong解釈で表示させたら
見事にaやbの上位ビットにゴミが入っていた。
2018/02/24(土) 09:43:12.88ID:uX3sJRXI0
structにstaticつけ忘れてましたとか
2018/02/24(土) 15:58:01.09ID:dhHE2VyNa
>>88
> 各要素をlong解釈で表示させたら見事にaやbの上位ビットにゴミが入っていた。
それは初期化の問題では?

確かに値渡しはスタックにコピーされるけど、関数の引数にはもう一回コピーはいるから、そこで関数不一致の弊害が発生すると思う
>>25のコードそのままだとaしか正しく渡されてなくて、b,cは明後日の方向からデータ取ってきてるかと
2018/02/24(土) 16:57:25.47ID:agv5rOmv0
>>88 に「期待の6」と書いたせいで誤解させてしまった。
もちろん実際の動作で6が表示されると思っていたわけじゃないのだ。
この流れの発端の質問 >>25 と、それに追加された >>52 の、
「実行結果も期待通りだった」て記述を受けての「期待の6」でね。

むしろ共用体への代入で、
 param.u[0].a = 1; // charの代入 byte転送命令
 param.u[1].b = 2; // shortの代入 word転送命令
 param.u[2].c = 3; // longの代入 dword転送命令
てな具合にアセンブルされ a, b の上位ビットに未初期化のゴミが残り、
関数に実引数の値として渡されるときもゴミごとコピーされるのは当然。
>>52 で期待通りに動いたのは、paramの未初期化での状態が
たまたま全部ビットで0だったんだなぁ、というお話。

>>25 のリストで、関数 func() の仮引数 a, b, c の型と、
共用体の要素 a, b, c の型が食い違ってる点にも注目して頂きたい。


…長々と失礼。誤解の余地なく説明しようとするとクドくなってしまう。
2018/02/24(土) 18:41:41.80ID:7mL/bBDwd
#include <stdio.h>
#include <string.h>
#define BUF_SIZE 80

void* my_realloc( void* p, size_t bytes, int line ) {
if( (p = realloc(p, bytes)) == NULL ) {
fprintf(stderr, "realloc failure called at line %d\n", line);
exit(EXIT_FAILURE);
}
return p;
}
int main () {
FILE *fp;
char *file = "data.csv";
char buf[BUF_SIZE], input[BUF_SIZE], format[20];
char **result, *s;
int i, size = 50, count = 0;

if( (fp = fopen(file, "r")) == NULL ) {
perror(file);
exit(EXIT_FAILURE);
}
sprintf(format, "%%%ds%%*[^\n]%%*c", sizeof(input) - 1);
scanf(format, input);

result = my_realloc(NULL, sizeof(*result) * size, __LINE__);

while( (fgets(buf, sizeof buf, fp)) != NULL) {
if( strstr(buf, input) == NULL ) continue;
2018/02/24(土) 18:42:06.22ID:7mL/bBDwd
s = my_realloc(NULL, strlen(buf) + 1, __LINE__);
strcpy(s, buf);
result[count++] = s;
if( count >= size ) result =my_realloc(result, sizeof(*result) * (size += 20), __LINE__);
}

for( i = 0; i < count; i++ ) {
printf(result[i]);
free(result[i]);
}
free(result);

return 0;
}
2018/02/24(土) 18:45:40.89ID:7mL/bBDwd
上記のコードから追加でしたいことが2つあります。
@検索単語を2つにしたい。
Aフォルダないの全ての.csvファイルを検索かけたい。

sprintf(format, "%%%ds%%*[^\n]%%*c", sizeof(input) - 1);
scanf(format, input);

result = my_realloc(NULL, sizeof(*result) * size, __LINE__);

ここの文の意味が分かりません。
2018/02/24(土) 21:35:03.63ID:h1W1dcA8d
>>94
クソコードだ。あまり参考にせず、自分で考えた方がいい。
my_reallocはreallocをラップした関数形式マクロのようだ。リファレンスのreallocを参照。
__LINE__は現在の行番号を表し、デバッグなどに使う。
sprintfはprintfを文字列出力にしたものだ。%%は%になって出力される。
2018/02/24(土) 21:47:22.04ID:h1W1dcA8d
sprintfで出力した文字列をscanfに書式として渡している。でもこのようなscanfの使い方はオススメできない。
2018/02/24(土) 21:56:26.92ID:h1W1dcA8d
%dは整数を表す文字列になるので、%%%dsは%(何らかの整数)sになる。これは文字列出力の幅指定だ。
幅指定には、別の直接的な方法があるので、こんなことしなくてもいい。
scanfの使い方は引数の個数がちょっと間違っているように見える。
2018/02/24(土) 21:59:21.39ID:h1W1dcA8d
バグや不具合の元になるscanfなんて使うな。fgetsとsscanf使えば充分で分かりやすくて確実だ。
2018/02/24(土) 22:25:03.02ID:tEhbLaK5M
>>94
@else ifでええやん
Areaddirでええやん

myreallocみたいなモンはマクロにしとけ
reallocみたいな糞ライブラリ使うな
sprintfみたいな糞ライブラリ使うな
2018/02/24(土) 22:45:22.29ID:PDjCLuF50
特定のフォルダ(dir)内のファイルを順次処理したいときは
Unix系ならばopendir()、readdir()、closedir()を使う。
VCならばFindFirstFile()、 FindNextFile()、FindClose()を使う。
2018/02/24(土) 23:27:16.76ID:yWQ45jBy0
たしかに realloc() はいろいろまずいところがあって、
一番悪いのは、やり直しができない、ということ
2018/02/24(土) 23:34:03.13ID:PDjCLuF50
一致した行をそのままファイルに書き出す方が簡単な気がする
2018/02/25(日) 01:42:29.02ID:XjF3qDop0
>>99>>101
え? realloc() が糞?じゃあ malloc() で確保したメモリ領域の大きさを変えるのは何を使えばいいの?
自分で確保し直して中身を自分で全部コピーしろと?
2018/02/25(日) 01:43:39.45ID:XjF3qDop0
>>101
てか、やり直しができないってどういう意味?幾らでもできると思うが。
2018/02/25(日) 01:50:49.81ID:I4H7O+d10
1バイトでもメモリ消費量を節約する事が求められた世代と
湯水の如く無尽蔵にメモリが使える世代のバトルが始まる?
106デフォルトの名無しさん (ワッチョイ df9f-IDeO)
垢版 |
2018/02/25(日) 03:25:39.33ID:XjF3qDop0
>>105
そういうことなの?w
107デフォルトの名無しさん (ワッチョイ 2f93-IXLF)
垢版 |
2018/02/25(日) 06:04:48.57ID:Jqo0wCIZ0
>>92 >>94
sprintf(format, "%%%ds%%*[^\n]%%*c", sizeof(input) - 1);
の行では input[BUF_SIZE] で BUF_SIZE が80であることから
format[] に "%79s%*[^\n]%*c" (\nは改行コード) が格納される。
よって scanf(format, input); は scanf("%79s%*[^\n]%*c", input); の動作。
変換指定の "%79s%*[^\n]%*c" の意味や妥当性は別のお話。

>>96
scanf() の変換指定文字列を sprintf() で生成するやり方は
『プログラミング作法』(カーニハン&パイク)に載ってる。
BUF_SIZE の定義を変えたときにscanf()の変換指定の文字数指定部分も
自動的に追随させる便法(片方だけ修正忘れの防止)として、悪くないかと。

バッファサイズは固定なのに変換指定を動的に作るのは非効率だ、
という気持ちも分かるけどね。
2018/02/25(日) 07:37:51.66ID:HIxgC6KA0
片山氏はscanfがまともに使えず
自分のミスを道具のせいにする手合いか
2018/02/25(日) 08:49:08.43ID:vTPpNH4CM
>>103
そう
糞と言った理由は冪等性がないから
冪等性よりも結果整合性を重視するならrealloc使えば良いと思うよ
2018/02/25(日) 09:18:29.86ID:Jqo0wCIZ0
>>109 「冪等性がない」の意味を補足してくれまいか。
realloc()に冪等性がない、結果整合性はある、という意味が分からないのだ。
2018/02/25(日) 09:35:20.85ID:miEl51/Z0
失敗したとき NULL を返しつつ 「引数で渡したポインタは開放されない」
あたりに引っかかりを感じてるんでない?

NULL 返すなら 引数で渡したポインタを開放しろと?
※ 使いにくいかもなぁ…
112デフォルトの名無しさん (ワッチョイ df9f-x4Or)
垢版 |
2018/02/25(日) 11:58:37.90ID:XjF3qDop0
俺も意味がわからん。
113デフォルトの名無しさん (ワッチョイ df9f-x4Or)
垢版 |
2018/02/25(日) 12:17:48.32ID:XjF3qDop0
>>111
それはメモリ確保に失敗してるんだから解放したらダメだよね。
自分で malloc() 使って新規にメモリ確保しようとした場合でも NULL 返されたら元の方は free() しないだろう。
2018/02/25(日) 12:28:06.03ID:vTPpNH4CM
>>110
結果整合性って言葉が悪かったな
reallocの機能を重視するなら使えば良いと思うよって書けば良かった

冪等性の意味が分からないならググってくれとしか
2018/02/25(日) 12:33:20.87ID:XjF3qDop0
あ、もしや >>109 の言ってる冪等性がないって引数で渡したポインタとは違うポインタを返してくることがあるって意味?
あまりにも当たり前すぎて思いつかなかったが。

しかし malloc() の実装にもよるだろうがこれを許さないと効率は悪くなるのではないか?
2018/02/25(日) 13:13:10.59ID:vTPpNH4CM
>>115
そう
違うアドレスを返して初めて発現するバグをボケナスビがやらかしてから俺は戒律で禁じた
他にも初期化やアライメントの点もある

メモリ効率を重視するなら自分で管理構造作るべきだと思ってる

冪等性を最重視する俺はこのスレじゃ少数派みたいだからあまり気にせんでくれ
2018/02/25(日) 13:20:16.07ID:L5EpKLxx0
>>115
× 効率は悪くなる
○ まともな実装は無理(ほぼ全ての場合に失敗する)

>>116
気持ちは分からんでもないが、
reallocはそれが仕様で、いやならstd::stringみたいにラップするしかない。
2018/02/25(日) 13:22:50.04ID:I4H7O+d10
(流れが質問者を放置しているように感じる)
2018/02/25(日) 13:26:41.23ID:qARxqxts0
>>118
質問者の疑問を解決する事より自分の主張したいことに論点がシフトしちゃうのはよく見る光景だね。
質問スレでそれじゃダメだと思うが。
2018/02/25(日) 13:38:08.00ID:vTPpNH4CM
>>118>>119
確かにそうだな
俺は安価ついた分しかレスしないつもりだったけど、この話題についてはもう止めとく

スレ汚しすまんかった
2018/02/25(日) 13:44:15.14ID:9iGYkY9C0
>>103 >>104
c++ には realloc() は取り入れられなかった

ちょっと例が悪いが
concat3(char *a, char *b, cha *c) 文字列a と 文字列b と文字列c を連結して文字列a に返す関数を書こうとする
ここで 2 回の realloc() を行うとする
1 回目の realloc() は成功するが、2回目の realloc() は失敗したとする
このとき「関数 concat3() が失敗したときは状態は不変、成功したときだけ状態を変更する」という原則を満たせない

我ながらちょっと例が悪いとは思うが、
realloc() を暢気に使っていると、エラー処理が一手ばったりになってしまう。
2018/02/25(日) 13:49:23.19ID:L5EpKLxx0
>>120
俺はいいと思うぞ。
むしろ無理に「質問者様が神様だ」的にしたがるゆとりは死ねと思うが。

そもそも質問には回答つきまくってるし、質問者側が何らかの反応しないと話は進まないだろ。
流れも読めない癖に流れを主張するゆとりは殺すしかない。
2018/02/25(日) 14:39:52.01ID:XjF3qDop0
なるほど。まあ確かに使い方や使い所を間違えればハマるものではあるな。

そういや以前 realloc() で渡すポインターがNULLではなくサイズの指定が0だとfree()されるってのでハマったことがあった。w
2018/02/25(日) 14:51:26.22ID:nm5odapXd
ANSI準拠のreallocは、mallocとfreeの代わりになる。
2018/02/25(日) 15:46:21.99ID:s5td7qK+0
未だにANSIの規格票を読んでる日本人いたのか
126デフォルトの名無しさん (ワッチョイ 6e23-Rch6)
垢版 |
2018/02/25(日) 15:51:12.25ID:DMbXpP0c0
俺も読んでる
127デフォルトの名無しさん (アウアウカー Sa11-CD0d)
垢版 |
2018/02/25(日) 15:59:51.76ID:NsuolzGja
未だにANSI使う理由って何だろう?
2018/02/25(日) 16:00:08.18ID:s5td7qK+0
ISOではなくANSIか?
2018/02/25(日) 18:48:53.57ID:bdk82RzyM
>>121
それはプログラムの組み方が悪いだけでrealloc()のせいじゃないだろ
2018/02/25(日) 18:55:32.39ID:I4H7O+d10
>>121
最初から文字列3つ分の領域確保すれば良いんでないの?
2018/02/25(日) 19:23:06.76ID:L5EpKLxx0
>>121
reallocではなく、自前でmalloc+memcpy+freeでやればいいだけ。
ただその前に、指摘されているとおり、
3つの「同時に変更される」生ポインタを掴ませるのが問題。
その場合には普通は一つの構造体に入れてそれ経由で使い、
「同時に変更される」という条件が外部に見えないようにする。(隠蔽)
勘違いしている奴も多いが、OOPは文法の話ではない。
2018/02/25(日) 20:09:18.52ID:fbPK05Px0
reallocをmallocとmemcpyでやろうとするともとのメモリサイズわからずに詰まない?
2018/02/25(日) 20:41:49.10ID:p0obWBJzd
結果何をどうすればいいんですか?レベルが高すぎて何を言ってるのかさっぱりです
2018/02/25(日) 20:48:37.39ID:Fc1MjilIa
だからreallocはいまいちって話なのでは?

>>132
malloc_usable_size, _msize
2018/02/25(日) 20:58:36.53ID:H8bVBKR7a
>>133
reallocに関してはひとまず気にしなくて良いかと
2018/02/25(日) 21:15:09.00ID:I4H7O+d10
>>133
見つかった行の保持ってメモリー上でやりたいの?
2018/02/25(日) 21:58:18.03ID:EC2T/8ze0
>>136
別にこだわりがあるわけではないです。

明日やろうと思うので、調べるあたりこれ見とけとかあります?
2018/02/25(日) 22:15:41.07ID:I4H7O+d10
メモリ上に保持しなくても良いなら、検索して見つかった行をそのままファイルに書き出してしまう方が簡単だと思う。
あと、どういう環境で実行するか分からないんだが、もしも正規現が使えるなら、それを利用する方が簡単だと思う。
複数のキーワードで探したいときは一括で指定できるし。
2018/02/25(日) 22:23:36.04ID:L5EpKLxx0
>>137
1. grep/sedでおk
2. AWK/Perl/Ryby/Pythonでもおk
3. つかその用途でC使うとか正気じゃねえ。メモリ上でゴリゴリやるにしても普通はスクリプト言語。
4. それも分かっててC使う覚悟のある奴がここでこんなアホな質問する訳ねえ。>>138
2018/02/25(日) 22:51:32.71ID:9iGYkY9C0
>>139
>用途でC使うとか正気じゃねえ。メモリ上でゴリゴリやるにしても普通はスクリプト言語。
その判断をするのには、一定の C での経験が必要だと思います
python なんかでやったほうがいい仕事をあえて C で書く経験はあったほうがいいでしょう
2018/02/25(日) 23:44:30.89ID:EC2T/8ze0
VBでは出来るんですがCの勉強をかねてトライしてます。
いろいろとありがとうございました!明日頑張ってみます!
2018/02/26(月) 11:24:56.97ID:0MjsGOBH0
>>132
メモリーサイズを知る必要はない
そもそも確保されてるサイズと strlen(a) が同じかどうかはわからんし
2018/02/26(月) 12:24:58.22ID:4Dp3S/v30
>>142
concat3とかいうQZのボケたろくでもない例のことは忘れろ
2018/02/26(月) 12:38:44.35ID:lxlU26hn0
>>132
メモリサイズは非公開というだけで、内部的には持っている
だからreallocの作者=mallocの作者が詰むことはない

それ言い出したらfreeも解放量わかんなくて詰むだろ
2018/02/26(月) 13:29:20.63ID:zI4VKzz4M
>>143
お前は何を言ってるんだ?
2018/02/26(月) 17:32:30.23ID:e+KVEn7XM
K&Rで説明されている筈なんだが
2018/02/26(月) 17:41:14.36ID:vVA9DKIbd
結果出来ませんでした...
2018/02/26(月) 20:20:31.75ID:4PPe6ndQ0
>>144
reallocの作者がmallocの作者と違う場合なんだが
149デフォルトの名無しさん (アウアウカー Sa0a-sySv)
垢版 |
2018/02/26(月) 20:38:25.45ID:9Qy36xi2a
違うかなあ?まあ、ソース公開されてれば違う人が書くことは楽だが。
2018/02/26(月) 22:00:57.86ID:Z8+ASG310
>>147
どの辺がどうできなかったかを書いてみて
2018/02/26(月) 22:08:57.21ID:lxlU26hn0
>>148
俺は、それを空想論と断言してみる
おまえさん恥をかかせてくれることはできるかな?
2018/02/26(月) 23:37:18.28ID:Il2cKkTC0
>>150
単純に自分でやってみようと思い一から書き始めたんですけど

CSVファイルのカンマ区切りで、行は日々増える形式で、13列の固定。それぞれ数字だったり文字だったり混合。
ファイルを開いて読み込んで一行ずつ検索文字列と比較して一致したら表示列を指定して表示させ、一致しなかったら次へを繰り返していこうと思って始めたんですけど。
配列に入らずうまいこといきませんでしたね。ネットの参考もいいのが見つからなくて...

前載せたコードを読みといた方が近道ですかね?
2018/02/26(月) 23:49:55.49ID:KRl4xBEA0
>>151
もともと標準mallocやfreeを使って自前realloc作るならという話の流れだっただろ
2018/02/26(月) 23:57:50.74ID:Z8+ASG310
>>152
CSVファイルを読み込むプログラムは作った事ありますが
CSVの仕様をどの位受け入れるかで複雑さも変わると思います。
この辺は汎用的に作るか、自分が処理したいCSVの仕様の範囲にするかで違うとは思います。

scanf系の処理はとても癖が強く、
よっぽど仕様に精通していないと填まります
私ならその道を行くのはやめます
2018/02/27(火) 00:01:01.54ID:uYnnRgds0
>>152
grepでおk、マジで
そんなことをCでやる馬鹿はいないから、ろくな参考情報がないのも当たり前。
2018/02/27(火) 00:04:33.43ID:vDZ6bRaj0
"と , のエスケープと 1要素の中に改行を含む文字列周辺 RFCみて頭いたくなった
2018/02/27(火) 00:13:39.83ID:euQ4hghrM
>>152
俺なら間違いなくpythonつかうね
https://qiita.com/okadate/items/c36f4eb9506b358fb608
158デフォルトの名無しさん (アークセー Sx5f-jcVR)
垢版 |
2018/02/27(火) 07:16:02.83ID:pdRg+OHax
配列とかポインタとか難しいと聞いていたから心配していたけど
とりあえず基礎の解説書レベルなら分かった。やれやれ。
2018/02/27(火) 09:21:35.76ID:uskldtVs0
指定したサイズの2次元配列を確保する関数を設計してくれまいか?
2018/02/27(火) 10:07:05.72ID:XfJ/ct090
>>159
無理じゃね?
char (*dim_alloc(size_t width, size_t height))[width];
最後の[width]がエラーになる[*]にしてもあかん
[]にすると通るには通るが、今度は
char (*a)[3][4] = dim_alloc(3, 4); //型の不一致になる
2018/02/27(火) 10:08:23.25ID:XfJ/ct090
あ、失礼
char (*a)[4][3] = dim_alloc(3, 4);
結果は同じだけど
2018/02/27(火) 10:55:55.07ID:XfJ/ct090
できた
void dim_alloc(size_t, size_t, char (**b)[*]);
int main(void)
{
size_t width = 3, height = 4;
char (*a)[width];
dim_alloc(width, height, &a);
}
void dim_alloc(size_t width, size_t height, char (**b)[width])
{
*b = (char (*)[width])malloc(width * height);
}
これならおk
163デフォルトの名無しさん (アウアウカー Sa0a-sySv)
垢版 |
2018/02/27(火) 11:45:45.08ID:r7ed+pq5a
ポインタの配列使っちゃえばいいじゃん。free()する時注意が必要だがアクセスがa[y][x]形式なのは同じだし。
2018/02/27(火) 11:48:09.27ID:w9pRJ3lK0
二次元配列とポインタの配列だと、使い方が違うでしょう
165デフォルトの名無しさん
垢版 |
2018/02/27(火) 12:21:42.40
mallocされたサイズが分からんとか何言ってるの?
gccもVCも確保サイズを返す関数を提供してるだろ
166デフォルトの名無しさん (アウアウカー Sa0a-sySv)
垢版 |
2018/02/27(火) 12:27:42.17ID:r7ed+pq5a
>>164
確保時と解放時が違うが読み書きアクセスする時の表現は同じにできる。
2018/02/27(火) 12:30:04.09ID:CvenUVT40
>>159
gcc6 では一筋縄ではいかなくなっているね…
2018/02/27(火) 12:35:33.45ID:XfJ/ct090
>>165
非標準のね
2018/02/27(火) 17:51:59.04ID:CvenUVT40
>>159
できた!
https://mevius.5ch.net/test/read.cgi/tech/1434079972/36
2018/02/27(火) 18:02:32.91ID:CvenUVT40
>>162
>>159
>char (*a)[width];
これに値をいれないといけないし、そもそも、それも含めて malloc() してほしい、のでは
2018/02/27(火) 18:07:21.76ID:BEzsaLeta
ああ。先頭にポインタを詰め込んで一発で全部free()できるようにしたのか。
2018/02/27(火) 19:24:56.97ID:LuvO3mbWM
>>165
非標準だし、そもそもそんなものを使う必要は滅多にない
173デフォルトの名無しさん
垢版 |
2018/02/27(火) 19:27:39.38
>>168,172
上の方でmalloc作者じゃなきゃrealloc作れないっていう主張があったから、普通は処理系で専用関数を提供してるだろって話
2018/02/27(火) 19:47:16.45ID:jvpYfXKk0
>>159
わざわざ関数を設計しなくても、例えば3次元配列なら
int ary[dim_z][dim_y][dim_x];
int (*ptr)[dim_y][dim_x] = malloc(sizeof(int) * dim_z * dim_y * dim_x);
とやれば ary と ptr は同等に扱えないかな。

ptrはfree()する必要があること、sizeofの対象にしたとき同じ値を返さないことは
別関数を作っても同じだし。
■ このスレッドは過去ログ倉庫に格納されています
5ちゃんねるの広告が気に入らない場合は、こちらをクリックしてください。

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