Cに関するよくありそうな質問。入門者向け。
ただ単品で i++; や ++i; と書くときには違いはありません。
次のようなコードで違いが生じます。
a = i++; b = ++i;
前者は
a = i; i = i + 1;
と同じで、後者は
i = i + 1; b = i;
と同じです。
(やや難しくいうと、i の値を評価してから1増やすか、1増やしてから評価するかの違い)
typedef が使えるところではすべて typedef を使うべきです。
#define で typedef の代用をすると、型名ではないところまで置換してしまうからです。
#define moji char struct card{ int moji; int suuji; };
このようなとき、メンバの名前である moji まで char に置換されてしまいます。
これは次と同じ意味です。
cond = function(); if( cond != -1 ){ ... }
つまり、cond=function(); を実行して、値が変わった cond を -1 と比較します。
while( (c=getchar()) != EOF ){ ... } なども同様です。
( c=getchar(); を実行し、更新された c を EOF と比べる)
const とは constant の略で、「この変数はこの関数の中で変更されませんよ」という意味です。
つまり、
char str[] = "This is a test."; printf(str);
としたときに、「str の中身をprintf関数が勝手に変更したりしません」ということです。
こんな風↓にしてませんか?
void function(int arg[][]){ ... } main(){ int test[][] = {{0, 1, 2}, {1, 2, 3}, {2, 3, 4}}; function(test); }
もしこうしているなら、function の引数を次のように直してください。
void function(int arg[][3]){ ... } // 要素数を後ろ側だけ指定 main(){ ... }
こうすればコンパイルできるはずです。
(両方指定して arg[3][3] としてもよいが、そうすると3×3配列しか渡せなくなる。後ろ側だけの指定ならn×3(nは何でもよい)配列が渡せる。)
なぜこうしないといけないかというと、二次元配列はメモリ上で次のようになっているからです。
↓test[1][0] [0][1][2][1][2][3][2][3][4] ↑test[0][0] ↑test[2][0]
つまり、呼び出された関数から見ると3×3配列は連続した9個のintにしか見えず、配列がどこで区切られているのかサッパリ分からない、というわけです。
後ろの3を指定してやることで、どこで区切ればいいのか分かるようになります。
同様の理由で、3次元配列ならば後ろ2つ、4次元配列ならば後ろ3つを指定してやらないといけません。
残念ながら、CおよびC++では配列を宣言するときの要素数の指定に変数は使えません。
ではどうするかというと、次のようにします。
#include <stdlib.h> // callocを使うのに必要 main(){ char* array = (char*)calloc(n, sizeof(char)); ... }
これで以後 array は要素数nのchar型配列とまったく同じように使えます。
(なぜこれでいけるのかは長くなるのでポインタと配列の関係とかその辺りを学んで考えてみてください。)
なお、C99(C言語の最新規格)では、char array[n];のような可変長配列の宣言が許されます。ただし、C99に対応していないコンパイラがまだ多くあるので注意してください。
やめてください
このように書いてもコンパイルは通りますが、数学的に表される意味とは違う意味になってしまうので、必ず
if(0 < a && a < 5){ ... }
としてください。
↓まともな説明(分からん人はスルー可)
a = -1 のときを例にとって、こう書くとどうなるのかやってみましょう。
0 < a < 5 の計算順序は (0 < a) < 5 なので、まず 0 < a を評価します。すると、 0 < -1 は偽ですから、0が返ります。(つまり、 式 0 < a の値は0ということ。ここがポイント)
次に 0 < 5 を評価して、結局全体として真になってしまいます。
まとめると、0 < a < 5 の意味は「0とaを比較して、その結果(aではなく、真か偽か)を5と比較する」となります。
これも上の質問と同じ原因で、見た目の意味(aとbとcが等しいならば、・・・)とは違う意味になっています。
if(a==b && b==c){ ... }
などとしてください。
昔の名残です。今では意味がありません。無視しましょう。
(ただ今後64ビット環境と32ビット環境が入れ食い状態になると復活するかもしれませんが・・・。現状スルーで問題なし)
↓以下C++が分かる人向けの説明
例えば、32ビット環境で64ビットのポインタを作ったとする
template <typename T> class ptr64{ private: unsigned long long adr; public: T operator *(){ ... } ...... }; typedef ptr64<char> pchar64; typedef ptr64<int> pint64; ......
このようなポインタ(実際には16ビット環境下での32ビットポインタ)を通常のポインタと区別するためにFARポインタと(16bitと32bitが入れ食いだった時代に)呼んでいた。また通常のポインタを特にNEARポインタと呼ぶこともあった。
「aが真ならばb、偽ならばc」という意味。三項演算子と呼ばれる。
次のように使って、if よりも短く簡潔に書くことができる。
void func(int x){ int y; y = (x>0 ? x : -1 * x); // yは、x>0ならx、そうでなければ-x ... } int even(int n){ return (n%2 ? 0 : 1); // n%2が1なら0を、0なら1を返す }
ちなみに、ifとの違いは、ifが制御構文なのに対して、こちらは式であるということ。
メモリのおかしな場所にアクセスした時にWindowsが出す警告です。
初心者がやりがちなミスで原因となりそうなものは、次のようなものがあります。チェックして下さい。
× scanf("%d", a); ○ scanf("%d", &a);
char str[512]; // 充分な大きさの配列を用意 str[0] = 'a'; ....とするか、malloc, calloc等を使ってください。
int a = 10; × printf("%s\n", a); ○ printf("%d\n", a);
char str[10]; str[0]='t'; str[1]='e'; str[2]='s'; str[3]='t'; printf("%s\n", str);良い例1
char str[10] = "test"; // こうすると自動で最後に'\0'が入る printf("%s\n", str);良い例2
char str[10]; str[0]='t'; str[1]='e'; str[2]='s'; str[3]='t'; str[4]='\0'; // 最後に終端文字を代入 printf("%s\n", str);
char *str = "VIP de yare"; str[5] = 'a';良い例
char str[] = "VIP de yare"; str[5] = 'a';上の例では、strにはリテラル領域のアドレスが入っているに過ぎませんが、 下の例では、strにはリテラル領域の中身がコピーされているから問題ないのです
同上
何もエラーが出ずに落ちる場合でも、メモリアクセスが原因な場合が多いです。
(というより、「落ちる」という現象の原因はほぼ全部なんらかのメモリアクセスエラー)
『「メモリがreadになることができませんでした」とかいって落ちるんですけど…』の項目をチェックしてみてください。