1 :
デフォルトの名無しさん :
2010/06/30(水) 10:22:47 なぜポインタで引っかかる人が多いのか 引っかかる人は何に困惑しているのか
ポインタは概念のようなものなので、知ってすぐ使える標準関数やSTL等と違い「何に使えるか」が不明瞭だから
*だから
なぜ雑談スレや質問スレに書き込まずにスレを立ててしまうのか。 クソスレ立てる人はバカなのか死ぬのかチンパンジーなのか。
物理メモリにデータをおいてるってイメージがないとね
6 :
デフォルトの名無しさん :2010/06/30(水) 14:28:09
ポインタって訳すと「指し示す」だろ。 それを*をつかって値を書き換えたらポインタじゃないじゃん。 参照するまでは許す!でもいじったらポインタじゃない!
数値扱いにもなるのが理由じゃないかな。 それ故に今扱ってるのがデータなのか、アドレスなのかワケが分からなくなる。 アドレス書くべきところにデータを書いてしまったり、逆をしたりしたときに 予測不能な挙動を引き出してしまい、何が起きたのか分からなくなる。 ポインタ型と数値型が明確に区別されて、キャストしないと数値にできず 型エラー出すなら引っかかりにくいと思う。
ポインタはね、メモリのアドレスでしょ。 そんなことは言われればだれだって理解できるんです。 要するに計算用紙の座標がポインタですから。 でも複合型の文法を出した途端、頭の中で処理できなくなるんです。 加えて、配列がポインタとかウソの説明をうけてごらんなさい。 ほら、分からなくなった! そうこうしているうちに関数ポインタの説明まで行くでしょ、 そこまで行ったらなーんとなく概念では分かるけど、 文法は全くわからんちーん。 アドレスだから数値と一緒で計算できるとか勘違いするようになる。 配列のポインタが配列の先頭要素指してるから キャストエラーを無視してもいいとか頭の中が沸いてるんじゃないかと 思う発送をしだすようになる。
配列がポインタ、は嘘なんだが 言語仕様からして割と混同されてるだろ あれでは分からなくなるのも無理はないかと
でも別にそんなのどうだってよくね? 物理メモリのどっかにあるその領域の先頭を指してるもん的な 意味でしか使ってないぞ>配列 あくまで俺はって話しだけど いちいち配列の先頭とポインタとでごっちゃになっちゃうような使い方してるのって 組んでるプログラムも怪しくねーか? 俺みたいに物理メモリのどっかを指してるモノってとらえてると 配列もポインタって言っちゃう人がいるのかもね ただ、その場合連続してるもんとしてないもんと脳みそに入ってないとダメだけど
配列式aをa[0]に自動変換できる一部の場合(要するにsizeofとか以外)、 a[2]=2[a]が*(a+2)と等しいことを取って配列と等しいことを指して、 配列とポインタが等しいなんて言う人がいるんだよなあ。 まあもっともポインタが引っかかる人はそれ以前だと思うが。
では配列とポインタはどこが違うのだろうか。 正確な答えを自分は知らない。
>>12 int a[10];
int *b;
この時、aは定数、bは変数
a[0]はアドレスa+0を参照する
*bは変数bが保持する値をアドレスとしてメモリを参照する
printf等でaとbを参照すると、どちらからもアドレスを得られるので混乱の元となる
等しいものとして扱うとどんなバグが発生するのか例を出して
領域が連続しているかどうかぐらいの違いしか見えないな でもポインタのアドレスずらして計算なんてぶっちゃけいまやらないからまったく害がないなw
ここはポインタを理解できない人の原因を探るスレッドだろ
使う→間違える→考えるの繰り返し わからない人はどれもやっていない
アセンブラもやってようやく理解といった感じ
プログラムのお勉強のいいところは やる環境さえあれば、身についてしまうことだ。
int a[10]; int *b=a; で、sizeof(a)=40, sizeof(b)=8;
>>18 俺もそうだったな。学生時代にBASIC→C言語やった時には
全く理由の解らない現象に悩んだもんだ。
メモリの状態によっては期待通りの結果が出たり出なかったりするからまたタチが悪い…
勉強に使った本が悪いんじゃないかな [C言語ポインタ完全制覇」とか使ってみて理解できないなら別の要因だろうけど それと自分で理解を深めたい時、どういった確認(プログラム)すればいいのかわからない
>>20 みて思った
俺いまだ32bitのアプリ作ってるわ
64bitは甘え
・アドレス値の変更可否 ・sizeofの値 違いはこれくらいかな。
アセンブラやってからだと全く困らないんだけどねー。
char *name[]とか見たらウンザリする。 これってname[0]、name[1]などはアドレスを意味するんですよね?
>>2 メモリアドレスという、非常に具体的なものなのだが。
java出身者はこの辺まるでダメだね なんでこんな糞を量産してるの?ってぐらいダメだね
珍班AIは糞スレ
javaにポインタはないと言ってる香具師は ぬるぽしたことがないんだろうか
>>32 本当はjavaでも理解してなきゃならないよな
でもさっぱりわかってねーのがほとんどなの
見て見ぬふり出来るということ
Z80とC言語を同時期に教えれば、ポインタではまず引っ掛からない 高専電気科は同時期に教えてた
>>32 Java, Cどちらもできるけど、「Javaにはポインタは無い」でいいよ。
ポインタが存在する、っていう説明でも別にいいけど。
あるのはあくまで参照。ポインタ演算は不可能。配列も大概アクセス遅いし。
Java出身者が引っかかるのはC++のオブジェクトが
Cで言うオブジェクトと近くて、要するに構造体で値型なところだろう。
>>25 あと&演算子も配列とポインタじゃ変わってくる。
うんと昔、はじめてCを勉強した頃を思い出してみると、 int i; char buffer[100]; char *ptr = buffer; scanf("%d", &i); scanf("%s", ptr); printf("%d", i); printf("%s", ptr); なにこれ?なんでscanfではポインタだけ&がなくて、printfでは両方&がないの? って不思議に思ったな。 この辺でつまづくんじゃね。って一番最初だが。
>>35 おれ6809だったけど、ポインタでは引っかからなかった
まあ、でも別に半年もすれば大抵の人間が理解しちゃうし 2〜3年たっても理解できない奴なんていないし 大した問題じゃないんだけどなw
理解できない奴は2年も経たずに離脱するからな。
理解出来ないものに2年も係わるなよ
だから、「2年も経たずに」って書いてあるじゃん。
開発1年半ぐらいの奴が、関数の引数で**pとせずに*pとしてハマってて、 試しに簡略なメモリマップ書かせてみたら、書けなかった という話を開発のリーダに聞いたんだが、ある意味すごいな
44 :
デフォルトの名無しさん :2010/07/01(木) 11:00:16
ポインタはわかるけど、引数がポインタのポインタとかになってるとわけわかめ
>>43 意味不明なことやらせんなよ
Windowsのはきついだろw
プログラマの人材募集に応募してきたプログラマの面接時の珍回答の例: ・「ダブルポインタは大丈夫ですけどトリプルポインタは自信がありません」 ・「Cの入門書を3冊読んだから書いたことがなくても大丈夫」 ・ランバラルの台詞を引用する
たまたま知る限り全ての処理系でポインタの実装がメモリアドレスを格納する変数なだけで 規格上はポインタがメモリアドレスである保証は無い
>>47 実際、Cインタプリタで、ポインタの実装が(実装言語の)配列インデックスである例はあるよ。
それって翻訳の処理が規格通りじゃないから そもそも規格準拠の処理系じゃないんじゃね? どういう解釈になるんだろ
>1 俺みたいなやつがいるから
PICなんかはコード領域とデータ領域が物理的に分離されてるからな
たしかに何に使えるのかのイメージが湧かない 極力使わないように回避出来ちゃいそうな気もした
53 :
デフォルトの名無しさん :2010/07/02(金) 08:51:15
>>52 アセンブラのインデックスアドレッシングを拡張したもんだから、
ハード寄りのプログラミングでは必須。
目次なしでも辞書を引けないことはないが時間かかり過ぎ、というのと似ている。
>>52 配列とインデックスを別々にすると変数や引数が多くなって気持ち悪い
int *ip; int n=1; ip += 1; ip += n; ip++; 上3つは全て等価だろうか? int ia[10];int *ip = ia; for(int n=0; n != 10; n++){ *ip + n = n; ip[n] = n; *(ip + n) = n;} 上3つは全て等価だろうか?
>>55 そんなゴミみたいな例を出すからわからなくなる
大事なのはイメージ
物理メモリにおいてある領域にアクセスしようとしてる仕組みのイメージを
脳みそに描くことであって
そんな読みにくいコードが読めるかどうかはプログラムの能力とちっとも関係ない
>*ip + n = n; なんだこりゃ
よくわかってない人間が、うっかりすると書いてしまうことがある。 だから、なぜ望んだ動作をしないのかわからず、引っかかるんだ。 読みやすい書き方を提示してやれば、ポインタで引っかかる人間は激減する。
>>55 後半はバグっていて話にならない(
>>57 のとおり)
前半は何が聞きたいのかわからない。ポインタに1を足してるという点では等価だが、
評価順序とかアセンブラコードとか言い出したら等価ではないが、質問のポイントが書いていないので
答えようがない。
結論:
>>55 はど素人。引っ込んでろ。
初期化してないものを演算するな
>>60 上のはメモリに対する演算ではなく、ポインタの値を変化させてるだけだから、
参照してない以上、何も問題はない。
>>59 いちいち罵倒せずにはいられんのか。情けない奴。
ポインタ自体は説明すれば分かってくれる。 問題は表記だな。 Cのポインタ関連の表記が悪い。
>>62 罵倒?
ど素人に引っ込んでろと書いたら罵倒なのか?
あんまり詳しくない方は書かないでもらえますか、とでも書けばいいのか?(ワラ
>>63 でも代替案も却って煩雑だったりするし、
今更ポインタの文法が変わるわけがないので、
慣れてちょうだいとしか言いようがない。
一番いいのはC言語使わないことだけどね。
>>66 *ip + n = n;
とか書いちゃうバカに言われてもwwww
>>67 それは私ではありません。横で見てて見苦しいので
重ねてバカを晒さないようお願い申し上げます。
>>64 罵倒だとも。それがわからない程度の人間というわけでもあるまい?
>>63 BASICにもVARPTRとかVARSEGとかあるけどCのポインタで躓く人には
分からないよ
アセンブラ学んだりしてメモリの存在意識してからじゃないと
スっと理解するのは難しいんだろうな
いきなりC言語から始めると値しか視界に入らないと思う
71 :
デフォルトの名無しさん :2010/07/03(土) 02:29:03
ジョエル・オン・ソフトウェア
http://bit.ly/8YScmk ">私のささやかな経験から言わせてもらうと、伝統的に大学のコンピュータ
サイエンスのカリキュラムで教えられているもので、多くの人がうまく理解
できないものが2つあった: ポインタと再帰だ。
"
短縮URLなんて全くクールじゃねえから。ソースはきちんと示せ
>>71 サンクス。クール内容だった。ポインタの理解が能力の見極めになるのはアセンブリ
言語を知らない人だけだろうな。知ってればレジスタ間接のようなものかですむ。逆に
アセンブリ言語知らなくてポインタを理解できる人は尊敬してしまう。どうやって理解に
至ったのかぜひ聞いてみたい。
俺ほとんどアセンブリ言語知らないけどポインタは理解できたぞ。 はっきりいってポインタはアセンブリ言語知ってるどうのじゃなくて、 どういう型システムがあってああいう文法になるのかっていうことが ポイントなんだと思うよ。
再帰が理解できないって 高校数学で数列が理解できないレベルだろ 池沼認定
アセンブラなんて知らなくてもポインタは理解できるだろ 大事なのは、いまいじってるデータが物理メモリのどっかにおかれていて そこにアクセスする仕組みがポインタだと理解することだ 一番面倒なところはプログラムの記述といくらにらめっこしてても理解できないっていうところかなぁ? 概念を理解しないといけないってのは義務教育ではあまりないからな 再帰はツリー構造のデータをいじらせれば一発で理解できるから そんなに重要じゃないと思うけどね はじめに教える人間がタコだと苦労しちゃうのかも・・・
>>74 .,
>>76 アセンブリ言語知らないと理解できないとは言っていない。知っていれば少しも
難しいものではないと言っている。ほとんど知らないとかではなくまったく知らない
人がどう理解していったのかを知りたい。そういう人を知らないので聞いてみたい。
>>77 データはメモリにおいてあるんだからそこにアクセスする方法が必要になるだろ?
って必要にせまられたら一発で理解できたけど
画像とかさ
Aパートでロードした画像をBパートでもう一回使いたいんだけど?的な場面で
>>77 俺も同意。
BASICから初めて、その後C言語をやった時はさっぱりだったが
情報処理試験のためにCASLをやったらあっさり理解できるようになった。
配列とポインタの違いとかもアセンブラ知ってると理解が速いわな。 char *ptr; char buffer[10]; _ptr: dc.l 0 _buffer: ds.b 10 puts(ptr); puts(buffer); move.l _ptr, -(sp) #ポインタを引数に渡す場合 jsr _puts lea _buffer, -(sp) #配列を引数に渡す場合 jsr _puts addq.w #8, sp
むむっ久しぶりなんで間違えた予感。 >lea _buffer, -(sp) #配列を引数に渡す場合 こんなアドレス形式はないな。 pea _buffer だった。失礼。
テレビの中身を知らなくても、テレビを操作することはできる感じでポインタは使うことはできたけど レジスタとメモリがどのように使われるのかを体感したら、理解が深まったのは覚えてる
おまえらレベルの低いところで理解できたつもりになっててワロスw
週末だというのに煽るしかやることがないなんて可哀想
賞賛からはほとんど得るものはない 人間は失敗から学んで成長する生き物
>>78 サンクス。
> データはメモリにおいてあるんだからそこにアクセスする方法が必要になるだろ?
それだとポインタを使わなくても普通にできることでは?
> 画像とかさ
> Aパートでロードした画像をBパートでもう一回使いたいんだけど?的な場面で
言語を限らないとメモリーコピー、参照渡しなどポインタを使わない方法もある。
意地悪く書いたけどポインタを使えていると思う。上から目線ですまぬ。
> データはメモリにおいてあるんだからそこにアクセスする方法が必要になるだろ? そこにアクセスする方法にアクセスする方法だろポインタっつーのは
どうやって理解するも何もそのままだろうが
理解出来ない奴が理解出来ない
番号付き箱にモデル化されたメモリでの説明で理解した。 アセンブリは知らん。
>>87 ポインタってそんなに複雑なものか?
>>80 のコンパイル結果を見ればただの long 定数があるだけだ。
これがポインタの正体。つまりポインタは int や long と同じようにメモリ上にとられる
エリア。それをデータのアドレスを入れるために使う。
このプログラムはぬるぽになる。本来の意味でぬるぽが使えるのに感動!
>>86 >それだとポインタを使わなくても普通にできることでは?
できねーだろ
ハンドル使ってもポインタは回避できないと思う
知らないことは誰だってあるけど、いいやんとか言って違いも調べず思考停止するやつは向上心もう少し持とうぜ
と、アンカーの使い方もわからない雑魚がいっております
>>92 aというメモリー上のデータにアクセスしてbという変数に入れる。
b = a;
どこにポインタを使う必要がある?
>>95 元の話題とまったくつながりが見えないんだけど
なんの話してるの?w
>>95 は固定長データしか扱ったことないんだよ
そっとしておいてやれよ…
アセンブリ言語を覚えたら 複雑な関数ポインタをスラスラ書けるようになっちゃうの?
アドレスにまつわる文法はアセンブリの方がわかりやすいかもな
>>96 >>87 でポインタを使わなければデータにアクセスできないと書いているからそりゃ無い
だろと書いている。揚げ足取りに近いのだけど。
Cを使って大量のデータを使いまわしするということならポインタについてはそうだろう。
だから後に言語を限らなければとわざわざ書いている。
>>98 スラスラ書けるようになるかどうかはわからないがCでなぜポインタを取り入れたか、
何を実現したかったは理解できるようになる。
>Cでなぜポインタを取り入れたか、 >何を実現したかったは理解できるようになる それはポインタの理解とは無関係
>>101 俺も同意見。
アセンブリ言語を理解したとして、
結局あの文法が正確に分かるようになるのかというと別だと思う。
たとえば、 const int (* const (*)(void))(void) なんてのがあったとしよう。
これはアセンブリ言語を知っていたとしても
ポインタ自体に精通していなければ文法上の
意味を正確に掴みかねると思われる。
裏でどういう値が動いているかは分かると思うが。
俺自身アセンブリ言語に詳しくないから
そう思うだけなんだけど識者の人はどう考えているの?
ポインタで引っかかる人はアセンブリ言語を知っていれば簡単にわかる程度の ところで引っかかっていると思うが。
アセンブリの感覚だと、とりあえずvoid*にして後でキャストすればいいと思えるし constはただの自己満足だとわかる。 必要最小限のサブセットと、自己満足のための高度な内容を分けて考えるべき。
105 :
デフォルトの名無しさん :2010/07/03(土) 20:33:39
int x=100; int y=50; int *p=&y; int z=x/*p; どうしてCはポインタの記号に*を採用したのか*/
>>104 あくまで
>>102 は例だ。
本当に箱を並べたポンチ絵程度のものが分からないことを
アセンブリ言語を知っていれば分かるようになるのか、
それともC言語的な文法も踏まえたことをアセンブリ言語を
知っていれば分かるようになるのか、そういうことを言いたいだけ。
ポインタで引っかかるって言っても、
↑の2種類はいると思うけどね。
ポインタと文法は別 ポインタはわかるが文法がわからないという話は聞いたことがない だから文法自体は悪くない 悪いのは、文法以前で引っかかってる人に文法を踏まえさせること
ポインタの配列とか ポインタのポインタとか 配列のポインタとか 言ってる人は判ってるなと思うけど ダブルポインタって言ってる香具師は 馬鹿じゃねーのと思う
int **x; ←*が2つついてダブルポインタ ではないの?
110 :
108 :2010/07/03(土) 21:51:01
そうだよ
ポインタ自体は難しいと感じないんだけど ポインタの配列、ポインタのポインタ 使うプログラムって難しく感じる 配列が多次元になったり 連結させたりするとな
おまいら int main(int argc, char *argv[]) を使う派? それとも int main(int argc, char **argv) を使う派?
後者。引数のブラケットほど虚しいものはない。
Javaですねわかります
なぜJavaで引っかかる人が少ないのか typedef void *Object; typedef Object *Vector; int main(int argc, Vector argv)
*p=10000;
めんどくさいときは構造体に入れて関数に渡すわ
yes we can
119 :
デフォルトの名無しさん :2010/07/04(日) 09:11:19
int* p; int *p; int * p; アスタリスクの使い方がポインタの理解を阻害する
>>100 >
>>87 でポインタを使わなければデータにアクセスできないと書いているからそりゃ無い
>だろと書いている。揚げ足取りに近いのだけど。
は?
>>87 俺じゃねぇし
まったく話が見えない
都合悪いから「俺じゃねぇ」ですね
>>119 俺のポインタ周りの混乱はそうだったなぁ。
俺の混乱履歴。
char *p;
char型の*pを宣言してるように見えてた。
char i;
i=1;
が正しいのだから、
char *p;
*p=1;
が正しくない理由が分からなかった。
加えて
char *p="abcd";
こんな表現もあって混乱した。*pにいきなり値を代入してるように見えるから。
実際にはpに代入していて、char*型のpを宣言してるんだと感じてから、
char* p;
と書くようにした。
そして
char* p,r;
でまた混乱。
意味を考えなくても文法知識をいじり倒すだけで正解を導けるという 思い込みのはげしい人が多いのはなぜか。
規格書が全てですが何か コンパイルできるかできないかはシンタックスが正しいか正しくないかであって実行して悪魔が出るかどうかはセマンティクスに違反しているかしていないか ポインタなんて単なる数値だよ *をいくつ使うかで表現できる階層が変わるだけ、C言語でオブジェクト指向っぽく連想配列でも実装すれば理解できるよ ポインタは複合型であり、構造体や共用体、ポインタ、配列などの複合型の不完全型をシンプルに表現できる抽象データ型なだけであって難しいことなど無い
Cを始めるにあたって、規格から調べる人は控えめに言ってマレでしょう。 普通は入門書や、入門webサイトを見る。 初めて見るソース、ソースから受ける第一印象はとても大事。
難しいことを難しく表現することは誰にでも出来る。 難しいことをやさしく表現することは、能力が無いと出来ない。 能力が著しく足りないと、簡単なことを難しく表現してしまう。 という言葉を思い出した。
>>119 C++ が普及する前の C のときは事実上
FILE *f;
だけだったはずで
FILE * f; とか
FILE* f; とか
書きかたしていると笑われたもんだ
なぜか C++ で本来の
HOGE *h;
は影を潜め
HOGE* h;
という書き方が横行するようになった
char* p,r; でまた混乱。 なんで*が実質右ってことを忘れてんの?
>>123 > char *p;
>char型の*pを宣言してるように見えてた。
それで合ってる
何か変な本読んで混乱させられたんだな
かわいそうに
ここは"なぜポインタで引っかかる人が多いのか"を考えるスレ。 初心者が起こす勘違いを考えるスレ。
対象者が何につまずいているのかを想像する能力がなければ、 真の意味で教えることは難しい。 教える側に対象者のつまずきを理解する能力が無い場合、 対象者が理解できるかどうかは対象者の能力に依存する。
>>133 そうだな
簡単な話が理解出来ないのは
耳が悪いのかも知れないし
そもそも話を聞いてないのかも知れないし
頭が悪いのかも知れない
ポインタは正常に動くのにたまに変になることがある int a[10]; a[10]=100; みたいにやると、稀におかしくなる コンストラクタとかifとかで複雑になってくると よく見ると int * p; *p=100; みたいなことを平然とやってたりする。 短いプログラムだとこんなことはしないだろうけど 複雑なプログラムだと初心者はわけがわからなくなる。 これが初心者にとって難しいんだろうな。
>>135 > a[10]=100;
正常に動くことがあるから、誤解の種になるよね。
必ず異常動作すれば気づくんだが。
char *p="abc";
*p='A';
これが正常に動く環境があって、
しばらく平気でそういう記述してた時期があった。
char* pの書き方だけは許さない。 こういう書き方をしている人物こそポインタで引っかかったままだと言える。
>>137 Microsoftのソフト開発者は全員そうだな。
結構、ポインタって文法の丸暗記で何とかなったりするからなぁ int (*ptr)[10]; int a[100]; //mallocの代わり ptr = (int (*)[10])a; ↑これを理解するのに結構時間かかった、特に関数の引数にしたときとか アドバイスもらって、アセンブラ入門を勉強したら理解できるようになったんだよな 多分、実際のメモリ範囲確保とインデックスによるアドレス指定が理解の助けになったと思う
>>137 char *p[10];
char (*p)[10];
*がpに付くのなら上記は同じ意味になるだろう。
*が宣言に付くからこそ(*p)とすることで違う意味になる。
そういう意味からは
char* p;
のほうがしっくり来る。
俺は char *p; と書くけどね、char* p0,p1; のことがあるからね。
やたら記述にこだわってて内容理解してそうにない書き込みが多いところが恐怖
C#では int* p,q,r; ってやったらpもqもrも CやC++でいう int *p,*q,*r; みたいに、全部が*つけたようになるな アンセーフモードでしか実行できないけどね Microsoftのポインタに対する意識が明確に現れてるな
>>126 第一印象に敏感な人は重症化しやすい、という意味では大事かもしれないが
本来は、印象なんてあやふやなものには鈍感なほうが良い
>>141 上のほうで、文法より意味を考えようぜっていう正論が出たから
逆の書き込みをすれば色々釣れるだろうと思ってるのかもしれん
char *p; char (*p); 上記は同じものを指すが、結果として同じものをさすだけであって、 その意味合いは異なる。 前者は"char*型のpの宣言"であり、 後者は"pはポインタであって、そのポインタはcharを格納するアドレスのポインタ"。 それが理解できなければ、 char *p[10]; char (*p)[10]; は理解できない。
ホント記述弄り好きだな こんなの覚えてもしょーがねーだろ
>>145 しょうがなく無いぞ
char *p[10];
char (*p)[10];
どっちの意味でも使うことあるから。
宣言の*が型に付くことを知っていれば、
前者のpが配列、後者のpがポインタだと理解しやすい。
int a[10][10];
それは別にポインタが難しいからじゃなくて、C言語が難しいから。 文法云々というか、プログラマの責任がデカい部分。
>>146 *(p[10]) か (*p)[10] かの違いだよ。前者の括弧は冗長だから書かないだけ。
*は前置単項演算子だから、被演算子は常に後ろに付く。型に付くことはない。
なにいってるんだ int * p の*は演算子ですらないぞ *p=&x みたいのでpの←の*は演算子だけど。
>149 知ったかは口はさむな
説明する能力がなければ、真実を伝えることが難しい例。 説明する能力があれば、真実でなくても実質問題の無い考え方を伝えることが出来る例。
mallocだのなんだの使うより new int のほうがわかりやすいよね
>>144 意味合いも型も全く同じ。
どちらも*演算子を適用するとcharになる。
>>146 の意味合いは演算子の
結合が異なるから意味が異なって当然:
・char *p[10] =>char *(p[10]) => pに[]したあと*すればcharになる
・char (*p)[10] => pに*したあと[]すればcharになる
変数の記述と同じように演算子を適用したら、
最終的に一番左に書いたcharやらが取得できるのがポインタの文法。
ポインタで引っかかる人が多い、というより Cのポインタで引っかかる人が多い、の方が正確だろうな。 C以外でポインタで引っかかる人の話はほとんど聞かない。 Cのポインタも、言語規格を読めば別に難しくないが、直感では理解しにくい。
とりあえず複数変数宣言を一行でやるのやめろや・・・ 見づらくて仕方ない。
>>131 あってないだろ
*pが変数名ならポインタ使用するときpって書くなよ
*とpを分解するなよ
*とpがくっついて初めてIDなんだろ?って思った
あくまでpはchar*型の変数だろ
だから*演算子でポインタpの値参照できるんだし
↑ char *p *pはchar型であってるよ *pは変数名じゃないよ pが変数名だよ。
char * p 「char型」の「ポインタ(*)」の「変数p」と言った方がいいかな。
*pの型がcharとかは関係なくて char *p;で宣言される変数はp てだけだろ。 *pは宣言されてない。 配列とかでも同じだからなぁ。 特に分かりづらいと思ったことなかった。
幅WID, 高さHEIのint型の二次元配列arrayを動的に確保したい。 1)arrayの宣言文を書け。 2)arrayのメモリ確保文を警告レベル最大でも警告が出ないように書け。 3)arrayの要素アクセス方法、具体的にはarray中の横からw番目、縦にh番目の要素に0を代入する文を書け。
>>159 char*型の変数pでいいよ。
変数名取り除いたのが型の名前だし。
>>161 とりあえずC99禁止しようか。
>161 二次元配列ってなんだ 配列の配列か 一次元的に取られたメモリを工夫して二次元的に扱える形にしろというのか 一次元的に取られたメモリを工夫して三次元的に扱える形にしろ ただし、外側から縦、横、奥を表すとし、メモリの確保は1回で実行せよ また、型は適当なものをtypedefし、type_t型として作成せよ ただし、縦、横、奥は実行時に与えられ、特定のtype_t型に依存するコードは書いてはならない 縦、横、奥はsize_t型を仮定してよいものとする 全ての縦、全ての横、全ての奥のアドレスに対してそのアドレスがそのアドレス以外のものと異なるようにせよ 全ての縦、全ての横、全ての奥の要素に対して任意の値を代入し不正なメモリ書き換えが行われないようにせよ メモリの確保には標準関数のvoid *malloc(size_t)を使用せよ ここで、言語仕様はC89またはC90を仮定してよいものとする ただし、コンパイラの拡張を使用することを禁止する 例 : type_t ***array3; array3 = alloc3(h, w, d); array3[h][w][d] = val;
前の問題の答も出さずに勝手な問題積み上げてバカじゃないの
スレタイとは何の関係もない話題になった。
*********************************************p
*は2つあれば十分 3つ以上使うコードがあっても、void**とキャストを使うコードに変換できる
変換できることに意味はないな。適当にtypedefし直せば充分。
バイナリエディタで機械語書かせれば一発で理解できると思うんだ
興味が無ければ答えない。当たり前だと思うが。
今まさに仕事でその手のコードを書いているからめんどくさいw CPUとGPUの違いやsimdの演算効率の違いを考慮して、 AofSとSofAを切り替えたりxyzの順序を変えたりの詰め替えばっかりw
>>161 1) int *array;
2) array = (int*) malloc(sizeof(int) * WID * HEI);
3) array[h * WID + w] = 0;
Java使いでちゃんとしたCのコードなんか書いたことないんだけど、
これでいいのかな?このスレにいる詳しい人添削頼む。
あと
>>163 も同じように処理しちゃまずいのかな。
>>174 2次元配列と書いてるから、array[WID][HEI}の形にしないと駄目だろうね
>>161 の条件だと↓でもよさそうだけど、
>>163 はひと手間かけないといけない
int (*array)[HEI] = (int (*)[HEI])malloc(sizeof(int) * WID * HEI);
一回でメモリ確保せよとはあるけど必要充分にとは書いてないから、 3次元分のメモリとポインタ分のメモリを確保してポインタの配列をこさえればいいんだけどね。
>>163 >ただし、縦、横、奥は実行時に与えられ、特定のtype_t型に依存するコードは書いてはならない
これは↓がだめだという意味?
typedef char (*type_t)[w][d];
type_t array = (type_t)malloc(sizeof(int) * h * w * d);
178 :
177 :2010/07/05(月) 22:17:04
1行目 typedef int (*type_t)[w][d]; ね。
typedefしなくていいのなら、
for文で回しながらmallocして、
for文で回しながらfreeすれば出来そうだけど。
(そこまでするんならいっそ1次元でとって、添え字演算したほうがすっきりしそうだが・・・)
>>163 は出来るのかね。
いや違うかこれ以上簡単にならんな 関数は配列を返す事が出来ないんだから、配列へのポインタを 返すと書かなければならない 本当はそうではないのだが、Cの文法がそう書く事を許さない仕様に なっている こういう場合は関数宣言の方を変えて、配列を返すのではなく関数への ポインタのポインタを返すと書けば簡単になるだろう 但しそう書くとプログラムが何をやっているのかますます不明瞭になり、 ほとんど読み解く事が困難になってしまう
array[h * WID + w]でいいよ ひとつ言えることは ポインタで引っかかる人は、文法や型にこだわる 問題を出されると出題者に盲従する傾向がある だったら、文法や型や規格書などに従うのを一度やめてしまえばいい たとえばアセンブラを使ってみるとか
それでいいなら、int array[W][H];も必要ないね
[W][H] じゃなくて [H][W] だというところには誰も突っ込まないんだなw
言いたいことそんな所じゃないだろうからな。
メモリのスキャン方向が変わるから実行速度等に影響が出るんだが
188 :
184 :2010/07/06(火) 11:56:51
>>175 から持ってきて、あんまり考えずに成形しただけだから気にするな
>array[h * WID + w]でいいよ だけだったら異論はないよ
あとあと、楽とわかってるから静的に2次元配列として宣言するんだよね
動的確保でも、その楽にする方法があるから、(言い方を借りれば)文法にこだわるんじゃないか
まぁ、これも>問題を〜の1文がなければ、そうですねって話だけど
>>184 array[WID][HEI] と array[h * WID + w] は等価ではない
191 :
184 :2010/07/06(火) 12:51:07
192 :
デフォルトの名無しさん :2010/07/06(火) 12:56:11
アホ曝しage 184 デフォルトの名無しさん [sage] 2010/07/06(火) 10:38:36 ID: Be: それでいいなら、int array[W][H];も必要ないね
>>183 良くない。
array[h][w]という表記ができない言語(例えばアセンブラ)ならそれでもいいが、
言語仕様で出来るのにやらないと、後から読んだ者を混乱させる。
理由があってわざとやらないのか、書いた奴が知らなかっただけなのか判断つかないからな。
それに、hとかwとかなら array[h*WID+w]でもarray[h][w]でも大差ないが、もっと複雑な式に
なった時に、[h][w]形式の方が断然読み易くなる。
ぶっちゃけ、二次元ポインタを使いこなせない奴の言い訳にしか見えない。
どうでもいい。 個人的にはベタバッファにポインタアクセスのが分かりやすいし、使いやすい。 C++は知らんがCなら慣れの問題。
array[i][j] と記述したくなるのは、ベクトルやマトリックス周辺だけかな。 数式をそのままに近い形式で記述したい という意図だけどね
結果あわせるだけならどっちでもいいんだけれども 配列のアライメントや入れ替えがあるから、用途によるよね
array[4][5]とarray[5][4]は違うと言っている。
こういう書き方はどうか?(長文失礼) ・char , int は型である。 ・Tが型ならば、T[N]は型である。(Tの配列と解釈する) ・Tが型ならば、$Tは型である。(Tへのポインタと解釈する) ・T,U1,U2,…,Umが型ならば、T(U1,U2,…,Um)という記号列は型である。(関数と解釈する) ・宣言方法は「T a;」とする。 具体例(カッコは適宜省略) C言語での表記法 ↓ ↓ int a; ( int a;という意味) int[N] a; ( int a[N];という意味) (int[w])[h] a; ( int a[h][w];という意味。wとhは逆になる) $int a; ( int* a; ) int(char) func; ( int func(char); ) $($int) a; ( int** a; ) ($int)[N] a; ( int* a[N]; ) $(int[N]) a; ( int (*a)[N]; ) $(int(char)) f; ( int (*f )(char); ) ($int)(char) f; ( int* f (char); ) int($char) f; ( int f (char*); ) ($(int(char)))[N] f; ( int (*f [N])(char); ) ($$(int(char)))[N] f; ( int (**f [N])(char); ) $(($(int(char)))[N]) f; ( int (*(*f )[N])(char); )
ご苦労さん
>>193 記述の問題ならマクロ導入すりゃいいでしょう
まあな。w
ポインタに1足したら、2byteとか4byteとか進む? ポインタってアドレスなんだろ? そんなもん1進むに決まってるんじゃないのか?
オブジェクト(要するに型)のサイズ分進む。 sizeof演算子で進む量を得ることが出来る。
>>197 判ってて聞いてるんだろうけど
array[h * WID + w] と array[w][h] はポイントしているアドレスが違う
array[h * WID + w] と array[h][w] が同じ場所を指さないと
共同開発してる仲間に迷惑だろ?
そんなことも知らないからポインタスレに来てるんですねわかります
前者は連続した領域にmallocすることが求められるが、 後者は離れた領域にmallocしても一向にかまわない。
208 :
163 :2010/07/06(火) 20:58:16
>>207 まあ、あせるな
みんな、なかなかいい線いってるよ
もう少し検討が深まるのを待とう
>209 >208ではありませんが、コンパイラとコンパイルオプションを示してみてください また、可能であればコンパイラのバージョンも示してみてください
>>210 PC@Test> gcc --version
gcc.exe (GCC) 3.4.5 (mingw special)
Copyright (C) 2004 Free Software Foundation, Inc.
This is free software; see the source for copying conditions. There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
PC@Test> gcc -Wall -pedantic -std=c89 Test6.c
Test6.c:14: warning: ISO C90 forbids variable-size array `type_t'
Test6.c:14: error: variable-size type declared outside of any function
Test6.c:14: warning: ISO C90 forbids variable-size array `type_t'
Test6.c:14: error: variable-size type declared outside of any function
ごめん、今やったら通らんわ。
212 :
221 :2010/07/06(火) 22:01:09
昨日、下記のソースをいじってら通ったんでC89ならOKなんだと思ってた。 昨日の試行錯誤の過程で勘違いした模様。 #include <stdio.h> #include <stdlib.h> #define MODE (0) #if MODE == 0 int w = 10; int d = 20; #elif MODE == 1 const int w = 10; const int d = 20; #elif MODE == 2 #define w (10) #define d (20) #endif typedef char (*type_t)[w][d]; int main(void) { int h = 30; type_t array = (type_t)malloc(sizeof(char) * h * w * d); printf("%d\n",((int)&array[0][0][1])-((int)array)); printf("%d\n",((int)&array[0][1][0])-((int)array)); printf("%d\n",((int)&array[1][0][0])-((int)array)); free(array); return 0; }
てか何が出来たら「三次元的に扱え」たコトになるのよ? 条件が色々付いてるけど、根本的な要求が分からん
下記のポンタ何を指すが書きなさい 15秒以内の回答できて小学1年レベルらしい int (*a)[N]; int (*f )(char); int (*f [N])(char); int (**f [N])(char); int (*(*f )[N])(char);
>>214 日本人ではないな?
だからどう、ということもないが
m行n列の二次元配列をメモリ上に表現する場合 1 サイズm*nの一次元配列として確保 2 サイズmのポインタの配列とサイズnの一次元配列m個 1の利点 ポインタの配列分メモリが必要無い 2の利点 列を削除追加できる 2は多段のポインタなので 普通の二次元配列として扱うと遅い 中継用のポインタ使う必要があるので可読性は低い
2 はargvですね
小学生すごいな、即答出来ても15秒で書けないな
>>214 上から順に
行列の行単位で動くポインタ
関数ポインタ
ポインタ配列
ポインタ配列
行列の行単位で動くポインタ
行列やポインタ配列の要素の型、つまりテンプレート引数に相当する部分は気にしない。
昔のJavaはそうだったというかむしろシンプルでわかりやすいとか言われて普及した。
>>216 >2は多段のポインタなので
>普通の二次元配列として扱うと遅い
これマジ?
ケースバイケースだろうけど、乗算して加算して参照より、
参照してさらに参照のが速いのかと思ってた。
>>223 上から順に
ポインタ
ポインタ
配列
配列
ポインタ
なのは分かるよね。
後は、ポインタだったらポインタの部分を削除して型を見て、
配列だったら配列の部部を削除して型を見る。
そうすると、
配列のポインタ
関数のポインタ
関数のポインタの配列
関数のポインタのポインタの配列
関数のポインタの配列のポインタ
となる。
合ってる?
>>180 の
int (*((*func(void))[]))(void)
これ鬼みたいだな
しかしプログラムを読むと大して難しい事をやっているわけではない
Cの宣言はどうしてこんなに難解なのだろうか
>>222 どっちが速いのかすら分からん。
あー、もしかしてキャッシュに乗らない?
あんまりその世界で生きてないんで
どう動くのかサッパリ。
>>225 アセンブリをやれば簡単に読めるようになる
int (*((*func(void))[]))(void) は保育園レベルでちゅよ この程度すぐに解らんと保育園レベル未満の国語力ってことでちゅ
>>225 リンク先は見てないけど、そんなに難解?
ちゃんと()付いてるし。
まず一見してfunc(void)な関数の宣言だろ。
*付いてるから返値はポインタ。
さらに[]なんで配列のポインタ。
次の*で、ポインタの配列のポインタ。
で、最後にint ()(void)なんで、つまりこの宣言は
そういう関数へのポインタの配列のポインタを返す、引数voidな関数funcの宣言。
長々書いて違ってたら、凄い恥ずかしいんだが、どないだろ?
>>229 >長々書いて違ってたら、凄い恥ずかしいんだが、どないだろ?
どのように解析するか自信ないのか?
コンピラー(C言語)の構文解析ルールをよく知らないってこと?
コンピラーはfunc(void)が関数の宣言ってどうしてそう判断するん?
職業PGにとって構文解析って基礎中の基礎事項で学校で当然習っているじゃないの
それすら知らないのに職業PGやってるって人居ないよね
趣味PGなら自分が必要と感じたら勉強してみるか程度で良いけど
>>221 キャッシュへのプリロードするのに間接参照だと予測不能。
書き方を修正。「$T」じゃなくて「T$」に変えてみた。 ・char , int は型である。 ・Tが型ならば、T[N]は型である。(Tの配列と解釈する) ・Tが型ならば、T$は型である。(Tへのポインタと解釈する) ・T,U1,U2,…,Umが型ならば、T(U1,U2,…,Um)という記号列は型である。(関数と解釈する) ・宣言方法は「T a;」とする。
以下、具体例。(長文失礼します)
具体例(カッコは適宜省略) C言語での表記法
↓ ↓
int$ a; ( int* a; )
(int$)$ a; ( int** a; )
(int$)[N] a; ( int* a[N]; )
(int[N])$ a; ( int (*a)[N]; )
(int(char))$ f; ( int (*f )(char); )
(int$)(char) f; ( int* f (char); )
int(char$) f; ( int f (char*); )
((int(char))$)[N] f; ( int (*f [N])(char); )
((int(char))$$)[N] f; ( int (**f [N])(char); )
(((int(char))$)[N])$ f; ( int (*(*f )[N])(char); )
$が入ってないものは
>>199 と
全く同じになるので省略しました。
>>232 の定義に
・Tが型ならば、T[ ]は型である。
をつけ加えると、
>>180 の
>int (*((*func(void))[]))(void)
これも232の形式で書けて
((((int(void))$)[ ])$)(void) func;
となります。
また、T[N]$ と書いたとき、これを(T[N])$と読むことは出来ても
T([N]$)と読むことは出来ないし、T$$とかT$[N]なども同様のことが
言えます。よって、ある程度カッコを省略できて、結局、
次のように表記できます。
((int(void))$[ ]$)(void) func;
関数ポインタ関連の表記は参考サイトや本を読みながらじゃないと読めん あれ何とかならんかったんか
typedef すれば大抵解決
>>230 煽りか本気か知らんが、職業PGに夢見すぎ
関数ポインタが読みにくいという事はCでC++モドキのプログラムを バリバリ書くのは不可能に近いって事ですね それか無駄な時間を多量に要するか
>>238 他の言語に比べて積極的な気持ちになりにくいかな?
ただ、昔読んだテキストエディタのソースでは、入力文字のアスキーコードを引数にして各入力処理関数に飛ぶように、関数ポインタの配列が使われていて感動した。
日本語処理をさせるのは難しそうだけど。
perlよりマシだな
エセクラスならgtk2が頑張ってなかったっけ?
gtk++/gtkmm よりも gtk の方が出来が良いという
>>229 あってますね
規則に従って変換したら、
>>234 になったw
int(void)*[]* (void) func
>>235 >>232-234 の書き方だと、日本語の語順そのままで
宣言できます。たとえば、
・(戻り値int,引数charの関数)へのポインタ
が欲しかったら、そのまま
(int(char))$ pfunc;
と書けます。
>>180 の
>int (*((*func(void))[]))(void)
だったら、これを日本語読みすると
(((戻り値int,引数voidの関数)へのポインタ)の配列)へのポインタ)
を戻り値とする、引数voidの関数
となるので、この語順そのままに
(int(void))$[]$ (void) func;
と書けます。
>>244 君は度々出て来るようだけど、結局主張は何?
>>245 型の表記は関数型言語スタイルがいいですね、ということです
typedef の方が判りやすいよ
>>248 typedefの方法は
>>232-234 と同じようなもんです。
あなたにとってtypedefが分かりやすいなら、
ほぼ同程度に232-234も分かりやすいことになります。
たとえば、C言語で
typedef int (*PFUNC)(char);
と書いておけば、もはや
PFUNC pfunc;
と書くだけで「(戻り値int,引数charの関数)へのポインタ」が
宣言できるようになりますが、このときの「PFUNC」という記号列は、
>>232-234 で言うところの「(int(char))$」という記号列と
全く同じ役割をしています。つまり
Func ≡ (int(char))$
ということです(誤解を招きそうな式ですが)。
別の言い方をすれば、「PFUNC pfunc;」と同じ語順の宣言が
最初から可能なのが
>>232-234 ということです。
ou.... × Func ≡ (int(char))$ ○ PFUNC ≡ (int(char))$
んなことは言われんでもわかっとる (int(char))$ よりも PFUNC の方が判りやすいだろって話
>>246 よく分かった。
新しい言語、文法を語りたいだけなら、引っ込んで黙れ。
そういうスレがあるし、そっち行け。
ポインタが分かりづらいのはCの文法が悪いから、とか
それなりのコト言うのかと思えば、ただのアホかよ。
>>251 どうですかね。「PFUNC」と書いてしまうと、中身が
見えなくなって抽象的になってしまう弊害があります。
(int(char))$だったら、そのまま読めます。
しかも日本語の語順のまま読めます。
僕にとってはint(char)$ の方が読みやすいです。
ただし、PFUNCという記号列を日常的に
typedef int (*PFUNC)(char);
の意味で使っている人だったら、その人にとっては
PFUNCという記号列は全く抽象的では無いわけですから、
その人にとっては、PFUNCの方が読みやすいかもしれません。
ひとつのプロジェクトに 独自のGC、newが複数混在してると 最高にわかり辛い それぞれが読みやすいものでもな
>>252 >ポインタが分かりづらいのはCの文法が悪いから、とか
>それなりのコト言うのかと思えば、ただのアホかよ。
え?それって同じことですよね?
「ポインタが分かりづらいのはCの文法が悪いからだ!」
という意見を煮詰めて行ったら、どのみち
「では、文法をこのように変えたらどうだろうか(例えば232-234のように)」
という意見にならざるを得ないでしょ?
// それぞれが読みやすい typedef int (*PFUNC)(char); typedef PFUNC T[N]; // 混在すると分かり辛い typedef int (*T[N])(char);
混在してると分かりづらくなる文法が悪い?どっちが悪い?混在が悪い?文法が悪い?
>>255 ならんよ。
だってその表記を見ても、Cのポインタが
理解できるようになりそうにないもの。
言語変えて済む話ならポインタのない言語を選べばいい。
>>258 言ってることが矛盾してませんか?あなたにとって、
>ポインタが分かりづらいのはCの文法が悪いから、とか
↑この意見は"それなりの意見"なんですよね?(
>>252 にそう書いてあるし。)
でも、これは「Cの文法を改良すればポインタが分かるようになる」と
言っているのと同じことですよ。
一方であなたは
>言語変えて済む話ならポインタのない言語を選べばいい。
と言ってますから、結局あなたは
「Cの文法を改良すればポインタが分かるようになる」
という意見を全否定しているわけです。つまり、あなたは
>ポインタが分かりづらいのはCの文法が悪いから、とか
↑この意見を"それなりの意見"としつつも全否定しているわけです。
●Cのポインタの文法は分かりづらい ●文法を変えたらもはやCではない これは矛盾しないぞ
>>260 これはひどいww
というのはさておき、
>●Cのポインタの文法は分かりづらい
これは単に文法の煩雑さを嘆いているだけであって、
>ポインタが分かりづらいのはCの文法が悪いから、とか
↑この意見とは全く別物ですよね。
>>259 それなりの意見というのは、少なくとも
ポインタで引っかかる理由を言っている点においてそれなり。
それを解決する方法として、別の文法を考えるという
アプローチは全否定してるよ。
理由はCのポインタの理解に役立たないから。
そんなに矛盾してる?
全否定はつられて言いすぎた。 なにがしかのモデル的に、理解の役に立つならいいんじゃないの? 「僕の考えた宣言の文法はスバラシイんです」 だから何なんだよ?
Cを普通に勉強すれば済む話。 どっかの馬の骨の自分ルールはもうお腹一杯。しかも読みやすくなってないし。
>>263 >「僕の考えた宣言の文法はスバラシイんです」
>だから何なんだよ?
それだけです。少なくとも
>>235 みたいな人の役には立ちますよね。
まあ、文法なんて結局は慣れの問題ですが。
>>264 あなたはCの文法に毒されすぎです。結局は慣れの問題に終始しますが、
それにしてもCの文法は「変」ですよ。誰でも一度はそう思うはずです。
熟練するほど忘れていく感覚ですが。
>>264 >どっかの馬の骨の自分ルールはもうお腹一杯。
「自分ルール」と呼べるほど独りよがりな書き方では無いでしょう。
日本語読みの語順に忠実な表記を採用しているだけです。
ていうか、日本語読みの単なる省略記法です。
ポインタの宣言を日本語読みしたら、「intへのポインタ」
「(戻り値int,引数charの関数)へのポインタ」のように、
必ず「Tへのポインタ」という言い回しをするわけです。よって、
日本語読みに忠実なポインタ宣言の表記法は「T$ p;」とか
「T* p;」になります。
配列の宣言は必ず「Tの配列」という言い回しをしますから、
「T[N] a;」とか「T[ ] a;」という表記法が忠実です。
C#の場合は、配列の宣言はT[ ]ですよね(採用した理由は違うでしょうけど)。
関数の宣言は「戻り値T,引数U1,U2,…,Unの関数」と言いますから、
「T(U1,U2,…,Un) func;」という表記法が忠実です。
これがそんなに「自分ルール」ですかね?
(長文ゴメンナサイ)
さて、どのような宣言の日本語読みも、構文解析のように分解すれば
「Tへのポインタ」「Tの配列」「戻り値T,引数U1,U2,…,Unの関数」に
分解できますから(ただし、ここでは構造体やクラスは考えないことにします)、
これらに忠実な表記を採用すれば、自動的に日本語読みと同じ語順の宣言になります。
この過程をよく見ると、単に日本語読みの省略記法を作っているだけの話ですよね。
>>229 が良い例ですが、宣言されたポインタを理解するとき、あるいは相手に説明するとき、
やっぱり"日本語読み"に直して理解するわけです。それは「
>>232-234 の表記法に直して理解する」
と言っているのと同じことです。なぜなら、232-234は日本語読みの省略表記だからです。
僕は232-234を強要するつもりはありません。
でも、232-234を無闇に否定する人は、日本語を否定してるのと同じことですよ。
>それだけです。
だから無意味だっつってんの
>少なくとも
>>235 みたいな人の役には立ちますよね。
立たない。
結局Cの宣言は読めないまま。
日本語でしか通用しない新規ルールイラネ。世界中で通用するCをマスターした方がいい。終了。
Cの文法はよくわからない。たぶん頭のよくない人が作ったんだろう。 よし頭のいい俺がもっといいルールを作ってやろう。 ↓ 出来た!これで皆便利になるし、俺に感心するはずだ。 たぶんこういう思考回路なんだろうな・・・。 まさしく中二病ってやつだな。 見てるこっちが恥ずかしくなるよ。
他人のオナニーを見たいわけがない罠
俺は目に入るコード量が減るからtypedef使うけど、最初からC言語がその文法だったらと思うな 日本語読みというより、型の遠い順だね(Cもそうなんだけど見ため的に)
>>232 が得意げに自分ルールを披露したら賞賛されるどころか袋叩きにあっててワロタ
Cの時代は終わってる
すきにすりゃいいんだけどな typedefしようがスマートポインタ使おうが 他人が使う事になれば苦労するって話
確かにコーディング規約も命名規則もないところで、typedefは使ってほしくはないな
個人的はtypedefを探すのにかかる時間次第かな。 2、3分くらいで見つかるようならtypedefして貰う方が読みやすい。
> typedefを探す 意味が分かんないんだけどどういうこと?
>>278 日本人ではないな?
だからどう、ということもないが
>>278 ある定義名について、それをtypedefで宣言している文を探す、ということ
>>279 日本人だよ
IDEなら言うまでもなく
非IDEでもタグファイルを作っておけば一瞬
最悪Grepしてもそんなに時間が掛かるものじゃない
探すという感覚が分からないんだよね
強いて言えば
coする程ではないが少し気になるものを
Webブラウザ上で眺めてるときかな?
と思ったりもしたけどそれはtypedefに限った話ではないしなぁ
元の定義読んで意図理解する必要があるのは変わらないからな メモリ確保がごっそり独自になってる事なんて よくある事なんだけど 仕様読んで実装全部書き直して 移植したりとかめんどくせーから嫌いです
>>265 Cの文法のどの点が変なのかわからない。多分毒されすぎたんですね。
さしつかえなければ教えてください。
>>268 >だから無意味だっつってんの
まあ、ポインタ自体の話をしたい人には無意味でしょうね。
>結局Cの宣言は読めないまま。
ああ、そういう意味なら確かに。あなたは正しい。でも、
>>235 の
>あれ何とかならんかったんか
この部分への1つの回答には なってますよね。
>>269 新規ルールではなく、既存のルールです。あなたは既に
このルールを使ってます。なぜなら、232-235は日本語読みの
単なる簡略表記に過ぎないからです。
あなたが日本語の語順で関数ポインタなり何なりを捉えたとき、
それは232-235を使っているのと全く同じことです。
あと、英語読みとも親和性は高いですよ。
日本語読みの語順を逆順にしたのが英語読みですから。
アンカミス...
× 232-235
○ 232-234
>>270 賞賛される必要はありませんが、無闇に否定されるのは
おかしいです。なぜなら、232-234は日本語読みの単なる
簡略表記だからです。これを否定するのは
>>229 の説明方法を
否定するのと同じことですよ。
>>271 オナニーと言えるほど独りよがりな文法では無いでしょう。
intがN項の配列 a;
戻り値int,引数charの関数へのポインタ pfunc;
このような日本語のベタ書きと全く同じ表現をしているのが232-234です。
こういう表現形式を無闇に否定するのは、
>>229 の説明方法を否定するのと
同じことですよ。
>>283 関数ポインタ周辺の宣言における表記の語順が、
自然言語の語順から かけ離れているところ。
日本語読みの語順と全然違うし、英語読みの語順とも全然違う。
語順はどうでもいいけど その定義の通るコンパイラ(またはプリコンパイラ)はもう実装出来てんの?
反論かと思ったら同じこと繰り返すだけのキチガイか
使用と宣言を一致させる、ってのはスルーなの?
ポインタがわからなくてこのスレ開いたが もう嫌だ何これorz
このスレって何について話すスレなの?
>>281 IDEの機能や、grepを使うのは充分「探す」だろう
>>290 たとえば、どんな所がわからないの?
引っかかってるところを(それがどうして分からないのか)知る為に、色々教えてくれると思うが
>>149 はいい線いってるね
最終的に型につくならよかった
>>292 後者はそうだね、自業自得だけど。
でも前者は違うな。
>>292 ↑というアンカーに対し292のレスを探すとは言わないでしょ?
カーソル合わせりゃToolTipが表示される(IDE) ← 探すとは言わない
クリックすればリンクに飛ぶ(tag) ← 探すとは言わない
ページ内を検索する(Grep) ← 探すとは言うかもしれないけど自業自得
>>286 演算子の結合順位が頭に入っていないと、たしかに読みづらいかもしれません。
覚えられなくてカッコを多用
>>294 int* p;は型についてるように見えるけど
int *p;はそう見えない
後者はどう解釈したらいいん?
型は宣言から識別子抜いたものだって教わりませんでした?
>>299 識別子抜いても意味合いは同じじゃない?
(int*)
int (*)
後者は型についてるように見えない
「括弧内の宣言子は、括弧なしの宣言子と同一となるが、 複雑な宣言子の結合が括弧によって変更されることがある。」 演算子の結合順序を変えるような括弧じゃない限り 論じるのは無駄。何を言いたいのかさっぱり不明。
>>301 記号操作の話ならそれでいいけど、俺は意味論的な話をしてるんだが
どうレスしたらいいものか
セマンティクスとも違う別の何かだろ。 ポインタの話ですらない。
括弧の位置なら単なる文法論の話じゃないのかな。 意味論なんて言うなら、ますます括弧なんて重要じゃない。 要するにどうでもいいような文字の置き方に さも意味があるような話を意味論って言ってるんじゃないのかね。
ポインタより重大なのが、 #define N 1 という使用方法。 N かまわず置き換える。
定数なら定数でいいし、文にまで使うのは分かりにくくなると思うけどな
>>305 へのレスですか?
定数Nを #defineするのは、辞めた方が良いよ。
関数名や、変数名で大文字のNが含まれたら全替えする。
これにより、マクロを含めて#define全面禁止をモットーにした。
本当の意味の定数でいったら、 constは変更不可能なだけで定数じゃない。 #defineが定数にあたるけど、別にどうだっていいよねー。
>>302 意味論てのが何か分からんけど、識別子を評価する方法も
型で表現しようとしてるから、*が右に寄るんじゃないか?
言葉遊びするなら、int *pは
intへのポインタな変数p、ではなく、
*pとして評価すると(実体があれば?)intの値が得られる識別子p、
という解釈。
関数や配列の宣言も全てそう。
規格の文面なんて後付けだしな。
今テキトーに考えた。
>>307 あなたの日本語は間違いではありませんが、内容は嘘ですね。
>>298 型を明示する陽表式ではなく、
「この条件を満たす何らかの型」という
陰伏式で解釈したらどうでしょう。
>int *p;はそう見えない
・識別子はpである。
・pの型は「演算子*による戻り値 *(p) の型がint型であるような、何らかの型」である。…(1)
この解釈だと、*は直接的には型に付きません。
ただし、(1)を満たす型はまさに(int*)ですから、
結果的には(int*)型の変数を宣言していることになり、
*は間接的には型についてると言えます。
そんな話してたらまたあのキチガイが出てくるぞ
316 :
302 :2010/07/09(金) 23:04:10
>>309 ,
>>312 これだ!!!こんな解釈がほしかった
やっと腑に落ちたわ、ありがとう
意味論ってのはあれだ、すまんかった
言葉のアヤってことでw
>>298 int a,b; a*b a * b a* b a *bすべて同じであるように
int* c; int *c; ・・・も同じint * cになる
みんなするどいね
例)int *a,b;は結合規則により(一度*が右結合して、型になる)
(int ((*a),)b) → int (*a), b;→ int *aと(,)int b
int *a→ int *(a)→(a)を評価するとint *が返り値つまり型となる
昔のCではint *p;のみでint* p;はダメじゃなかったっけ?
>>318 存じませんが、K&Rあたりでそう記述されてたからとかそのあたりでしょうか?
>>319 ありがとうございます、笑われただけで禁止はされていないみたいですね
wikiには方言もあったと書いてるので、これかもしれませんよ
まぁ、寝ます
ひさびさの2ちゃんなんだけどそろそろ前橋ポインタ本を越える文章って出てきた?
で、そろそろポインタでひっかかるポイント分かったのか?
・変数とメモリの関係がイメージ出来ない ・ポインタを絡めた宣言が理解出来ない(しづらい?) 今のところ出て来たポイントはこの辺?
>>323 よく挙げられる理由の典型例じゃん
これじゃ他のポインタスレと変わらんなあ…
>>322 はポインタとポイントを掛けたかっただけでは
>> ↑ これがポインタ
C言語のポインタがわかりやすい言語ってある?
アセンブラへの展開見れば一目瞭然。
文法はそう記述したら、そうなるわけだから、詰まる要素ではないだろう ポインタを知る為にメモリの状態とアドレスをPrintfとか使ってチマチマ やってた時、頭が沸騰しそうだったな 覚える為に、ここら辺の関係ない所で時間がかかり過ぎるのが問題じゃないかな
俺はそんなところで時間かから無かったよ
文法を理解しているというのなら、 ポインタの中身なんて理解する必要なっしんぐ。 C言語のプロトコルに従って書いてりゃどこでも動くコードのいっちょ上がり。 と言いたいところだが、そうもうまくいかないと思う。
TOEIC満点のやつは英語で嘘をつかない、なんて保証があると思う?
誰か翻訳頼む
指を見るな 月を見ろ
文字列処理や勉強の時のscanf()で何度も不正な処理起こしたC言語初心者時代 普通の高級言語なら実行時エラー、エラーメッセージで分かる内容のところを いきなりOSからエラー返されたらビビるって
いや エラーでりゃましなほうさ
ポインタ挫折歴3年の俺が来ましたよっと char *s = "test"; なんでこういうことが出来るのかいまだに理解できない char s[] = "test"; とは違うの?
どのレベルで分からんのかが分からんなぁ。 ちょっと嘘を交えて言うと "test"は、プログラム開始時にOSが用意してくれる5バイトの 書き換え不可な配列として解釈される。 char *s = "test"; は、アドレスを指す4バイト変数sを作って OSが用意した"test"の先頭のアドレスを代入する。 char s[] = "test";は、OSが用意した"test"と同じサイズの(5バイト) 配列を作って、"test"の内容をコピーする。 違いは大体こんなもん。 どこが嘘かは、興味があるならまた別途。 何故そう書けるのか、と言われると、Cが文法として、 そういう機能を持ってるから、としかよう言わんなぁ。
int main(int ac, char **av) { char *a = "test"; char *b = "test"; char c[] = "test"; char d[] = "test"; printf("%d", a == b); printf("%d", c == d); printf("%d", a == c); return 0; }
俺の彼女をボインだ!って言われるようにするにはどうすればいいですか?
>>337 前者
_s: dc.l __lbl
__lbl: dc.b 'test', 0
後者
_s: dc.b 'test', 0
引っかかりそうな所には近づかない・逃げるという方法もあるな。 scanfとか初期化式とかは、ポインタと少しは関係あるが、 関係ない部分で引っかかってる感じもする。 あまり関係ない所に全力でぶつかったり、教える側も全力で教えようとしていたら それは学習コストは高くつくに決まっている。 自ら望んで全力で引っかかるのは構わないが、 「なぜ引っかかることを望んだのか」を他人に問い詰めるのは迷惑だ。
char *s="test"; sprintf(s,"Hello world"); 正常に動くようですがこのまま使って大丈夫でしょうか?
はいはい、ネタは自分の頭の中でやってくれ それが嫌なら窓を捨ててペンギン村にようこそ
>>345 だめです。
というより
char *s = "test";
という使い方はほとんどしないから忘れてもいいかと。
const char *s = "fuck me please!!" ならいいよ
そこはfackだろ
>>345 char *s="test";
*s="Hello world";
でいいじゃん
s="Hello world";
配列は関数に渡すとポインタに変換される ポインタはprintfされるとアドレスに変換される 暗黙に二度の変換を受けるという理不尽な仕様
おいおい。 全て間違ってる。
#include <stdio.h> int main(int ac, char **av) { char *a = "test"; char *b = "test"; char c[] = "test"; char d[] = "test"; printf("%d\n", a == b); printf("%d\n", c == d); printf("%d\n", a == c); c[0] = 'C'; printf("%s\n", c); try{ a[0] = 'A'; printf("%s\n", a); }catch(...){ printf("catch\n"); } return 0; } どうしたら Segmentation fault を出さないように出来ますか?
cでtrycatchとかありえない
ポインタは変数のある場所を指している(キリッ)って説明がわかりにくい 「指してる」ってなんだ?と思ってた はじめにscanfや関数の引数としての使い方から説明するから単体での理解がしにくい アドレスを格納する変数ならそう言って欲しい。分かりやすくしようとしてわかりにくくなってるよたぶん あと変数宣言の説明を軽くスルーしたり、ポインタをポインタ型とかint*型とか言わないのも問題な気がする ポインタだけ特別枠の別物みたいにみえる プログラミング習いたての人との共通認識を探ろうとしないのも分かりにくさに拍車をかけてる 難しいんだよー、だから分からなくても仕方ないんだよー、俺のせいじゃないよーってか 分からせる気ないだろうと
英語で言えば ある場所を指している=ポインタ だからな ポインタは変数のポインタ(キリッ) じゃ判らん罠
再帰的定義だよ
360 :
359 :2010/07/11(日) 17:00:11
>>359 変数って書いてあるけど、値のことだよね。
ポインタとは、オブジェクト(≒変数)の位置を表現する値を指す。
C/C++ではメモリアドレスを使用するのが一般的。
ポインタに対して演算を行うことにより、オブジェクトを取得したり、
またオブジェクトに隣接する位置のオブジェクトのポインタを
取得することができる。
ああ、こう書いちゃうと分からないんだよなあ。
やっぱ箱のポンチ絵無しには説明は無理だわ。
で、更に変数宣言がどういうものなのか書こうとすると
rationaleとか持ってこないと荷が重い。
だからアセンブラに落とせば一発だっつーのに
アセンブラにCみたいな複合型ねーだろ。
ワロタw 結構いるよね、ほのめかしはするけど実は答え知らない自称上級者
>>364 値の出し入れ以外に
まともな文明の利器があったなんて知りませんでした。
もちろん人が解釈するものじゃないよね?
きちんと文法なんだよね?
ポインタはクイックソートをする為だけにある。 それ以外の用途では不要。
>>360 ポインタと言った場合、ポインタ型のことか、
ポインタ型の変数のこと。
値はアドレス
クラスメンバへのポインタで質問です class Inner{ pubilc: void hogehoge(); int fugafuga; }; class Base{ public: Inner *in; void hoge(); int fuga; }; Base b;としたとき b.hoge は &Base::hoge b.fuga は &Base::fuga b.in は &Base::in でいけると思うのですが b.in->hogehoge b.in->fugafuga 相当のアドレスを得るにはどうしたらいいのでしょうか?
C++ちゃんと知らんけど 「いけると思うのですが」の時点で いけてないことないか?
普通に&Inner::hogehogeじゃダメなの?
でいけると思うのですが
373 :
デフォルトの名無しさん :2010/07/17(土) 08:34:26
ぬるぽ
ポインタはすんなり理解できたけど「逆参照」だけはずっと意味不明だった。 なんだよこの訳語は。
デリファレンスでいいだろ
俺なら訪問と訳して全国のポインタ脱落組を3000人くらい救っていたな。
入門書の間違いなんて、たまたまそれに嵌って痛い目にあったりでもしない限り、 いちいち覚えちゃねえよ。 複数言語やってりゃ、記述が間違っていても何となく回避してるし。 子どもの時に乗った三輪車の善し悪しなんぞ誰もわからんのと同じ。 ぐだぐだ言うのは、三輪車をとっかえひっかえしている池沼と、書いた著者くらいのもん。
それはそうかも
>>377 複数言語やってても、そこにポインタ扱う言語が無かったらやっぱ引っかかるよ
381 :
デフォルトの名無しさん :2010/07/17(土) 21:35:13
>>373 Javaじゃないからガッしてくれないよ
スマポで引っかかりました。 詳細は以下から。 class Dog : public IAnimal というクラスがあって、これをスマポでインスタンスを生成すると void fanc(IAnimal animal) に入れられなくなります。 どう書けばいいんですか?
>>380 間接参照の概念さえあれば大体何がやりたいのかは分かる
>>382 そのスマートポインタをデリファレンスすれば委員で内科医?
言語が判らんから具体的には答えられないが。
C++(VC10)です ↓こういうふうに書いたら駄目なんですか? 何で入らないのかわかりません std::shared_ptr<IAnimal> dog_ptr(new Dog());
>>386 ポインタを実態を必要とする関数に渡したいなら、>384の言うように逆参照すればいい。
もしかして、単項*演算子も知らないのか?
>>382 auto_ptr<Dog> p(new Dog());
func(*p);
うーん何言ってるかよくわからないのはポインタを避けて通ってるからなのですかね とりあえず.get()って付けたら入りました func(dog_ptr.get()); 全く意味わからないですが急場しのぎって感じで満足してます やっぱもってるなってことで
その程度で満足できるんなら勝ちでいいよ
たとえばK&Rは第5章がポインタで第6章が構造体だが ポインタは避けるが構造体 (クラス) は避けないって感覚は異常だよ ふつうに考えるとポインタよりもクラスのほうが難しいはずだよ なぜか知らないがクラスが難しいと言うのは恥ずかしい ポインタが難しいと言うのは恥ずかしくないという空気に支配されている気がする
知らんがな
何かあるとK&Rという空気に支配されている気がする
>>392 ポインタのほうが構造体よりずっと難しい
ポインタは存在しないオブジェクトを指す可能性があるし、表記が複雑になる
構造体のほうが難しいと勘違いするのはポインタを理解していないからだ
逆をいえば、存在しないオブジェクトを指さなくて構文が複雑にならない ようなポインタなら問題ないんだよな JavaやPythonの間接参照に引っかかるようなヤツはそうそうおるまい
398 :
289 :2010/07/18(日) 14:02:41
ポインタ難し・・・・^^;
ポインタや構造体のどこが難しいのか理解できない。 Win32APIのほうがよっぽどマンドクセ。
%pに細工して printf("%p", p)の出力を 〒100-0014 東京都千代田区永田町1丁目7−1 とかにすればイメージ沸きやすいと思う。 アドレスと値を同列に扱えてしまうのがコンピュータの強みでもあるが 混乱の基なんだよ。
哲学的な話になりそうだなw
もう黙ってアセンブラから入るしかないのでは? というか、我々の頃は、ちょっと言語がわかるようになったら、自然とそうなっていたような気がするのですがね。
ポインタは間接参照を表現するものとしては 無駄の多い構文になってるのがアレなだけで こういうアプローチ以外は不可能ってほど難しいものじゃないだろ アセンブリなんてしなくてもよろしい
そんなものやって役に立つの?
アセンブリは知らなくても良いけど、メモリ上でプログラムが どうやって動いてるかの概要ぐらいは分かってないと、 ポインタまわりのデバッグはやりづらいね。
>>403 無駄が多いと感じる原因は「型」。
メモリ上には型がない。なぜなら必要ないから。
型がなくてもプログラムが動くことが分かれば、
ポインタが (構文の見た目ほどには) 難しくないことが分かる。
>>406 そのような「型の必要がない」という事実を体験するには、アセンブラをやるのが手っ取りばやい、と。
byte, word, dword, qword ...
masm は構造体も使えたな
メモリアドレスを指す、参照でしょ?
ポインタで引っかかるのはC言語のせいだと思う Cって、ポインタ依存度高過ぎる上に そのポインタの扱いがフリーダムだからなぁ… ポインタ依存度低めで、扱いがそこそこ厳しめなDelphiだと ポインタで引っかかった経験はあまり無いぞ
Delphi って ObjectPASCAL だろ
TurboPascal for Windows
>>412 えーと、つまりポインタ自体の概念は難しくない。
そこに掛かる、文法や、派生概念が捕らえ方を
面倒にしてる。
ってこと?
>>415 そんな感じかな?
例えば、C言語で
「純粋な参照渡し」が使えたり
ポインタに厳格な型があって明示的なキャストをしたとき以外では互換性が無かったり
文字列型をコア機能として持ってたり
動的配列をポインタを使わず使えたり
ポインタを利用するような標準ライブラリがそもそもほとんど無かったら?
本当にメモリを扱いたい時にしか使わない概念になってたら
ポインタで引っかかる人は相当に少なかったと思う
配列か… 配列の制限の多さにビビッタ覚えがあるなぁ。 配列を返すためにも、ポインタを使うんだよ。 と、師匠に言われて、俺の価値観が崩壊した。
>>415 typedef使えばすっきりするからやってごらん
もともとアセンブラで書いてた人用の物だから仕方ないね
たしかにポインタの概念自体は馬鹿でも理解できるものだと思うな
>>416 C#のアンセーフポインタみたいな感じ?
俺はそのアンセーフポインタとやらを詳しくは知らんが 要は本当にメモリアドレスを扱う以外には使う必要がないってことか? だとしたらまあそこそこ合ってるかな ポインタ自体も出来る限り安全であるべきだとは思うが
今となっては、安全でないところが、Cのポインタの利点だからなぁ・・・
Cはポインタのせいで最適化がしづらくなっている。 バグもでやすいしね。 一番いいのはCを使わないことだ。
低レイヤー向けの使いやすい言語を作ろう。 って話はないのかな。
LLVM。
>>424 ほとんどの言語はヒープのせいで最適化しづらい。
バグがいやなら、一番いいのは最適化を諦めることだ。
>>427 Cの領域にGoが侵食し始めてるな。
あと5年もすればC?なにそれ?Cobolとかいうやつ?みたいな扱いになるよ。
カーネルの実装やらC++がいる時点でそれはないわwww 業務開発ではもうC使わないけどな
ポインタのポインタのポインタで初めのポインタに繋がったらどうなるの?
普通は型エラーになるよ。
元彼の元彼の元彼で初めのエイズに繋がったらどうなるの?
アッー!
ご冥福をお祈りします
int hoge(int a[10]) { int b[10]; return *a; } aはポインタなのにbは配列 ポインタと配列は似ているとかいう解説にかなり混乱したもんだ スタック・ヒープ・文字列あたりがメモリのどこに確保されるかも 一緒に覚えないとポインタはいつまでたっても和歌ランと思う
この10とかも飾りだしなぁ
仮引数に[数字]使ってもいいけど、 それが許されるのは配列へのポインタとかだな。 要するに可変長配列とか多次元配列の要素とか。
しなくていい(実際の業務では必要ない)表現のフェイクが多すぎて混乱してるだけじゃね? 配列は領域が単に連続してるだけで先頭だけみたらポインタ 違うとかそういうもんじゃなくて比べられないだろ ポインタの指す先が問題なんであって・・・ mallocでとった領域だって連続してるし char *p = (char*)malloc(sizeof(char)*100); char p[100]; とかどう違う?とか聞いたらまた混乱してしまうんだろか?w こんなもんの違いなんてそう意識することないと思うけどねw
Cのよいところはポインタを無駄にちりばめすぎてて 理解してないとろくなコードをかけないところだな。 ASSERT(ポインタを理解している);と先頭に書いてあるようなもん。 中には五里霧中なままコード書いて金もらってる猛者もいるけどさw
sizeof(p) p=(char*)0
俺から見たら
>>444 の回答を知った上で
>>441 の言ってることはもっともだと
思うけど、人によって違うのか
ここら辺もポインタと配列の違いという話をするときに誤解しやすい要因かもね
>>441 は何が言いたいのかよく分からない
配列もポインタも大差ないと言いたいのか?
ぶっちゃけこの違いを意識できないヤツがメモリリーク起こすんだから、「そう意識することはない」というのは乱暴だと思うw
>>441 は多次元配列を考察すれば納得できると思う。
>>444 sizeof(p)は微妙だな
ソースまたぐと正常な値返ってこないからなw
本質的には(アセンブラ的に見れば) どちらも同じ。
>>451 わかる人からすればもっともな意見だとも思う
ただ、言語仕様的には明確に違うし、教える側がその辺はしょって初心者に説明するのが混乱の元だと思う
よく分からない例え、不正確な同一視は問題の先送りにしかならないんだよなぁ
いつかまとめて乗り越えなきゃいけない日が来る
int main
つまづく
配列のコピーとポインタのコピーは明確に違う その点はjavaだろうとpythonだろうと引っかかる人は引っかかる
ポインタ □→□□□□
でっていう
下記は完全に同じ意味ですか? p->a; (*p).a; 同じだとしたら、なぜわざわざ -> という演算子を用意したのでしょうか? 由来とかあったら教えてください。
Cの目的の一つにタイプ量を減らすことってのがあったな。
(void)p->a->b->c; (void)(*(*(*p).a).b).c; いちいち先頭に戻って挿入するなんて考えられない (void)p[0].a[0].b[0].c; これもなんか嫌だよ
461 :
デフォルトの名無しさん :2010/08/02(月) 22:45:29
どんだけネストしてんだよw
超巨大構造体ですか?
(void)foo->super(foo)->hoge(foo->super(foo)); (void)(*(*(*foo).super)(foo)).hoge)((*(*foo).super)(foo)); 一時変数使えよって感じだな DirectXとかlibjpegとかCore Foundationでは構造体と関数ポインタでこんなことやっていたと思う オブジェクト指向っぽく書けるから良く使う手法だけどね
>>460 > (void)(*(*(*p).a).b).c;
配列のポインタとか、関数のポインタとかが複数からむと、そんな記述に現実なるじゃん。
何故構造体だけが -> なんてものを用意されて優遇されているのか。
配列[]も関数()も構造体->も後置 *だけが前置
優遇前 (*p).a *(p+i) (*p)() 優遇後 p->a p[i] p()
関数ポインタpfに対して(*pf)();とpf();が同じになるのだから、 ドット演算子もこの仕様に変えたら p.a.b.c; って書けるじゃん。
ポインタの問題 初級 int a[10]; a[0]=10; a[1}=20; a[2]=a[0]+a[1]; このとき、a[2]はいくつになるか答えなさい
それは配列
30?
それよりコンパイルが通るのかそれ
ポインタ関係ないだろ
int *x
汚い尻穴だなあ
475 :
デフォルトの名無しさん :2010/08/07(土) 21:35:18
int* x
476 :
デフォルトの名無しさん :2010/08/08(日) 23:22:06
C++をやるようになってからは、以前よりポインタを使わなくなったな。
ポインタでひっかかるってよく言うけど、具体的にポインタの何でひっかかるの? 初期化せずに使うミスをするとかそういうこと?
478 :
デフォルトの名無しさん :2010/08/09(月) 12:32:32
>>477 こうゆう書き方とかしちゃうんじゃね?
*ptr++ ptr+sizeof(*ptr)
正直なんでポインタで引っかかる人がいるのか理解できない
時間をかければ理解できるが無駄な時間がかかりすぎる気がする もっと近道は無いのか 無いなら無いことを証明するべきじゃないか と愚痴っているうちに時間が過ぎて行きます
x=20; p=&x; *p=30; これで、pしか変更してないように見えるはずなのにxが30になる これがポインタのすべてだろ
こういうのだろ。 struct node { int value; struct node *next; } struct node *list = ???; struct node *n; // 略 struct node **tmpnext = &n->next; struct node *orgnode = n->next - sizeof (int);
ミスった。 struct node *orgnode = tmpnext - sizeof (int);
>>477 本来ポインタとは、ポインタが何かを理解してる人間だけが使えばいいところを
C言語は初心者ですらポインタと無関係ではいられないからだろう。
ポインタ周りの日本語訳が酷いってのが関係してると思うな どうしてそんな言葉を当てちゃったのみたいな まあ俺自身まだポインタのとこまで勉強してないからよくわからんが
昔内輪ではやったネタなんだけど
車輪だな
しかも劣化した汚ねぇ車輪だな
車輪(スポーク)はケツの皺だな
∴ 汚ねえケツだなぁ
ってことで
>>482 のような小汚いソースを見るとTDNを思い出す
2重ポインタあたりは引っかかると思う。 char** t = (char**)malloc(10*sizeof(char*)); int i; for( i=0; i<10; i++){ t[i] = (char*)malloc(100*sizeof(char)); } こんなふうにメモリ取ってこなきゃならんのをけっこう忘れると思う。 あとは int test(int a[5][10]) のような引数の時とか微妙に型チェックがめんどくさかった気がする。
//引っかかる言語 int[] a; a[0] = 0; #引っかからない言語 my @a; $a[0] = 0;
大きいオブジェクトを生身で渡すとか ローカルで定義された参照を返すとか
ひたすらもし〜でなければを繰り返して切り分けるより 宣言多くなるかもだけど実質中身同じだしってことか
>>488 それさ、単に宣言してない変数を
自動的に作るってだけじゃね?
2重ポインタw
しかたなしに、という表現の意図がわからん。 ポインタはなるべく使うべきでないというドクトリンでもあるのか? ポインタを使えばできることはポインタを使えばいいだけのことだろ。
>>495 C++ なら参照渡しにするでしょうね。そのほうがわかりやすい。
int func(int &*p)
constでない参照の乱用がわかりやすい?どこの初心者だよ。
>>497 それは参照を理解していないだけでしょう?
こんな場所なら初心者でも熟練者でも同じように発言できるからなあ。 どっちが正しいか客観的に判断する方法もないし。 「あんたがわかってないだけでしょ」って言ったもん勝ちか。
ポインタの一次元配列を、二重ポインタって呼ぶんだ へー
まじで? 初めて聞いた。 int **a;これが二重ポインタだと思ってた。
なんか、会社ってところはどこもそうなのかも知れないけどさ 良い人から辞めちゃうよね んでもって、腹黒い人とか、他人を動とも思わない人とか、 図々しい人とか、・・・そんなのばかりがテコでも動かないぜって 感じで居座り続ける・・・ 会社って、こうやってダメになっていくんだと思う それはそのまんま居座り族・牛耳り族が社会をダメにしていくって ことなんだ。。。
二 重 ポ イ ン タ な ん て 言 葉 は な い
505 :
499 :2010/08/29(日) 17:52:20
>>504 その程度の問題だったらポインタのポインタも、ポインタの参照も使わずに出来るよ。
ポインタだけで出来るから頑張って作ってみ。
507 :
499 :2010/08/29(日) 18:08:12
じゃあ最初からそれ出せばいいじゃん・・・ 何で無駄なポインタのポインタやら、ポインタの参照やら使ったプログラム書くんだか。
>>507 >>506 では、2分木に対するデータの追加を扱っていますが、
ポインタのポインタをつかわなければ、
struct node *root;
root = add_node(root, data);
という形になり、これでは再帰が深くなると無駄な代入が繰り返されてしまいます。
本当に代入が必要なのはデータの追加が行われたところのノードですから、そのようにコードをかいて、
sturcut node *root;
add_node(&root, data)
という形にするのが本当だと思います。
参照を使うと、
struct node *root;
add_node(root, data)
とかけますし、add_node() 自体の記述でも、* や & を省くことができます。
具体的なコードの記述例は
>>504 のリンク先を参照してください。
509 :
499 :2010/08/29(日) 18:27:00
>>508 >ポインタのポインタをつかわなければ、
>struct node *root;
>root = add_node(root, data);
>という形になり、これでは再帰が深くなると無駄な代入が繰り返されてしまいます。
それはロジックが悪い。
そんなに代入を繰り返さなくても出来る。考えてみ。
>参照を使うと、(略)* や & を省くことができます。
参照でコードが短くなるのは当然だが、同時に、「値が変化している」という情報も失われる。
以下のような例がありえる。
struct HOGE* hoge;
func1(hoge);
func2(hoge); //ここで内容をいじっている!
func3(hoge);
これは以下のようにしたほうが「中で何かしている」感じがわかりやすくなる。
func1(hoge);
func2(&hoge);
func3(hoge);
値を変える参照は、短くなるからと無闇に使うと危険ですよ。
>>509 >それはロジックが悪い。
>そんなに代入を繰り返さなくても出来る。考えてみ。
いや、二分木や線形リストに関しては、ポインタのポインタを使わない限り、無駄な代入が繰り返されてしまいます。
ポインタのポインタを使わずにスマートな方法があるというのなら、それは是非知りたいところです。実例を示してください。
>値を変える参照は、短くなるからと無闇に使うと危険ですよ
それはそうですね。後から読むときにわかりにくいのは事実です。
511 :
499 :2010/08/29(日) 18:34:27
「参照を理解してない人」に教えてもらってどうすんだ?あ? 無礼な言葉を吐くだけ吐いて、出来ないとなったら教えてください、かよ。おめでたいガキだな。
>>511 コードを示してください。
私はすでにポインタのポインタを使ったもの、参照をつかったもの、ポインタのポインタを使っていないものの3種を示しています。
時間はたっぷりありますから、アイディアだけでも示してください。
話はそれから。
513 :
499 :2010/08/29(日) 18:43:48
バカバカしい。 無礼者に使う時間などないよ。 おまえよりもうちょっとマシな誰かが、気が向いたらコード書いてくれるかもしれないから、それまで待ってれば?
>>513 よく検討もせずに
>それはロジックが悪い。
というほうがよっぽど無礼ですね。こちらは動くコードを書いた上で意見を述べているのですけれども。
コードを示してください。
話はそれから。
515 :
499 :2010/08/29(日) 18:50:23
そこのロジックの悪さはきちんと検討して発言してるよ。 代替コードも書いている。 おまえにタダで見せる気にならないだけだ。 誰か奇特な人が見せてくれるまで口あけて待っとけ。
>>515 口だけなら「コードを書いている」といくらでもいえます。
それに「見せるきにならない」といいますが、
仮に見せていただければ、私はただただ低頭して「私が馬鹿でした」と発言して消えるでしょう。
それはあなたにとっては、大満足なのではないですか
動くコードを見せて実証することで、私の今までの発言を全否定できるにもかかわらず、それをせずにうそぶくのみ。
これでは、実証するアイディアなりコードなりがないと推測せざるをえません。推測ですけれどもね。
コードを示してください。
話はそれから。
518 :
499 :2010/08/29(日) 20:38:25
>>517 おまえが馬鹿(というよりは、低レベル)なことは、俺にとってははっきりしてるからな。
別にこんな便所のラクガキの場所で示す必要も無い。
逆に、示してやればおまえのレベルを引き上げてしまう。バカバカしい。
ど素人はど素人のまま、学生の宿題でもやってるのがお似合いだよ。
だいたい、教えて欲しいなら、最初の暴言を謝るのが順序だろう。それもせずにただ出せなんて図々しい。
ポインタのポインタなんて、Win32のCOMみたいな使い方をする程度かな ふつーに struct data *alloc_root(void); int add_data(struct data *root, char *name, char *tel); とか作る
>>518 ひとつ可能性があるとすれば、非再帰的に記述することですかね。
線形リストならば、それは可能ですが、2分木だと結構むずかしくなります。たぶん各ノードにワークエリアを設けて
戻るノードを記憶しなければならないでしょうね。
昔書いた非再帰的なコードをあげておきます。ポインタのポインタは使っていますが。
http://codepad.org/n3raxalm 私は再帰的なアルゴリズムのみしか念頭にありませんでした。
もし非再帰的に記述することを示唆しておれば、大変失礼いたしまた。
binary treeで戻る?
>>521 ええ、binary tree でノードの追加なり総なめを行うのなら、親ノードの位置を逐一記録しておかなければならないでしょうね。
ノードを追加したら、追加されて新しく親ノードになったところをなんらかの形で覚えておかないといけないでしょう。
もしかすると、ノードごとにワークを持つ必要はないかもしれませんが。(
>>520 はノードごとにワークをもっていました。)
親ノードとかいらねーよアホ バイナリツリーでそんな実装するわけねーだろ
char *name[]ってどういう意味なんでしょうか?
char **name;
えっ
アセンブラでメモリの番地は理解していても
C言語での書き方で悩む事もあるんだよ
>>55 みたいな
constの位置とかもな
面倒なので括弧を付ける
>>530 const は一番近い場所にある token が変更不可能になる
っていう理解で桶?
int const * 近いってどっちだよ、ってなるぞ
暗記したほうがマシ、というか十分。 理屈を抜きにするから応用利かないけど。 この二つ暗記な。 int *p[] ポインタの配列 int *const p コンストポインタ だからこそ、 配列のポインタが (*p)[] だと想像できるし、 コンストintへのポインタが int const *だと分かる。
そんなにむずかしくないだろ constが後置修飾になってるほうがポインタそのものをconstにしてるだけ int const もconst int も int を定数にしてるだけなんだし
いや いまは int const * p; と int * const p; の違いの話をしているんじゃまいか
イメージ的には、 ポインタの配列は(*p)[] これだし、 constポインタはconst (int *)pこう書きたい。 イメージどおりには行かんね。
>ポインタの配列は(*p)[] これだし、 hds
const int *p
const int *p; int const *p; int * const p;
de?
掛け算以外のアスタリスクがあまり出てこないプログラムを心がけてますよ
>>542 それは C の最強の武器を放棄したに等しい。
const int *p => const int の ポインタ int const *p => const int の ポインタ int * const p => int の const ポインタ
1行目と2行目の違いが判らんのだが
違いはない
えっ
int main() { int i = 100, j = 200; const int *a, *b; // const int int const *c, *d; // const int int *const e = &i, *const f = &i; // intへのconstポインタ int const *const g = &i, *const h = &i; // const intへのconstポインタ *a = j; // error: assignment of read-only location *b = j; // error: assignment of read-only location *c = j; // error: assignment of read-only location *d = j; // error: assignment of read-only location e = &j; // error: assignment of read-only variable `e' f = &j; // error: assignment of read-only variable `f' *g = j; // error: assignment of read-only location *h = j; // error: assignment of read-only location g = &j; // error: assignment of read-only variable `g' h = &j; // error: assignment of read-only variable `h' return 0; }
ちょい修正: const int *a, *b; // const intへのポインタ int const *c, *d; // const intへのポインタ
えっえっ
* の左側に const があるか右側にあるかだけですべてが決まるんだよ * が複数ある場合も何番目の * と * の間に const があるかで意味が決まる
難しいのー
far * near * とかに比べると、ああらくちんだ〜。
>>522 巡回するなら普通はスタック使うだろ?
ってマジレスしたらいかんかったのだろうか
>>523 仕様によってはあり得る
556 :
522 :2010/09/15(水) 00:28:58
>>555 この問題、宿題スレの方で実際にコードを書いていただき、戻る位置をおぼえない方法を示していただきました。
すぱっと非再帰に書き下すきれいなコードでした。
>>524 でも、二分木は再帰で書きたいもの、とちょっと負け惜しみ。
スタティックな変数を使えば再帰でも書ける 最適化を考えると大概再帰よりループ展開の方が速い どっちでも書けるようになったほうがいいぞ
558 :
522 :2010/09/15(水) 00:50:29
だから言ったじゃねえかよ。 あと、再帰でも、無駄なコピーなしに実装できる。 絶対できないとか頑張ってないで、ちょっと考えてみ。
学生だと思うけど若い割に頭硬そうだよな アルゴリズム関係は向いてないんじゃないかね
>>559 二重ポインタか参照を使わないと無理ではないでしょうか?
出来るって言ってんのにわからんガキだな〜 人を疑う前に少しは考えろよ
嘘ってこともないだろ。判り難い説明ではあるが。
面倒だから型の左側以外に const 使うなよ。 const int* a とかにしとけ。 int const *aとか使わんだろ。
int * const a
const int const *a;
>>567 int const * const a;
or
const int * const a;
えっ
>568 修飾子が被っているが無視されるので問題ない まぁ、書いた人はそんなこと意図していなかっただろうけど
>>571 どっちが正しいっていうか一般的っていうか
どっちも正しい
const int const * const a;
だから被ってるって、まぁいいけどね
ひっかけ問題みたいだから。
579 :
デフォルトの名無しさん :2011/02/06(日) 10:06:34
>>579 ○ const int
○ int const
× const int const
○ const int * const
○ int const * const
× const int const * const
○ const int *
○ inst const *
× const int const *
○ int * const
○ int *
>>580 > ○ const int *
> × inst const *
○ int const *
inst になってしまそうになる気持はわかる
int const C::* p; // Cはクラス型 C* c; c->*p;
我々のポインタ。いやらしい・・・
ぬるぽ
int const * const ** const *p; コンストintへのポインタへのコンストポインタへのポインタへのコンストポインタでOK?
間違ってるぞ
指定子に対する const 以外はすべて後方から修飾。 (指定子に限りどちらからでも修飾可能) マンバポインタといっしょで *const で一塊と見たほうがわかりやすいね。 int *p; int ( *func )(); int *const q = p; int ( *const funk )() = func; class C; int C::* r; int ( C::* funx )();
う〜、マンバ!!
char *abc[][3] = { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"fgh", "cde", "@ab"}, {"aho", "baka", "shine"} }; で初期化出来ますが、 *abc[][3] の配列(というか *abc[][3] を指すポインタ)は、 char **xyz[][3] = {abc, def, ...}; じゃ代入出来ないんですけど、 どう書いたら良いでしょうか?
*[][3]のポインタは、*(*[][3])ね *abc[][3]に対して、abcは*(*)[3]ね &abcは、*(*)[][3]ね char *(*xyz)[3] = abc; {abc, def, ...}は、配列とかにしたいんだろうから char *(*xyz[])[3] = {abc, def, ...}; 配列にしたいので、xyz[]になる
ありがとうございます char *(*xyz[])[3] = {abc, def, ...}; ならいけますが char *(*xyz[])[3] = { { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"fgh", "cde", "@ab"}, {"aho", "baka", "shine"} }, { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"fgh", "cde", "@ab"}, {"aho", "baka", "shine"} }, { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"fgh", "cde", "@ab"}, {"aho", "baka", "shine"} }, ... }; は無理ですよね?
char *xyz[][4][3] = {略}; {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"fgh", "cde", "@ab"}, {"aho", "baka", "shine"} } が[4][3] ""が* その配列なので、[] char *(xyz[])[4][3]の方がわかりやすいかな
あ、説明不足でした。すみません。 char *abc[][3] = { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"fgh", "cde", "@ab"}, {"aho", "baka", "shine"} ... ... ... }; なら縦方向に可変長に出来るのに、 char *(*xyz[])[3] = { { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"fgh", "cde", "@ab"}, {"aho", "baka", "shine"} }, { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"} }, { {"xyz", "uvw", "rst"}, {"opq", "lmn", "ijk"}, {"aho", "baka", "shine"} }, ... ... }; は無理ですよね? ってことです。
594 :
デフォルトの名無しさん :2011/03/05(土) 06:14:40.27
馬鹿には無理
595 :
デフォルトの名無しさん :2011/04/11(月) 21:47:20.95
馬鹿ばっかだな。 **、が何示してるか分からないだろ。
>>593 二次元配列へのポインタを[]を使わないで書けるようにまずはなれ
>>71 > ">私のささやかな経験から言わせてもらうと、伝統的に大学のコンピュータ
> サイエンスのカリキュラムで教えられているもので、多くの人がうまく理解
> できないものが2つあった: ポインタと再帰だ。
> "
なんか再帰が苦手だったというか、最初ハノイの塔とかでも自分で自分を呼ぶというのが
どうもよくわからなくて苦労したが、同じ名前の関数を呼ぶと考えたらあっさり理解できたな
初心者の頃。
でもトレースするのはやっぱり苦手。
>>597 自分で自分を呼ぶと考えるからややこしいんだよ。
例えば階乗なんてこれだけだ。
自然数nの階乗とは:
nが0か1の場合、1
そうでない場合、n-1の階乗にnをかけた結果
これをCでそのまま書くだけ。
unsigned factrial(unsigned n)
{
if (n == 0 || n == 1) return 1;
return factrial(n - 1) * n;
}
ハノイの塔だろうがフィボナッチ数列だろうがツリー探索だろうがみんなこれの応用だ。
なんなら解説してもいいけど、ここではスレ違いだな。
すみません
末尾再帰は単純なループに置き換え可能だそうですが
>>598 はどうなりますか?
>>599 考え方をちょっと変える必要がある。
>自然数nの階乗とは:
>nが0か1の場合、1
>そうでない場合、n-1の階乗にnをかけた結果
ここで、n-1の階乗を返す関数の代わりにn-1の階乗にnをかけた結果を返す関数を用意する。
unsigned fact_sub(unsigned n, unsigned r)
{
if (n == 0 || n == 1) return r; // 0! * r or 1! * r
return fact_sub(n - 1, n * r); // (n-1)! * r
}
これを呼ぶように>598を書き換える。
unsigned factrial(unsigned n)
{
return fact_sub(n, 1);
}
C++なら、unsigned factrial(unsigned n, unsigned r = 1)としてしまえば一関数で済む。
尤も、こんなことするまでもなく最近のgccは>598をループに書き換えるけどね。
再帰は自分自身を呼び出したとき、関数の最初に戻るのではなく 呼び出したところに関数と同じ内容のブロックがあり、 それが延々と続いてるとイメージすれば分かりやすい それはまさにツリーと同じ構造だ
なるほど、そういう馬鹿げた置き換えイメージを持っているから、実際に 再帰で書けと言っても書けない奴が多いのか、再帰のロジックを読める くせに書けない奴が多い理由がわかった。
そういう書けない人には 「まずuntilループに書き換える」 「ループ内で再代入する変数は全て引数として渡す」 って2点を教えてやれ
またいつもの再帰番長か。
アセンブラやらないからだよ
末尾最適化が出来る再帰と出来ない再帰があるな 例えばアッカーマン関数は末尾最適化出来ない
ttp://kmaebashi.com/programmer/pointer.html >K&Rによれば、Cの宣言は、「変数が現われ得る式の構文を真似た(P.114)」そうである。
>しかし、本質的に全く異なるものを無理に似せようとしたため、
>結局わけのわからない構文になってしまっている。
># Cの作者Dennis Ritchieが最近開発した新しい言語 Limboは、一見
># C like だが、宣言の構文はしっかりPascal風のものに直してある...
># ずるい(^^;
変数宣言の*と、デリファレンスの*で混乱したのを思い出すわ。 同じ*だと思っていたからどうにも理解できなかったわ。
もしもポインタの*が後置きだったら…… 前置版 → 後置版 int *p; → int p*; int (*p)[99]; → int p*[99]; int *p[99]; → int p[99]*; int *(**p[5])[3][2]; → int p[5]**[3][2]*; int (*pf)(void) → int pf*(void); int (*(*(*(*pf)[5])(void))(void))[3]; → int pf*[5]*(void)*(void)*[3]; 結果: 後置版はカッコを付けなくても一意的に意味がつくので、 宣言がすっきりする 後置版の方が本来は読みやすいはずだが、今のところ、 個人的にはかえって読みにくい(「*p」表記に毒されてしまったのか^p^)
宣言の読み易さはともかく、それは実行コーディングの使い易さ・読み易さと 両立できるのか?
掛け算との使い分けは?
>>610 使い易さ・読み易さが格段に落ちることは無いと思う
ポインタpに対して、(*p)という記述を自発的に封印して
p[0]という記述に統一した場面を想像すればいい
この場合、実行コーディングの使い易さ・読み易さが
格段に落ちることは無いよね
(*seito).kokugo = 100; → seito[0].kokugo = 100;
*(*hoge).(*fuga) = NULL; → hoge[0].fuga[0][0] = NULL;
後置版の場合、(*p) が p* に置き換わるだけだから、これは
*p が p[0] に置き換わったのと同じことで、やっぱり
使い易さ・読み易さが格段に落ちることは無いのでは?
(*seito).kokugo = 100; → seito*.kokugo = 100;
*(*hoge).(*fuga) = NULL; → hoge*.fuga** = NULL;
>>611 p* = 10; と p *= 10; の区別が紛らわしいって話だよね
それは別の記号を使えばいいだけでは…
ポインタの「*」には、*のかわりに@を使うことにする、とか
p@ = 10; p *= 10;
玉突き衝突的にあちこちいじりまくるなら そのままでよさそうだな
>>614 弄りまくるも何も、文法的には何も弄ってないのだが…
仮に弄っているように見えたとしても、
>>612-613 で全て完結してる
(*p) のかわりに p[0] を使う文法は、今のままのC言語で普通に使える。
後置版を考えるというのは、「全ての(*p) を p[0] に統一しましょう」
と言っているのと同じことで、実は何も弄ってないでしょ?
( (*p)は前置き表現なのに対し、p[0]は後置き表現 )
ただ、それなら「p*」とか「p@」とか書かないで
「p[0]」でいいじゃんって話になるなw
メタやメタメタな話をすると速攻思考が固まる人がいるけど ポインタもその類の気がする。
つまり、アリス以下なんだな。
>>609 では前置版とか後置版とか書いたけど、実行コーディングのときは
p[0]っていう後置版が存在してるから、わざわざp* などという後置版を
新設する意味は無かった。単にp[0]を使えばよいのだった
ならば、宣言時にもp[0]を拝借して、「int p[0];」を
ポインタの宣言として使ったらどうか?
今のC言語では、「int p[0];」という宣言は使われてない(コンパイラに怒られる)。
従って、この宣言に何か意味をつけて使用しても、問題は起きない
使用例 int *p; → int p[0]; int (*p)[99]; → int p[0][99]; int *p[99]; → int p[99][0]; int *(**p[5])[3][2]; → int p[5][0][0][3][2][0]; int (*pf)(void) → int pf[0](void); int (*(*(*(*pf)[5])(void))(void))[3]; → int pf[0][5][0](void)[0](void)[0][3]; なんか微妙だ^p^ 配列との区別がつきにくい… 宣言自体は読みやすい(先頭のintを無視すれば、左から右へ英語読みできる)のだが
___________ ∧_∧∩ /先生!土曜日の早朝にそんなこと ( ´∀`)/< 考えるだけでなく、書込までするのは _ / / / \あまりにも不毛だと思います! \⊂ノ ̄ ̄ ̄ ̄\  ̄ ̄ ̄ ̄ ̄ ̄ ̄ ̄ ̄ ̄ ̄ ||\ \ ||\|| ̄ ̄ ̄ ̄ ̄|| || || ̄ ̄ ̄ ̄ ̄|| .|| ||
>>620 懐かしいAAまで使って凝ってるなw
そんなお前の方がよっぽど不毛だわ
表記も分かりにくいけど 最大の問題点はC言語やそのライブラリが ポインタ酷使し過ぎなことだと思う ・参照型が使える ・参照渡しが使える ・文字列は、基本型もしくは構造体である ・配列は明示しない限り範囲外アクセス不可 始めからこのくらいやれば 半端な知識の人間がポインタに触れることはまずなかったはずだし 本当にアドレスを扱いたいとき以外ポインタ使わなかったろうから うっかりさんが大量発生することもなかっただろう C/C++以外にも確かDelphiとかポインタあるはずだが あんまりポインタで詰まるのを聞かない それは本当に滅多に使わないからだと思う
しかし、それをやったらCが何の為に作られたかがボケてしまう。 アセンブラの代わりに使える低水準言語でなくなる事は、本末転倒だと思う。 Cをそのように使いたいなら、C++でお茶を濁せばいいだけだと思う。
参照こそイラン。 参照なんて結局んところ、[]演算子のオーバーロード時に使いたいだけちゃうんかと。 MyData &operator[](foo f)して、mydata[bar][baz]させたいだけちゃうんかと。
>>623 いや、当然C言語なんだから低水準なものは扱えるべきだよ。
何もポインタ自体を廃止しろと言ってるワケじゃない。
ポインタに依存するものが多過ぎるでしょ、って話。
配列参照だとインデックスが必要 速度を上げるためのポインタ参照だから
Cが特定の人々にとって分かりにくいのは確かだけど 別にシステムプログラミング用途で不合理、不適合ってわけじゃないからな。 変えろとか言われても知らんがな。
つまりシステムプログラミング以外をする人は全部「特定の人々」なのか 実際は何故かプログラミングの教材にすら使われる言語なのだが
だからEasy Cだかなんだか好きな言語でっちあげろよ ごね方が謎過ぎる
Joel氏の言う通り、この程度が理解できないならこの仕事するなって事だな
つまりアレだな、スレタイに対する回答は 「高級言語でもないものを、高級言語であるかのように扱ったから」 だな
実際には色んなところで使われちゃうから、そこが問題点
引っ掛かる人間の割合は、さほど変わっていないと思う。 先天的に才能がない人間も、プログラミングをするようになっただけなんじゃね?
先天的に才能が無い人間がいきなりCに触っちゃう機会があること自体がおかしい
Cってかっこいいの?
C△
Cが駄目ならGoを使えばいいじゃない。
これ ; デリミタっていうんだけどさ、これをつけなきゃエラーになるような そんな言語使ってる奴ってどうみてもゴミだと思うんだけど もしかして「;」これ打ち忘れてコンパイルエラー出すのが楽しいの? そうか、二度と話かけんなよ ゴミは何いってもゴミ
天使ちゃんマジ天使
∨∨∨ 「「「「「 つまりアレだな、スレタイに対する回答は 」」」」」(キリ ∧∧∧∧∧∧∧∧∧∧(きリッッ!キリッッッッ!!きリッ!!!キリッッッッ!!!! ∨ 「「「「「 「高級言語でもないものを、高級言語であるかのように扱ったから」 」」」」」(←きリッッッ!!!!キリッッ!!!! ∧∧∧(キリ!!
ポインタなんて出来るだけ使わなければ良い。 Cが出た頃の昔はポインタで高効率とかあったけど、 今のコンパイラなら最適化が優れてるから 無理に使う必要ない。 ワザと判り難いプログラミングの手本見て、 真似するとか低脳な教え手から習うなよw コンパイラがショボかった昔の技法なんだよ。
クソ本が巷に溢れてるのが悪い
>>20 sizeof(b)=8;
になるのは何で?bはint型だから4じゃないの?
すくなくともintじゃなくて int *だわな。
アドレッシングが64BITの環境かもしれんし。
ポインタの理解に苦しむこと30年、その間書籍も20冊以上買ったが 全くモノにならなかった。 しかしついさっき、ようやく、ようやく開眼した。 ポインタ変数と、それが指し示すデータの実体を 同じメモリの中で考えるから位置関係が気になって混乱する。 メモリが一つで一直線にあるのではなく、 データ専用のメモリと、ポインタ専用のメモリが別々にあると 考えて、二つを並べることで理解できた。 つまり地面にメモリが二本垂直に突き立っているイメージ。 右のメモリにはデータがあり、 左のメモリにはポインタ変数があって右のメモリのアドレスを指している。 ポインタのポインタなら、三本めのメモリが立っていて 真ん中のメモリのアドレスを指している。 こんな簡単なことだったんだ。 人生が開けた感じ。とても清々しい気持ちだ。 次は構造体を頑張るぞ。
ポインタをインクリメントしてデータを破壊しても、原因が理解できなかったりw 30年もかかるんなら、プログラマに向いてねぇよ
ダメだこりゃ。
アドレスって名前はダテでは無いんだけどなー。
>>647 の頭の中ではMapみたいにkey領域とvalue領域があるのかなー・・・。
人が人生のすばらしさに目覚めて感動しているのに 相変わらず君らはレベルの低いレスを付けるのだな。 ポインタで引っかかる人は、みな私のように 頭の中にメモリのイメージを作れずに苦労しているのだよ。 二段メモリの概念は素晴らしい。これでポインタの本を書けるかも。 私を30年も苦労させたリッチーを、卒業祝いに一発殴りたいところだが 亡くなったそうだからチャラにしてやる。 これからはプログラムを作りまくるぞー。
いや30年わからないってそれ明らかに向いてないから
俺の理解だと int *hoge; の場合、宣言してるのはhogeという変数。 *をつけると変数の中身をアドレスにしてアクセスする。アクセス先の型はintとする。 最初に*hogeはint型だが、*を外したらアドレスになる、と覚えようとしたら混乱した。
>>653 そこで多段メモリですよ。
以下のようにイメージすれば、ポインタなんて簡単だ。
先にも言ったとおり、初心者がポインタでつまづくのは、
リニアなメモリの中にデータとポインタが混在したイメージを
頭の中で処理しきれないから。
だったら、処理しやすいイメージに変えてしまえば良い。
実際のメモリ配置がコンパイラの実装に依存する以上、
このイメージが決定的にまずいということはないはず。
1.右のメモリにはデータがある
2.真ん中のメモリには1へのポインタ変数がある
3.左のメモリには2へのポインタ変数がある
4.ポインタ変数に付く*の数は、その変数のあるメモリの
何段前のメモリを見に行くかを示すものである
リンクドリストは要素の数だけメモリがあるのか。大変だな。
多段メモリ革命あげ
ポインタで躓く人って、再帰とかでもつまずくの?ポインタがわからない理由もわからないが、再帰が難しいとされる理由もわからんわ。 モナドとか理解できんのかね?
相関はあるかもしれないけれど、一致はしないよ。 ばりばりに書いているエンジニアで再帰がよく判っていない奴がしばしば居るし、 ポインタをイメージできないのに再帰を使いたがる奴が少なくとも一人居たし。
再起は段数が読めてないとスタックオーバーフローするからなー。 ポインタはホントにメモリの一点をポイントしてるだけなんだがねぇ。 そこに自分用の領域が有るかはわからんけどさ〜。
再起が難しいと言う奴が躓くのはスタックとかそういう次元じゃないだろう
※文系出身の方も多数活躍しています
読書をしない技術系の人は覚えが悪い 読解力が無いので
そもそも、ポインタの概念自体が古いから。 END。 いくら元になっているつっても、もうC言語自体古いし、使う技術者も古い。 いまどき使ってるのはサポート系の企業でしかないよ。覚える必要はない。 もう現場ではC言語は業界標準ではない。今はPHPとRubyとJavaだけ覚えていればそれ以外の言語は不要。
それはないな。 ポインタ知らないとドライバすら書けん。
>>663 Java の new にあたるのは ruby や PHP では何?
RubyはClass#newじゃないの、PHPは知らんが
PHPは普通にC++ライクなnewできたと思う。
10年以上メンテされ続けるシステムとか見てれば ありがたみが分かるよ
逆参照だの参照外しだの言われても初心者は分からんだろ。
ポインタの参照とかね、もう そういやどこぞで間違えた奴がいたね。
結論: ポインタとはメモリーアドレスである
否! ポインタとは、メモリの一点を指し示すための変数である。(ドヤッ
否! ポインタとは、ある形をした入れ物に入れるための、その空間とぴったり同じ形をしたもののことだ。
ポインタはきもちいい
Javaとかやってると、素直にプリミティブ型へのポインタが欲しくなるのも事実。
>>678 もう4年も前だし。その連載、今思い出したけどポインタに限らず間違いだらけでどうしようもなかった。
5年になるが‥‥忘れてもらえないのか‥‥ エピさん大活躍、C++やテンプレートの人かと思ってましたが
エピたんはC++の大御所ですから。 あの企画、編集サイドの都合があるからだろうけど、没にしない限り筆者は赤っ恥晒したまんまだね。
ホントにコンパイル試さなかったのかね?
>>681 でもエピさんの本は、まだよくわからないんだ‥‥。
エピタフ
「ポインタはメモリ上のアドレス」みたいな表現に対して、 すべてがメモリ上とは限らない、単にレジスタである場合もある、みたいな反論を聞いたことがある。
だから?としか言いようがない的外れな反論
>>686 そういやregister変数に対してポインタ取ったらどういう形にコンパイルされるのかって規格で決まってるのかな?
>>688 レジスタ変数のアドレスは取れると規格で決まってる
>>689 §6.5.3.2 Address and indirection operators
Constraints
1 The operand of the unary & operator shall be either a function designator, the result of a
[] or unary * operator, or an lvalue that designates an object that is not a bit-field and is
not declared with the register storage-class specifier.
これはどう訳すべきか
>>690 function designator ってなんだろう?
C++ではもはやレジスタ変数は意味がほとんどなくなってるという事だな
規格を探すと
5.3.1 Unary operators [expr.unary.op]
2 The result of the unary & operator is a pointer to its operand. The operand shall be an lvalue
or a qualifiedid. In the first case, if the type of the expression is “T,” the type of the result is
“pointer to T.” In particular, the address of an object of type “cv T” is “pointer to cv T,” with
the same cv-qualifiers. For a qualified-id, if the member is a static member of type “T”,
the type of the result is plain “pointer to T.” If the member is a nonstatic member of class C
of type T, the type of the result is “pointer to member of class C of type T.”
これでC99と違って特にregisterに言及してない辺りなのかなあ
>>691 http://publib.boulder.ibm.com/infocenter/comphelp/v8v101/index.jsp?topic=%2Fcom.ibm.xlcpp8a.doc%2Flanguage%2Fref%2Flvalue.htm これの「関数指定子」って奴か?
JISのもあるけど読んでも意味わからん
ポインタは、どこも指していないことがある。 ポインタは生成されたが、まだ値が代入されていない。 値が代入された場合、ポインタが指しているデータは、削除されるかもしれない。 削除されても、ポインタに通知されない。 ポインタに値が代入されることで、以前、指していたデータを見失う。
「なぜ引っかかるか」 入門書にも間違いが多いからかな
「指す」だのなんだの言うからおかしいんだよな。 アドレスと、アドレス操作だけの話なのに。
アセンブラの経験ないと、その概念が理解できない罠
int *n; *n = 5;
700 :
デフォルトの名無しさん :2012/10/18(木) 16:28:53.87
ダングリングポインタでぐぐるとわかる
703 :
デフォルトの名無しさん :2012/10/18(木) 19:35:23.12
>>701 当たり前だろ
*pointer だから pointerの指す値を取り出して +1 してるだけなんだから
>>701 は(*pointer)っていう値に+1してtempに代入してる。
(pointer)は変数testのアドレスを指してるので(*pointer)の中身はtestである。
んで、(*pointer)=2600ってしてるので、その瞬間にtestは2600になる。
(*pointer)+1をtempに代入しているので、tempは2601になる。
アドレスをいじる要素はないね。
ポインタ厨発生源発見!
ポインタとアドレスを混同してるバカがまだいるな。
くやしかったらアドレス以外でポインタ実装してみろ
実装がアドレスだと言ってるバカも未だに存在するんだな。
高2病っているんだな。
変数定義時につく*と、式の中に出てくる単項演算子*の関係を考えちゃうと混乱する。 俺も考えると、一番最初はそれで悩んだ覚えあり。 今は、いかなる解釈も考えないようにしている。
今月からCを勉強し始めた者です。ポインタも何となく分かったつもりだった のですが、2次元配列(配列の配列)のところでワケわかめになってますorz int a[3][3] = { {10,20,30,}, {40,50,60}, {70,80,90} }; このような配列を宣言した場合、 @ &a A a B *a (または a[0]) これらのアドレスはみな同じになります。 全部違う意味をもつということは 漠然とは分かるのですが、@〜Bが具体的に何を意味しているのか、自分 では説明できません(今使っている入門書籍でもこれらの違いについては 記述がありません)。 たぶん、これが理解できていないと、配列のアドレス情報を関数へ渡したり、 他のポインタ変数へ渡す時に困ると思うのですが・・・。 アホな私でも分かるように説明して頂けないでしょうか?
>>711 ->* を理解するまでに半万年かかりそうだなw
int (*b)[3][3] = &a; int *c = *a;
>>712 初心者にも分かるように説明してやれ
それが出来ないなら、お前も同じレベルということだw
初心者に説明する必要はない 金払えと思う
716 :
デフォルトの名無しさん :2012/10/22(月) 08:39:03.91
俺もよーわからん。 教えてケレ
>711
int a[3][3] = { {10,20,30,}, {40,50,60}, {70,80,90} };
このような配列を宣言した場合、
@ &a
A a
B *a (または a[0])
これで違いが出る理由について考えて見てはどうですか?
http://codepad.org/TGTgwWBK
720 :
711 :2012/10/22(月) 13:54:44.85
>>718 どうもご親切に有難うございます。
おっしゃるように、それぞれをインクリメントした場合のアドレスの増分
を確認してみました。
@ &a : 0x24 = 36 バイト → 2次元配列全体のサイズ(3x3x4)
A a : 0xC = 12 バイト → サブ配列( {10,20,30} )のサイズ(3x4)
B *a : 0x4 = 4 バイト → サブ配列の要素(int)のサイズ(4)
( a[0]は*aと同じ)
これからいえる事は、
@は、この2次元配列(配列の配列)をひとかたまり(?)として捉えた場合の先頭アドレス
Aは、その中のサブ配列をそれぞれひとかたまりとして捉えた場合の先頭配列の先頭アドレス
Bは、その先頭のサブ配列の中の先頭要素(int)のアドレス
用語が適切かどうかは別として、理解の仕方としては正しいのでしょうか?
それから、これらのアドレス(あるいはポインタ)を使って配列の中の要素を参照する場合、
AとBは使えるけれども、@はスケールが大きすぎて(?)使えない、ということになりますね?
>>719 さんも有難うございます。
全部の要素を一度に上書きとかなら(1)もあり
722 :
デフォルトの名無しさん :2012/10/29(月) 11:54:10.24
Javaはポインターが見えなくしていると本に書いてあった それがJavaの理解をさまたげているらしい。
723 :
デフォルトの名無しさん :2012/10/29(月) 12:03:23.51
int *p,i; i=123; p=&i; *p; pはiのアドレス *pには123が int *p;と*pは別物でしょうか?
int *p; pはポインタですよ宣言 *p はポインタpの指してる先の内容をくださいな 「*p」という同じ記号列だけど意味が違う
Java はポインタがないはずなのに ぬるぽが多発するよね
726 :
デフォルトの名無しさん :2012/10/29(月) 12:11:50.28
>>724 どうもありがとう
なんかここだへんがまぎらわしい
int *p; 「*pはintですよ」 って考える方がまぎらわしくなくていいと思う。 初期化だけは注意が必要だけど。 int *p = hoge; 「*pはintですよ。ただしpはhogeにしといてね。」
int* pの方が好き。宣言してるのは変数pであって*pじゃないし。
int (*p)(void); みたいなのはどうする?
int p, q; q*p; 同じ *p という列でもこの例だと掛け算。 * という記号は状況によって意味がいろいろあるの
それはまずいっしょ
hoge *a[]; とか hoge (*a)[]; とか 優先順位が判らんくて混乱する
733 :
デフォルトの名無しさん :2012/10/29(月) 12:31:27.18
pはint*という型は宣言していると考えればわかりやすいですか ところでc言語って20年ぐらい続いているんだよな その前がpascala,fortranとかこの二つの言語はどうなっている? なんかプログラム自動的に書くツールができてPG,SEは減りそうなんだよな
>pはint*という型は宣言している 日本語でどうぞ。 >ところでc言語って20年ぐらい続いている は? >pascala そんなものありません。 >なんかプログラム自動的に書くツールができてPG,SEは減りそうなんだよな 日本語でどうぞ。 まぁ、「プログラムを自動的に生成」できたとして、それを行なう為にSEは必要だね。
735 :
デフォルトの名無しさん :2012/10/29(月) 12:44:05.08
C言語が続いているのはハード寄りのプログラムが書けるからなんだが それはunixというOSを書いた名残でそういう言語仕様になっている。 そのハードウェアよりのプログラムのところがポインタなんだよな C言語自体は進化しているか分からないが、発展したのにC++という言語があるけど 元のCもつかわれている。 自分のことでもうしわけないが、組み込みの勉強をしようとポリテクに申し込んだら 落ちた。多分若者優先だったかも ハードウェアよりのC言語を勉強するとJavaの理解もはやいと思う。
>>729 そこなんだよな。
普通の変数を 「型名 変数名 (= 初期値);」 で一般化して覚えてると
ポインタが絡んできた途端に頭の中のルールが混乱する。
>>729 自分で書くときはtypedefしとけ。
738 :
デフォルトの名無しさん :2012/10/29(月) 13:03:21.26
>>736 pascala->pascalですよ
>ところでc言語って20年ぐらい続いているよ
バブル崩壊あたりからあるよ
Javaは2000年あたりに知った。
pはint*というintのポイント型を宣言している
ただ上流工程のSEだけ生き残ってあとは減りそうだけどな
人海戦術から自動化していきそうだけどな
739 :
デフォルトの名無しさん :2012/10/29(月) 13:08:04.97
>>736->
>>734 C言語は他の言語より長く使われていることは確か
>>738 とっくの昔から自動化できる部分はじゃんじゃん自動化してる。
ITなんて自動化の固まりじゃん。
何が自動化できてないって?
741 :
デフォルトの名無しさん :2012/10/29(月) 14:30:58.31
Cが長く使われているのは、ポインタがあることなんだろうけどね 危険性が高いのものも同上
それだけだとポインタの無い言語も長く使われている理由が説明出来ない
型情報とポインタ情報をわざわざ別階層であつかえるんだから、 int *a, **b, c, ***d; こう書けるようにしてくれてるんだから、 int* a; int** b; int c; int ***d; などと書く必要性をまったく感じない。
見やすさだろ int* func(); の方が int *func(); より見やすいとかその程度の理由
int p; を「int型の変数p」と訳すから混乱する。「トークンpはint型の数値を返す」と読めば int *p; は「トークン*pはint型の数値を返す」 int (*p)(void);は「トークン(*p)(void)はint型の数値を返す」 と統一的に読める。でも直感的では無い。
どう見やすいのかサッパリわからんがw
int *a と書くのはCな人 int* a と書くのはC++な人 int * a と書くのは変な人 と習った。 そーいうもんかと思い、以後Cで書く場合はint *a、C++で書くときはint* aと書くことにしている
748 :
746 :2012/10/29(月) 20:28:54.89
C++ならこんなんもあるね int & a
bool function(std::string & a);
関係ないが、 int* p; と書くのは許せるが、 INT _tmain(INT iArgumentCount, LPTSTR *lppszArgumentVariables, LPTSTR lpszEnvironmentVariables) と書いた奴は許せない。
システムハンガリアンが許せないの? argc, argvをぐだぐだにしてるのが許せないの? LPTSTR *こういう使い方が許せないの? もっとシンプルにconst TCHAR **としないから許せないの?
mainの第2第3引数はconstじゃねーぞ。 俺はchar *argv[]の書き方が好みだ。 何故なら、char型へのポインタの配列だからだ。
仮引数での配列の宣言なんて何が嬉しいのかわからんな。 ポインタで受け取ってるだけであって配列ではないのだから。
二次元以上だと、またかわってくる
単一の要素へのポインタ(で副作用期待)なのか、 配列なのか、 表記で分けて読みやすくするのがいいかと。 何ごとも読みやすさ優先。by リーダブルコード
副作用の有無はconstで表現すれば十分じゃね?
副作用はどっちでもいいや。 単一なのか配列なのか区別するのに * と [] を使い分ければいいんじゃねってこと。 文字列操作のライブラリは軒並char *だから完全に統一とか無理だけど。
可読性と伝統と芸術性は重要だな。
実体1個のポインタ 実体は配列の先頭 終端は別途もらうか、約束事の番人で打ち切るまでにするか
761 :
デフォルトの名無しさん :2012/10/31(水) 06:57:33.91
参照渡し
マロックを使い出してから余裕になったわ
カロックもいいよね
リーロック最強
alloca 海峡
// realloc()の実装例 void * realloc(void * allocated, size_t size) { void * ptr = NULL; if (size) { ptr = malloc(size); if (ptr == NULL) return NULL; if (allocated) memcpy(ptr, allocated, size); } free(allocated); return ptr; } #if 0 最強って言ってもそれだけではなんもできない子。 #endif
769 :
デフォルトの名無しさん :2012/10/31(水) 21:48:57.55
reallocの仕様を理解していないばかりか、mallocで実装するバカ。晒しage
reallocなんて使った事ないね。
new/deleteとVirtualAllocとHeapAllocしか使わない。 そもそも、再アロケートが発生するような設計などしない。
>766 >770 見てると眠くなるコードだったので書き直した void *realloc(void *allocated, size_t size) { if(size){ if(!(void *ptr = malloc(size))) return ptr; if(allocated){ memcpy(ptr, allocated, size); free(allocated); } return ptr; } return allocated; }
if の () の前後はスペース空けろや
読めない奴が悪い
ptrのスコープが狭すぎる気がするんだが そもそも初期化は式なのか
C++なら通るコード
778 :
デフォルトの名無しさん :2012/11/01(木) 20:53:09.72
sizeに0渡した時にfreeしてないバカ。晒しage
779 :
デフォルトの名無しさん :2012/11/01(木) 21:00:29.75
Cのポインタは可読性がゴミ
780 :
デフォルトの名無しさん :2012/11/01(木) 21:21:40.77
C++の「配列の参照」は知らないと呪文コードだな
ポインタを!演算子でチェックしてるけど、まさか、C++で使うコードじゃないよな?
>773 そもそもこれ標準ライブラリ関数のreallocと動作違うし
リーロック人気だな
mallocの実装が return realloc(NULL, size); でも構わないが、reallocの中でmallocを呼び出すことは決して有り得ない。
>>784 それは振る舞いが同等であればよいというだけの話。
規格は実装を規定しない。
786 :
デフォルトの名無しさん :2012/11/01(木) 23:43:38.64
frontalloc
>>781 6.5.3.3 Unary arithmetic operators ぐらい読んでから喋れ
メモリのアドレスが、 1、2、3ではなく 甲、乙、丙、だったらどうなのか??? ポインタはアドレス「甲」を指す。
欧米の住所(アドレス)は、数字だけだからいいが 日本の住所(アドレス)は、漢字プラス数字だから、比喩として成り立ってない。
>>790 >欧米の住所(アドレス)は、数字だけだから
本当か?
>>766 の発言の意図がわからんw
何を狙って書いたの?w
> 最強って言ってもそれだけではなんもできない子。
これだけでも良いから説明してほしい。
>785 malloc が realloc 呼んでたらどうすんの?
mallocとreallocは同じstdlib提供者が実装するんだからそいつの責任だろ
const char const * const * const p const;
Cのポインタの構文がカオスだから
C++のポインタは楽しい。
>>791 コメは
番地 通り 都市名 州名 zip
なので数字は番地とzipだけ
欧はしらん
釣りなのかマジなのか分からんレスが多すぎるぞ
>>766 以降…
マジレスすると、 毎回メモリコピーをするreallocは糞。
real糞
糞rt
>>63 ,797
例えば
int i=1.5;
int j=(int)i;
int* k=&i;
上のキャスト演算子のほうは「iをintに変えて」って意味だけど
じゃあ「iのポインタを示して(iをポインタに変えて)」ってしようとすると*じゃなく&になるのが感覚として馴染めないな
と言ってもC#で画像扱う際にちょっと触った程度なんだけど
宣言する時には*=ポインタ
キャスト?する時には*=値
ややこし過ぎる
805 :
デフォルトの名無しさん :2012/11/11(日) 18:48:30.20
馬鹿庭無理
806 :
804 :2012/11/11(日) 18:50:30.75
ごめん風邪気のせいかムチャクチャだw とりあえずC#で実際試したら float f=1.5f; int* i=(int*)(int)f; ってやれば問題無いみたいだからまあ、これなら多少は解り易い
ポインタへの参照やポインタへのポインタは 関数の戻り値ではなく「引数を使って」ポインタを返すのに使える 関数が一つしか値を返せないC/C++では重要なテクニック かもしれない
初期のCコンパイラでなければ構造体が、C++ならクラスが戻り値にできるんだが。
出来るけどオーバーヘッドが大きくてメリットないでしょうに
できるかできないかの話にオーバーヘッドを持ち出すバカがいる。
よくやるミス char str[] = "This is a pen."; ... strcpy(str, "You is a big fool man.");
>>809 そもそも、例えばstruct {char r, g, b;}を返す位なら全くオーバーヘッドはない場合もあるよ。
>>811 >strcpy(str, "You is a big fool man.");
~~
この辺?w
これははずいw
you isネタも知らずに突っ込んでるほうが恥ずかしいわ
本質と違えたミスを意図的に指摘している時点で指摘自体がネタだと気付けよ。
今日も大漁ですね
All your pointer are belong to us.
中学時代のパソコンの授業でインターネットを使った時 みんなが自分の好きな漫画や野球のページを見てる時に 自分だけこれみよがしに2chにつないでAAとかを周りに見せてたこと しかも「このページって何?」って聞かれた時に「ヤバイ奴らの集会所みたいなもん」とか答えたこと さらに友達に2chへの行きかたを教えるためにヤフーで2chって検索させて でてきたリンクをクリックして2chのトップページが表示された瞬間に そいつの耳元で「Welcome to Underground」ってささやいたこと
>>815 まじめに質問
you is ... って語感としてどう訳すればぴったりになるの?サリンジャーとかが使っているの?
>>820 ストリートファイターシリーズの元ってキャラがプレイヤー倒した時に言ってた気がする。
仮定法現在
仮性包茎罪
吉原の刑で天国行きだな。
825 :
デフォルトの名無しさん :2012/11/14(水) 11:17:33.53
葦原の姉ちゃんは性病の知識も豊富。
うごごご・・・
I were
She don't care.
>>11 ロングパスだが、これ結構判ってない奴が多い
>a[2]=2[a]が*(a+2)と等しいことを
等しくない
これを等しいと思ってる奴はバグ製造機なので身近に居たら要注意
>>829 それ、何かのギャグ?
つーか、理由を書かなきゃギャグ認定しかできないよ。
>>829 仕様書に
The definition of the subscript operator [] is that E1[E2] is
identical to (*((E1)+(E2))).
と, 書いてあるんだが
> >a[2]=2[a]が*(a+2)と等しい
の, なにが問題なの?
>>830 それ、何かのギャグ?
つーか、理由を書かなきゃギャグ認定しかできないよ。
(*((E1)+(E2))). メガネに見えた
あなるふぁっく
829 名前: デフォルトの名無しさん [sage] 投稿日: 2012/11/23(金) 10:09:53.15
>>11 ロングパスだが、これ結構判ってない奴が多い
>a[2]=2[a]が*(a+2)と等しいことを
等しくない
これを等しいと思ってる奴はバグ製造機なので身近に居たら要注意
C++だと等しいとは限らないかな
839 :
デフォルトの名無しさん :2012/11/27(火) 06:54:14.90
馬鹿には無理
↑これってよく見かけるけど、ボットなのか?
>>837 a がポインタ変数であり有効なアドレスが代入されているという
前提をつけないとイコールにはならないね。
それだと代入になっちゃうから a[2]==2[a]
普通に数学的な意味で等号ととったけど? 普通, 代入は := だろ? メタ文法的には…
2[a]みたいなどうでもいい書き方で熱くなる前に コード一行でも多く書こうぜ、人生は短いぞ
ポインターわかんない奴って, マシンの構造[1]を知らないやつばっかじゃね? [1]の文言は, 計算機の構造でもコンピュータの構造でもいいけどさ
>>845 そんなことはない。
TK-80の頃からアセンブラでバリバリやってきた俺でさえ
Cのポインタはわからないのだから。
わかる頃にはもう人生が終わろうとしているだろうよ。
>>847 才能ないヤツがワンボードマイコンで
バリバリアセンブリ組めるかよw
コンパイラの実装がどうなっているかを考えるといいんだけど 普通はそこまで考えないよね。 ある程度Cが使えてるような人でさえ、コンパイル時に決まる もの、リンク時に決まるもの、ランタイムに決まるものの区別 ができてない人がいたりする。
アセンブリをバリバリ組んでんなら、コンパイラが吐いたコードくらい眺めて、 ポインタがどう実装されるかくらい見てないのかよ
面倒なのでいちいちそんなの触らない
BXレジスタでアドレス操作だっけ? 10年前に覚えたからぼやけてる。
アキュームレーター、ベース、カウンター、データ、ソースインデックス、デスティネーションインデックス、ベースポインタ、スタックポインタ、インストラクションポインタ、フラグ。 と覚えた。
TK-80にそんなものはない
昔のPC板に行けよ
連結リストが全くわからんな。 構造体の定義の中に*付きの自分自身の構造体名…? なんだこれは。もう頭が爆発しそうだよ。
>>857 コンピュータで処理されるのは名前で処理されるんではなくて数字で処理されるのさ。
ただ、コード書くときは高級言語では名前を使うんだ。
んで、連結リストの((*selfname)は次の要素の場所を指す数字が入ってるわけさ。
コンピュータはその数字が正常値なら、見ると直ちにメモリ上の場所を把握できるんだ。
で、次を探すときはそれを使う。
配列だって、あれの(*selfname)は常に隣を指すから暗黙の省略が可能なだけで、意味的には存在するだぜ。
連結リストを初期化するのに自分自身を指し示す方法があるけど、これをセルフフェラと呼んでいるのは俺だけじゃないはず。
>>857 ポインタについて正しく理解してないとそういうことになる。
連結リストは、 最初に登録したノードが最後に・最後に登録したノードが先頭に来るのが 馴染めなかった。
へ?
もしかして : スタック
双方向循環なら、ケツに入れてケツから出すのも、ケツに入れてチンコから出す(業界用語で心太)も簡単。
配列とリストの違いは前後の要素の番地が隣か隣じゃないかだろ
>>858 ああ何となくわかった。
構造体定義での*構造体名は、構造体の中に自分自身があるという
魔術的な宣言ではなく、「同じ構造をしたデータのアドレスを
この位置に持ちますよ」という宣言に過ぎないのか。
ポインタは難しいな。1バイトのアドレスを示すに過ぎない
ポインタ変数自身にデータ型があることが、ポインタの宣言の解釈を
必要以上に難解にしている気がする。
ポインタというより型宣言とは何かがわかってないんじゃないの。 C++やJavaのクラス宣言なんかもそうだし。
そうかもな。 宣言って、ロジックには直接関係しない部分だから 頭が無意識に理解を拒否しているのかも知れん。
871 :
デフォルトの名無しさん :2012/12/09(日) 12:33:03.41
>>868 そうそう。そんな感じ。
ちなみにポインタはsizeof(pointer)でサイズ取れるよ。
だいたいCPUのint長になってる。
>>871 その表現は誤解を招く。
sizeof(pointer)で取れるのはポインタそのもののサイズで、
ポインタが指しているオブジェクトではない。ということが
その表現ではわかりにくい。
だれもそんな話しとらんが
>>871 その表現は誤解を招く。
先ず、CPUのint長という言葉が判り難い。
次に、CPUアーキテクチャとcの整数型のサイズとは直接関係がない。
最後に、整数型のサイズとポインタのサイズも一致するとは限らない。
結局ポインタでつまずく奴って、他の事もきちんと理解できてないんだな そりゃポインタ理解出来なくて当然だわなw
馬鹿には無理
カバには道理
ネコには小判
879 :
デフォルトの名無しさん :2012/12/18(火) 15:05:13.25
ポインタ自体はわかりやすい概念だが Cの文法がハードルを上げてる
新文法の提案よろしく
D
*int p;
構文木理解しろ。コンパイラでも書け
884 :
デフォルトの名無しさん :2012/12/19(水) 22:14:27.88
Cほどポインタで躓きやすいクソ言語はこの世に存在しない
「Cで書いたプログラムがたいてい高速に動作する(中略) Cはカミソリのように切れ味鋭いツール」 『プログラミング作法』p.105 ブライアン・カーニハン、ロブ・パイク
「速い」か「動かないか」しかない。
じゃあ動くように修正すればいいだけじゃん
888 :
デフォルトの名無しさん :2012/12/21(金) 05:49:16.75
Cのヌルポは動かない原因がわかりづらい
馬鹿には無理
日本語で説明しようとする書籍にも原因あるかも
ないない(AAry
892 :
デフォルトの名無しさん :2012/12/22(土) 16:46:53.73
>>891 確かに
そういえば中古ソフトがかなり安く買えた
激安中古ソフト販売( ・∀・) イイネ!
ググってみて
右辺値と左辺値で参照レベルが違うから
レベルが違うよね
895 :
デフォルトの名無しさん :2012/12/24(月) 01:39:22.03
ポインタがアドレスだなんて嘘だ。 ポインタはアドレスを記憶する変数だ。 配列名は先頭要素1byte目のアドレスをもつ定数だ。 ポインタがわからないのはなんで先頭1byteのアドレス を記憶するだけで配列要素にアクセスできるのか、そして C++ではdelete [] p でどうしてメモリが正しく解放できるのか? そこを理解できるかどうかだろう。
896 :
デフォルトの名無しさん :2012/12/24(月) 07:20:26.19
ポインタもわからないバカは死ねw 常識だろw
897 :
デフォルトの名無しさん :2012/12/24(月) 08:57:23.89
わかっててもうまく使えないバカも氏ね!!
>>895 > ポインタはアドレスを記憶する変数だ。
ちがうな。整数は変数じゃないだろ。
ポインタ変数がアドレスを記憶する変数だ。
表記がなあ… int **s;とか何のことかさっぱり。 それに二次元配列やら構造体やら関わってくるとカオス。
やっぱり変数=箱理論をちゃんと教えるべきだよな 箱では破綻するとかいうやつはちゃんと理解してないやつ
やっぱアセンブラから触らせるべきなんだよ 中途半端に抽象化されてるとこから教えても 初心者には理解しづらいし用途も分からない
>>900 オレは類似でバケツがいいんじゃないかと思ってる。
水のかさが変数のバリュー。
配列の説明はいいけど、ポインタの説明は難しいな。
C++になったら、ポインタより参照だし、ナマのポインタってライブラリ書かないと使わないしなぁ。
変数はすでに書かれているようにバケツとか箱にすぎないよね。 箱に荷物が入っているのが普通の変数。 荷物の置いてある場所を書いたメモが入ってるのがポインタ。
Cはメモリアドレスというハードに密着した概念と データ型という高級言語の概念がごっちゃに なっている無茶な言語。 実用的な高級言語がCしかなかった時代はとうに過ぎており、 組み込み系の仕事をする人、処理速度が必要なアプリを 開発する人、言語研究者など以外は、今からCを学ぶ必要は ないと言える。
制御構造とデータ構造という概念のある高級アセンブラだから
Cのポインタ考えたやつは間違いなく頭悪い つまずいてる人は時間を無駄にしたくなければ、こんな欠陥言語捨ててアセンブラやるべき
907 :
デフォルトの名無しさん :2012/12/25(火) 00:48:15.43
馬鹿には無理
別にポインタはメモリに密着というほど密着してないだろ。
キチンとポインタとして動作していれば、メモリアドレスと対応しない実装でも問題ない。
>>893 でも書いたが、同じ表記が左辺値と右辺値で参照レベルが一つずれるのを無意識に使っているのに、
ポインタが出た時点で急に参照レベルを意識させられるから混乱するのだろう。
a=a+1;
最初のaは変数という入れ物、次のaは変数aの中身の値を意味している。
変数名と変数の中身という使い分けを無意識にやっているわけ。
とにかくこのスレに来ている人の9割は 理由もなく漠然とCを学びたいって人だろうから 回れ右してポインタのない言語に行くことを勧める。 変数に*や[]を付けたり外したりして参照レベルを 切り替えるような言語には手を出すな。 一つの言語をマスターできなかったことは口惜しいだろうが それで世の中の誰が困るわけでもない。割り切れ。
3年くらいでポインタの概念が感覚的に理解できるようになってた そんなもんだよ
ショートカットって物があるだろ? 変数名はメモリのある位置を指し示すショートカットのような物だと思えばよい。 もしこのメモリの中身にもショートカットが入っていたら、それがポインタだ。 そのショートカットが指し示す先にもショートカットが入っていたら、それがポインタのポインタだ。
>>908 たしかに右辺に*が来て、デリファレンスして使うのはイメージしやすいが、
左辺に*付きできて、そこに代入する、ってところは確かに一瞬不安になる。
913 :
デフォルトの名無しさん :2012/12/25(火) 12:58:53.35
* 実はこれ、ポインタじゃなくてアナルな。 アナルに指突っ込むと気持ちいいだろ? 違う世界に行けるだろ? そういう意味合いでこの記号が使われてるんだよ
ポインタの概念を理解するのに5分も必要ない。 だがCのポインタが本質の理解を妨げる害悪であることは確かだ。 入門者にCを勧めるやつは地獄に落ちろ。
無能「馬鹿には無理」
アセンブリ言語やれば身に付くんじゃね
シェルスクリプトやればいいよ A=$A;"hogehoge" みたいに、明示的に変数の中身を取るという演算子を要求される。 あるいはTclでもいい。 値のイレモノの中身を見るには何らかの操作がいると認識させれば1段階クリア。
>18 >26 >53 >59 >70 >73 >77 >80 >99 >103 >104 >139 >183 >227 >328 >361 >385 >402 >407 >419 >605 >697 >901 >906 >916 のいうとおりかもしれない
アセンブラをつかってるときの寿命がちじまるようなあやうさ
920 :
デフォルトの名無しさん :2012/12/26(水) 07:13:41.04
頭おかしいなお前らw こんな簡単なの人に教えてもらうなよグズw
>>920 int f(void *&arg);
このarg ってなんだ?
戻り値でポインタを返したいけど、戻り値は別の約束(おおよそエラーステータス)なんで 引数書き換える方法でGO でもポインタ渡しにしたくないので参照渡しで
ポインタのリファレンスも分からない奴は引っ込んでろクズ
こんなところまでわざわざみはりにくる男のひとって
見張りだと?偶然見つけたのが見張りのうちに入るのかw
偶然見つかるわけないだろ 見張ってるから見つかるんだよ
自分がいつも見張ってるから他人も見張ってると思い込んでるんですね それを「投影機制」と言います
↑は境界例
>>928 デタラメ
それに医師でもないのに他人に病名を付けるのは医師法違反
930 :
デフォルトの名無しさん :2012/12/27(木) 11:52:55.84
>>920 その仮引数名をaugと名付けたのはお前だろw
お前が答えられるはずだ
>>927 見張ってないのにどうやって見つけるのさ
まさか偶然ググって飛んできたとか言わないよね
新しい書き込みがあるスレを専用ブラウザでポチポチクリックしていけば 見えるだろうが アホか
> 新しい書き込みがあるスレを専用ブラウザでポチポチクリックしていけば それを見張ると言うのだよ
日本語を正しく使えないのか もしかしてチョン?
ポインタ値の変更やヌルポへのアクセスを見張るHookはどのように実装すれば良いのでしょうか?
鼻フックにするのは簡単です。鼻をテープで止めればよいのです。
先ず使用するデバッガを決めます。
>>935 その対象へのポインタと値を保存しておいて定期的に比較します。
対象のポインタがスタックにあるような場合にはスコープアウトにも気を使います。
難しくないものを難しくしてしまったCのポインタの罪は重い なお今でも被害者続出中
罪じゃねーよ アセンブラの代替言語として最初から開発されたんだからポインタを入れてガリガリに 使うのは既定路線だった
941 :
デフォルトの名無しさん :2013/01/09(水) 07:14:31.73
ポインタもわからないバカは死ねw 猿でもわかるぞw
ポインタを知っててもうまく使えないバカが多いんだよ、だったら使うなヴォケと。
教えられてもわからなかった沖林正紀の悪口はやめろ
バカ「ポインタもわかんない猿がwwwww」 「配列とポインタの違いは?」 バカ「同じだよそんなこともわかんないのか低能wwwww」 「出直してこい猿」
945 :
デフォルトの名無しさん :2013/01/09(水) 12:57:52.14
図で説明しないと伝わりにくいな。
上で文法どうのっていってるレスあったけど、 Cの文法を制限するだけで落ちこぼれがだいぶ減らせるかもね。 ポインタ経由の配列的アクセス禁止 (p[i] と書くな、*(p+i) と書け) 配列名のポインタ化禁止 (p = array と書くな、p = &array[0] と書け) みたいな。
劣化Javaの出来上がり
puts(&"Hello, World!"[0]);
>>946 そんなこざかしい事考えるなよw
配列は、データの集合を表す構造であり、
ポインタは、メモリのアドレスを格納し、そのアドレスへアクセスできるようになる変数だぞw
これ以上でも以下でもないw
ポインタの参照先アドレスにどういうデータを格納するかは、定義には関係ないww
Cの配列はアドレスに対してインデックス可能な名前を つけただけだからねぇ。 そもそもデータの集合とかそういう概念すらない。
問題は**だなあ。 いくら本を読んでもわからないや。 ポインタのポインタって一体何なんだろう。 何のためにそんなものを使うんだろう。 蛇が自分のしっぽを飲み込んでしまう話、 クラインの壺、メビウスの輪…果てしないポインタの連鎖に 俺の脳は思考を停止する。
>>951 まあ、配列をデータの集合と解釈するのも、
使う側のアプローチの問題でしかないからなw
配列が本質的に持ってるものではないな
>>952 ポインタ変数を引数にとって確保した領域のアドレスを入れて返すときに要るだろ。
アドレスという概念を捨てるとポインタは簡単に理解できるんだけどな。
ポインタの理解は、ポインタの存在意義を考える事で先にすすめると思うんだがな。
お姉ちゃんはボインダ
958 :
デフォルトの名無しさん :2013/01/10(木) 01:43:18.84
実装を知ってれば色々発想も捗るのだけど、初心者には無理だしな。 間接参照のありがたみって、最近だとCPUパワーで吹き飛んでしまうからなぁ。 それでも、一歩上に行くんだったらできないと困るけど。
ぬるっぽいんたー
ちんぽいんたー
962 :
デフォルトの名無しさん :2013/01/11(金) 20:45:02.36
unbound な状態が (NULL|NIL) に束縛されるからだ 言語仕様が誤っている
ポインタが難しい理由は なぜポインタが必要なのかがわからないからだ ポインタが必要な理由がわからないのは 40年も昔のチープな環境向けの苦し紛れの小細工であり現代では不要なものだからだ 博物館にでも並べるべき見事なテクニックの数々ではあるが 実用的な価値はほぼゼロ 零戦に乗って21世紀の戦争を戦うようなものだ
現状参照あれば大体片付くからね。 メモリーマップドIOとかも時代遅れだしな〜。
> メモリーマップドIOとかも時代遅れ ??? I/O 空間持ってるCPUの方が時代遅れで、汎化できてない糞CPUなんだが
>>965 PC98とかあのへんを思い出しながら書いたんであんまり深い意味は無い。
ビデオメモリがメインメモリにマップされてたんだよな。確か。
PC9800シリーズはメモリーマップドI/Oじゃないと思うけど。 MC68000みたいなアドレス空間が最初から広いCPU向け。
>ビデオメモリがメインメモリにマップされてたんだよな。確か。 それのどこがメモリーマップドIO
>>965 つうてもほとんど盲腸だろう
PC/ATでもPCI以降はメモリ空間の方にI/O持ってる板のほうが多いんでないかい
PC9800はアドレスバス20bitで1MBのメモリ空間を持ってて うち メインメモリで使うのが 640kB、それ以外を適宜 I/Oに割り当て 権限のない領域の操作をCPUが弾いてくれるなら I/O空間を分ける必要はないかなぁ
でたらめすぎる
>>971 残念だがx86はメモリマップドじゃない、9801も同じこと
i386〜ならIOを隔離することは可能だったはず
CPUで決まってしまうのか なるほど〜
>>974 IO空間が独立しているCPUでもメモリマップドで実装することもあるよ。
>>971 たしか
A0000h - BFFFFh:VRAM
E0000h - E7FFFh:VRAM
F0000h - FFFFFh:BIOS
だったような気がするが、空いている所を適宜I/Oで埋めてくれ
入出力装置のI/Oポートをメモリ空間にマップするか、I/O専用の チップに接続してI/Oのアドレス空間を別に持つかってだけの話。 もちろんI/O専用チップを使っていてもメモリマップドにすることは できる。 だが一般論としてx86 PC(の16bitモード)はメモリマップドI/Oとは 言わない。
978 :
デフォルトの名無しさん :2013/01/14(月) 07:42:37.97
ゴミスレ終了w 死ねやお前らw
組み込みだとメモリマップドI/Oはごく普通だ
それは単に使っているCPUがI/O空間持ってないだけだろう
>メモリーマップドIOとかも時代遅れだしな〜。 に対する話なので、誰もそういう事は言っていない
四十年近く昔のPDP-11でさえI/Oを排除したというのに
そういう問題じゃない。
PC98とかあのへんを思い出しながら書いたんであんまり深い意味は無い。 ビデオメモリがメインメモリにマップされてたんだよな。確か。
蒸し返して欲しそうな流れだったし
987 :
デフォルトの名無しさん :2013/01/15(火) 10:49:10.63
オワコンなクソスレ乙w
何でムスカ大佐
もしどうしても次スレを建てたいという酔狂な人がいるならば 次はマ板でどうぞ
ぬるぽ
俺の子を埋め
孕め
梅干し
梅毒
はどーけん。
ポインタかポインターか
it's pointer!
998 :
デフォルトの名無しさん :2013/01/16(水) 01:17:41.73
地獄行き決定
999 :
デフォルトの名無しさん :2013/01/16(水) 01:19:07.65
このスレは死にます
1000 :
片山博文MZパンク ◆0lBZNi.Q7evd :2013/01/16(水) 01:20:45.46
1000GET!!!!!!
1001 :
1001 :
Over 1000 Thread このスレッドは1000を超えました。 もう書けないので、新しいスレッドを立ててくださいです。。。