C言語なら俺に聞け( ..
[2ch|▼Menu]
809:デフォルトの名無しさん
07/11/03 05:01:20
>>807
CPUのアドレッシングモードを活用している場合もある
例えばx86のintならレジスタに+04hずつ足しながらレジスタ+レジスタで
アドレスを算出している事が多い
RISCの場合は完全にコード内に組み込まれてしまう場合が多いね

810:デフォルトの名無しさん
07/11/03 05:02:23
#include <windows.h>

int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
MessageBox(NULL, "ないよう", "たいとる", MB_OK);
return 0;
}

XP2、VC++2005Express、cl.exeなんですが、たったこれだけのソースなのに以下のエラーが出てコンパイルできません。
なんでなんだぜ?

test.obj : error LNK2019: 未解決の外部シンボル __imp__MessageBoxA@16 が関数 _WinMain@16 で参照されました。
test.exe : fatal error LNK1120: 外部参照 1 が未解決です。

811:デフォルトの名無しさん
07/11/03 05:43:32
c言語で、田 (←は辺の数2x2の例) のような図形の辺上での経路探索プログラムを作っているのですが、
自分で作ったプログラムでは、再帰処理を用いたせいかやたら速度が遅く、
辺の数を増やすと数時間では終わらなくなってしまいます。
なるべく関数を再帰呼び出ししないように無駄な探索は再帰前にチェックさせたのですが、それでも遅いです。
再帰を非再帰にしたいのですが、自分の技量ではまったく記述が浮かびませんでした。
再帰の内容は、開始位置から上下左右の4つの方向にそれぞれ移動させ、さらに再帰でその処理を繰り返しながら
目的位置までの経路を探索する、というものです。
どうすれば、非再帰にできるか助言頂けないでしょうか?

812:デフォルトの名無しさん
07/11/03 06:08:11
そうすを出しなさい

813:デフォルトの名無しさん
07/11/03 06:17:14
>>811
経路データをどう与えるんだ? あと一筆書きするって事?

814:デフォルトの名無しさん
07/11/03 06:26:38
1、2、・・・N 点に対して (1,2) (4,8)という風にデータを与えればいいか
ただし(a,b)書いたときa<bとしaとbはつながっているとする

前回にすすんだ経路を記録しておいて、進んだときにその経路を消していけばいいのでは?
戻るときは前回のデータを復元していく

たとえば、A=(a1,a2)などとして、ABCDEFGという繋がりが与えられれば、各段階でAからGの経路へ進めるということを繰り返す

初めに(進めるとして)Aへ進めたらBCDEFGが残る 次にAを進めようとするがここは残っていない BCD・・・とチェックしていく

815:デフォルトの名無しさん
07/11/03 06:36:49
繋がりの個数 N、 繋がり (a0,b0)・・・(a(N-1), b(N-1)) 
探索の深さ d
現在の探索位置 k0,・・・k(N-1)
現在位置 x

i = 0・・・N-1
xから繋がり(ai,bi)で移動できるか 出来るならkd = i、d++ とし繋がりを消す すべての繋がりがなくなれば成功でリターンする
どの繋がりにも行けないならd--。kdのつながりを復元して上へ戻る
もしd=0なら失敗でリターンする

816:815
07/11/03 06:44:37
最大1000点の繋がりを処理できるプロクラム作ってやるぜ!

817:デフォルトの名無しさん
07/11/03 06:57:02
最短距離となる経路を知りたいのか?経路の総数を知りたいのか?目的がわからん。
経路の総数なら数式で出したほうがラクだと思うが。

818:815
07/11/03 07:15:55
ねむくなりました とちゆうまで貼り付けておきます

#include <iostream>
#include <time.h>
using namespace std;

main(){
unsigned int A[32768];
int n,a,b,c;
for(n=0;n<32768;n++)A[n];

for(n=0;n<20000;n++){
a=rand()%1024;b=rand()%1024;
if(a!=b){c=a+1024*b; A[c/32] |= 1<<(c%32);}}

int v[10000],d=0,x=0;
for(n=0;n<10000;n++)v[n]=0;

for(n=v[d];n<1024*1024;n++)
if( (A[n/32]>>(n%32))&1 ){
a=n>>10; b=n%1024;
if(x==a){x=b;v[d]=n+1;d++;A[n/32] &= ~(1<<(n%32));}
else if(x==b){x=a;v[d]=n+1;d++;A[n/32] &= ~(1<<(n%32));}
}
}

819:デフォルトの名無しさん
07/11/03 08:07:29
>>810
user32.libをリンクしろ。

user32.libはPlatform SDKもしくはWindows SDKに入っている。
VC++ 2005への設定が容易な分、Windows SDKがおすすめ。

820:デフォルトの名無しさん
07/11/03 08:35:56
>>799
scanf()の戻り値を見て、0だったら終わればいい。

821:デフォルトの名無しさん
07/11/03 10:52:55
>>798
具体的にどこをどう直せばいいですか?

822:デフォルトの名無しさん
07/11/03 10:58:41
>>821
#include<stdio.h>
int tosi(int a,int b){
return a - b;
}
main() {
int year;


printf("あなたの生まれた年は?\n");
scanf("%d",&year);
printf("あなたは今年%d歳\n",tosi(2007,year));
return 0;
}

823:デフォルトの名無しさん
07/11/03 12:17:33
>>811
メモ化

再起をループに書き換えたところで改善されるパフォーマンスは微々たるもの

824:815 出来たけどバグあるよ 直して使ってくれ
07/11/03 14:24:44
#include <iostream>
#define TEN 3
using namespace std;
main(){
int N=TEN*TEN,n,a,b,c;unsigned int A[TEN*TEN/32+1];
//経路生成
for(n=0;n<=N/32;n++)A[n]=0;
for(n=0;n<100;n++){
a=rand()%TEN;b=rand()%TEN;
if(a<b){c=a+TEN*b; A[c/32] |= 1<<(c%32);}}
//経路総数
int keirosu=0;for(n=0;n<N;n++)if( (A[n/32]>>(n%32))&1 )keirosu++;

int *y=new int [keirosu+2], z[TEN], d=0,x,s=keirosu;;
for(n=0;n<keirosu+2;n++)y[n]=0;

for(x=0;x<TEN;x++){ z[0]=x;
while(1){ n=y[d];
if(n>=N){
if(d==0)return 0;
d--;n=y[d];a=n%TEN; b=n/TEN;
if(a==x)x=b;else x=a;
A[c/32]|=1<<(c%32);s++;
y[d]++;continue;}

if( (A[n/32]>>(n%32))&1 ){
a=n%TEN; b=n/TEN;
if(a==x){s--;if(s==0)goto end; x=b; d++; y[d]=0;z[d]=x; A[n/32]&=~(1<<(n%32));continue;}
if(b==x){s--;if(s==0)goto end; x=a; d++; y[d]=0;z[d]=x; A[n/32]&=~(1<<(n%32));continue;}}
y[d]++;}}
end:for(n=0;n<keirosu-s;n++)cout<<z[n]<<"→";}

825:815 すこしなおした バグ取るか、参考として利用してくれ
07/11/03 15:04:39
#include <iostream>
#define TEN 20
using namespace std;
main(){
int N=TEN*TEN,n,a,b,c;unsigned int A[TEN*TEN/32+1];
//経路生成
for(n=0;n<=N/32;n++)A[n]=0;
for(n=0;n<100;n++){
a=rand()%TEN;b=rand()%TEN;
if(a<b){c=a+TEN*b; A[c/32] |= 1<<(c%32);}}
//経路総数
int keirosu=0;for(n=0;n<N;n++)if( (A[n/32]>>(n%32))&1 )keirosu++;

int *y=new int [keirosu+2], z[TEN], d=0,x,s=keirosu;;
for(n=0;n<TEN;n++)z[n]=-1;for(n=0;n<keirosu+2;n++)y[n]=0;

for(x=0;x<TEN;x++){ z[0]=x;
while(1){ n=y[d];
if(n>=N){
if(d==0)break;
z[d]=-1;d--;x=z[d];
A[n/32]|=1<<(n%32);
s++;y[d]++;continue;}

if( (A[n/32]>>(n%32))&1 ){
a=n%TEN; b=n/TEN;
if(a==x){s--;if(s==0)goto end; x=b; d++; y[d]=0;z[d]=x; A[n/32]&=~(1<<(n%32));continue;}
if(b==x){s--;if(s==0)goto end; x=a; d++; y[d]=0;z[d]=x; A[n/32]&=~(1<<(n%32));continue;}}
y[d]++;}}
end:for(n=0;n<keirosu;n++)cout<<z[n]<<"→";}

826:>>811
07/11/03 15:17:37
もう少し悩んでみることにしました、ありがとうございました。

>>825
ソースまで公開して頂きありがとうございます。
じっくり読んで参考にします。

827:デフォルトの名無しさん
07/11/03 16:07:57

すみません。

struct


828:デフォルトの名無しさん
07/11/03 16:12:56
すみません。質問です。

struct AAA{


};

AAA* a = (AAA*)malloc(sizeof(AAA)*4);

とコーディングしていたら客先の担当者からおこられました。
「一応サイズは保証されているけどそれはAAA a[4]ではない」、と。
C++では
AAA* a = new AAA[4]で保証されていますの、といったら
「それはC++だからでしょ?」といわれました・・・・。
AAA* a = (AAA*)malloc(sizeof(AAA)*4);ってやっちゃいけないんですか?

829:デフォルトの名無しさん
07/11/03 16:19:04
a[4]と宣言できる状態なら、客先の反応が正しいかな。

830:デフォルトの名無しさん
07/11/03 16:24:50
>>828
べつに問題ないと思うけど。
固定長で、配列で済むところでmalloc()なんか使うなって意味かな?
malloc()禁止だったら、可変長の配列はどうするんだろ。


831:デフォルトの名無しさん
07/11/03 16:31:19
>>828
それはそもそもCではエラーなわけだが。そのコードが通るなら、C++なのだからvectorかnewを使うべきだし。

832:デフォルトの名無しさん
07/11/03 16:32:26
「AAA* a = new AAA[4]で保証されていますの」

口調に萌えた

833:デフォルトの名無しさん
07/11/03 16:33:57
C99

834:デフォルトの名無しさん
07/11/03 16:35:28
>>830

回答ありがとうございます。
言葉足らずでした。「構造体のmallocなんてありえない!」といわれたのを書くの忘れてました。

すみません。ついでに便乗質問なのですが

(1)
AAA aaa[30][30];
int x=...;
int y=...;
aaa[y][x]=...;

(2)
AAA aaa[30*30];
int x=...;
int y=...;
aaa[y*30+x]=...;

で(2)の使い方をしていたらこれも理解しづらいということで怒られました。
2次元配列のほうが処理的にも良いのでしょうか。

835:デフォルトの名無しさん
07/11/03 16:36:30
ここ見てると自分とのレベルの差が分かるんだが皆学生とかなのか?
レベルの差ってのは自分よりすげーってことね

836:デフォルトの名無しさん
07/11/03 16:37:20
(1)でいい状況でわざわざ(2)を選ぶ神経がわからない。

837:デフォルトの名無しさん
07/11/03 16:38:02
そりゃ、特別な理由がないかぎりa[30][30]と書くべきだろう。

838:デフォルトの名無しさん
07/11/03 16:39:57
>>834
(2)が明確に処理的に優れてることを示せないのに(2)を選んだ理由は?

839:デフォルトの名無しさん
07/11/03 16:40:15
「構造体のmallocなんてありえない!」
callocを使えということかな?

840:デフォルトの名無しさん
07/11/03 16:40:49
>>834
いいえ、処理的にはどう考えても(2)の方が何かと都合がよいと思います。

私のところではパフォーマンスを要求する場合は(2)の形式でかつ、オフセット計算関数を用意することが多く、
そうでない場合には(1)の形式を配列の配列ではなく配列へのポインタ配列の形で使うことが多くなります。

841:デフォルトの名無しさん
07/11/03 16:46:45
>>834
> 言葉足らずでした。「構造体のmallocなんてありえない!」といわれたのを書くの忘れてました。 

その担当者は、プログラム知らなさすぎなんじゃね?



842:834
07/11/03 16:46:55
>>838

処理的には
a[y]のアドレスにとんで、そこからsizeof(AAA)分移動になるかと思いそれなら最初から1次元で
計算したほうがよいのかなと思いまして。

>>839

担当者いわく、構造体の領域確保はできない、ない、ということらしいです。

843:デフォルトの名無しさん
07/11/03 16:49:51
>>842
>841に同意。後学の為に、その担当者の所属企業をお教え願えませんか?w

844:デフォルトの名無しさん
07/11/03 16:51:15
>>840
コードの可読性は(1)のほうがいいだろ。
パフォーマンスの問題なら、(2)のほうが何パーセント処理時間が短くなるとか、具体的な数字を見せて説得すれば
いいんじゃね?

845:デフォルトの名無しさん
07/11/03 16:51:51
>>842
>処理的には
>a[y]のアドレスにとんで、そこからsizeof(AAA)分移動になるかと思いそれなら最初から1次元で
>計算したほうがよいのかなと思いまして。

だからなんで「よい」と思うのか教えてくれ。

846:デフォルトの名無しさん
07/11/03 16:52:00
>>842
>a[y]のアドレスにとんで、そこからsizeof(AAA)分移動になるかと思い
コンパイラの最適化をなめてはいけない。
2次元配列のほうが、より最適化される可能性が高い。

847:デフォルトの名無しさん
07/11/03 16:55:06
>>840
>いいえ、処理的にはどう考えても(2)の方が何かと都合がよいと思います

>>842
>計算したほうがよいのかなと思いまして

自分なら、思うだけじゃ、可読性を犠牲にする気にはならないな
実測して優位性を証明しる

848:デフォルトの名無しさん
07/11/03 16:57:59
おまいらwwwここ見てみwwwwwww天才降臨wwwww

1 :以下、名無しにかわりましてVIPがお送りします。:2007/11/03(土) 14:02:12.45 ID:o3DHz8v00
URLリンク(id13.fm-p.jp)


裏絵バロスwwwwwwwwwwwww


やあ、ここのBBSのパス解析できたら明日おにゃのことセックルできるお(^ω^ )

スレリンク(news4vip板:698-698番)

849:デフォルトの名無しさん
07/11/03 17:06:48
int a[30][30];
int b[30*30];

int foo(int x, int y) {
return a[y][x];
}

int bar(int x, int y) {
return b[y * 30 + x];
}

gcc でコンパイルしたら、foo も bar も同じコードになった。
pushl %ebp
movl %esp, %ebp
movl 12(%ebp), %ecx
movl 8(%ebp), %eax
popl %ebp
movl %ecx, %edx
sall $4, %edx
subl %ecx, %edx
addl %edx, %edx
addl %eax, %edx
movl _a(,%edx,4), %eax ;または _b(,%edx,4), %eax
ret


850:デフォルトの名無しさん
07/11/03 17:47:38
fseekについて教えてください。

fseekでファイルのヘッダ部分を飛ばしたいのですが
fseek(fp,???,SEEK_SET);

4行飛ばしたい、というときは第2引数はどうすればいいのでしょうか?
4行をバイトにどう直すのかわからなくて


851:デフォルトの名無しさん
07/11/03 17:54:06
ワロタw

852:デフォルトの名無しさん
07/11/03 17:56:39
1行を80バイトとしてCRとLFの2文字分を足して82バイト。
4行だから、82x4=328バイト分seekするというのはどうだろう?

853:デフォルトの名無しさん
07/11/03 18:01:55
>>852
ありがとうございます、やってみます

>>851
やり方がおかしいのか、おかしいことを言っているのでしょうか?
普通はどうやるんですか?

854:デフォルトの名無しさん
07/11/03 18:02:31
>>822
ありがとうございます。

855:デフォルトの名無しさん
07/11/03 18:10:11
>>853
1行が何バイトかは、その行を読み込んで改行まで数えないとわからないのが普通

856:デフォルトの名無しさん
07/11/03 18:10:58
>>853 オイオイw

857:デフォルトの名無しさん
07/11/03 18:20:44
でも、データをストリームとして扱うようになったのはUnix以降じゃなかったか?
だからそれ以前の人ならそういう発想でも別に変でないかも。

858:デフォルトの名無しさん
07/11/03 18:51:46
何とかできました。

>>855
普通はそうなのですか…結構アナログな感じなのですね


ただ、すんません。疑問が増えました。
データをストリームとして扱う、の意味がわかりません


859:デフォルトの名無しさん
07/11/03 18:57:10
>> 850
現在位置を取得し、これをAとします。
getsを4回呼び出します。
現在位置を取得し、これをBとします。
fseekで現在位置をAの場所に戻します。
fseekB-Aバイト分現在位置を進めます。

860:デフォルトの名無しさん
07/11/03 18:58:30
>>858
「データをストリームとして扱う」というのは、普通にファイルを頭から順に読んでいくこと。
これに対して、「固定長のブロックとして扱う」やりかたがあって、
ひとつひとつのカタマリが同じ大きさの場合に、
fseekを使ってデータの開始位置にシークして、必要なデータのみを直接読み書きすることを言います。

861:デフォルトの名無しさん
07/11/03 19:24:43
>>859-860
ありがとうございました。
今まで何も理解せずにプログラムしてたのだと思い知りました。。。


862:デフォルトの名無しさん
07/11/03 21:41:45
KOUZOUTAI data[num];

KOUZOUTAI *data
data = (struct Point *)malloc( sizeof(struct Point) * num );


の違いって何なのでしょうか?
どちらもnum分の構造体を確保していると思うのですが。


それとポインタの場合で二次元の構造体を確保する方法教えてください。

863:デフォルトの名無しさん
07/11/03 21:42:36
>>862
すません。
4行目はPointじゃなくて KOUZOUTAIです。

864:デフォルトの名無しさん
07/11/03 21:59:04
int data[10]

int *data = (int *)malloc(sizeof(int)*10)
の違いと同じ。

865:デフォルトの名無しさん
07/11/03 22:00:33
質問です。
ファイルの更新時刻を得るのってどうすればいいんでしょうか?

866:デフォルトの名無しさん
07/11/03 22:01:13
OS次第

867:デフォルトの名無しさん
07/11/03 22:07:28
右クリック → プロパティ

868:デフォルトの名無しさん
07/11/03 22:17:23
>>866
標準ライブラリにそういう関数が入ってたりしないってことでしょうか

869:デフォルトの名無しさん
07/11/03 22:23:21
>>868
ファイルシステムに依存する情報なんで、標準には入ってない

870:デフォルトの名無しさん
07/11/03 22:23:32
>>864
んー
その、構造体で宣言するのと、ポインタで宣言するのと利点があまりわからず

あと2次元の構造体をポインタで確保する方法もお願いします

871:デフォルトの名無しさん
07/11/03 22:27:14
まず「二次元の構造体」を詳しく。
構造体の二次元配列をポインタで確保するのなら、普通の変数を二次元に確保するときと同様でいい。

872:デフォルトの名無しさん
07/11/03 22:28:40
スコープを跨ぐことができない・できるの違いぐらいじゃない?
あと、後者の方が前者と比べると遅そうだけど、これは実装・環境に依存するのかな。

873:デフォルトの名無しさん
07/11/03 22:34:47
>>871
KOUZOUTAI data[10][10]
のようなことです。

二次元配列をポインタで確保するやり方調べてきます。やったことなくて。

>>872
なるほど。
というと、グローバルでKOUZOUTAI data[10]とするのと同じような感じになるのでしょうか?


874:デフォルトの名無しさん
07/11/03 22:49:42
KOUZOUTAI (*data)[10]

875:デフォルトの名無しさん
07/11/03 23:25:23
struct e{
int value;
};

int main(){
struct e **p,(*q)[5];
int i;

//こんな方法とか
p = (struct e**)malloc(10 * sizeof(struct e*));
for(i=0; i<9; ++i){
p[i] = (struct e*)malloc(5 * sizeof(struct e));
}

//こんな方法がある
q = (struct e(*)[5])malloc(5 * 10 * sizeof(struct e));

//こんな風に使う
p[0][0] = 0;
q[0][0] = 0;

//忘れずに解放
for(i=0; i<9; ++i){
free(p[i]);
}
free(p);
//qの方が解放は楽
free(q);

return 0;
}

876:875
07/11/03 23:26:22
繰り返し部分間違えた
for(i=0; i<9; ++i)

for(i=0; i<10; ++i)

877:デフォルトの名無しさん
07/11/03 23:38:36
**ばっかりでわかりにくいなぁ。

878:デフォルトの名無しさん
07/11/03 23:42:07
だいぶはしょりますが・・・
int i;
char b;
char a[] = "abcdefg";
scanf("%s",&b);
while(b != a[i])
i++;
・・・略
みたいなbで打たせた文字をaの配列から探させるようなプログラムを書くとwhile文のところで、
「char型はchar*型に変換できない」
というエラーが出ます。どうすればbの文字ををaの配列から参照できますか?
わかりにくくてごめんなさい

879:デフォルトの名無しさん
07/11/03 23:46:27
>>878
>534の下3行。

880:デフォルトの名無しさん
07/11/03 23:49:25
>>878
とりあえずそのようなエラーは出なかった

881:デフォルトの名無しさん
07/11/03 23:50:30
>>879
ありがとうございます。%cにすればいいってことですか…?

882:デフォルトの名無しさん
07/11/03 23:56:23
878ですが、エラーはでなくなったのですがどうやらwhileで無限ループがおきている気がします。
上のプログラムだと、例えばscanfでdを打ったらiが3になった時点でwhile文から出るはずですよね?

883:デフォルトの名無しさん
07/11/03 23:57:23
iを0で初期化してる?

884:デフォルトの名無しさん
07/11/04 00:01:00
>>883
しています。無限ループではなく他のどこかがおかしいのかもしれません。
もう一度しっかり見てみます。
ありがとうございました。

885:デフォルトの名無しさん
07/11/04 00:03:57
>>884
bにhを入れたらどこで止まると思う?

886:デフォルトの名無しさん
07/11/04 00:06:17
>>885
そうなると無限ですか?
でも実際はこの配列はa〜zで、cとかdとかを入れても次の動作にいきません。

887:デフォルトの名無しさん
07/11/04 00:08:04
whileに入る前にbに入ってる文字をprintfで確認したか?

888:デフォルトの名無しさん
07/11/04 00:08:12
出来る限り、はしょらないで載せてくれ
>>883みたいな細々したことを指摘しないといけなくなるから

889:デフォルトの名無しさん
07/11/04 00:08:54
>>886
その実際のソースを書け

890:デフォルトの名無しさん
07/11/04 00:12:11
>>878のソースで「char型はchar*型に変換できない」というエラーが出るというのが気になるんだが……
scanfの%指定ミスでこんなエラー出ないよな? せいぜい実行時エラーであって

891:デフォルトの名無しさん
07/11/04 00:15:44
ごめんなさい
#include<stdio.h>
int main(void){
char a;
char letter[] = "abcdefghijklmnopqrstuvwxyz";
int i = 0;
printf("半角英小文字を1文字入力。:");
scanf("%s",&a);
while(a != letter[i])
i++;
printf("%s",letter[i]);
return 0;
}

です。


892:デフォルトの名無しさん
07/11/04 00:18:27
scanf指摘されたところ直ってないじゃん

893:デフォルトの名無しさん
07/11/04 00:20:18
>>891
%sを%cに置換してこい。printf("%s",letter[i]);文字コードがポインタだと解釈されてしまう

894:デフォルトの名無しさん
07/11/04 00:20:29
returnの位置

895:デフォルトの名無しさん
07/11/04 00:22:40
>>892
>>893
ありがとうございます。
>>894
何か場所おかしいですか?

896:デフォルトの名無しさん
07/11/04 00:23:01
>>892
直す前のを載せたんじゃないの?

>>891
printf("%s",letter[i]);

printf("%c",letter[i]);

897:デフォルトの名無しさん
07/11/04 00:23:21
バッファオーバーラン

898:デフォルトの名無しさん
07/11/04 00:26:45
>>897

899:デフォルトの名無しさん
07/11/04 00:32:21
みなさんありがとうございます。
おかげでなんとかできました。

1つ気になるのですが"%c"と" %c"の違いはなんですか?

900:デフォルトの名無しさん
07/11/04 01:04:58
>>899
%cと%sの違いなら、

・scanfの場合
%cなら、引数として渡されたアドレスの指し示す先に、標準入力からとってきた一文字を入れる
%sなら、引数として渡されたアドレスの指し示す先を配列の頭と見て、文字列を入れていく。最後に\0を付加

・printfの場合
%cなら、引数として渡された数値に対応した一文字を標準出力に出力
%sなら、引数として渡されたアドレスを配列の頭と見て、その指し示す先を\0にぶち当たるまで出力する

901:デフォルトの名無しさん
07/11/04 01:10:39
' '

902:デフォルトの名無しさん
07/11/04 01:22:51
>>900
ありがとうございます。
よくわかりました!

903:デフォルトの名無しさん
07/11/04 10:02:07
ところで、何が出来れば入門を卒業したことになるの?

904:デフォルトの名無しさん
07/11/04 10:25:43
言語の文法マスターしたぜ!もう学ぶことねぇwwww

え・・ポインタのポインタ?関数ポインタ?どうだったっけ?

まだまだだな俺 ←この辺から入門卒業

905:デフォルトの名無しさん
07/11/04 11:07:39
>>904
それ面白いなw
初心者に教えてるつもりが逆に初心者から教わってることに気付いたあたりも入門卒業っぽいぜ。

もう一歩踏み込む気構えが出来た頃合がそうなのかもね。

906:デフォルトの名無しさん
07/11/04 11:37:37
>>904
俺的にはそこで、宣言の問題だと気がついたら卒業としたい。

907:デフォルトの名無しさん
07/11/04 11:44:38
「このスレ卒業テスト」と称して課題を出せばいいんじゃね?

908:デフォルトの名無しさん
07/11/04 12:29:28
CでGUIアプリを作れたら入門者卒業

909:デフォルトの名無しさん
07/11/04 12:42:34
>>908
んなわけあるか

910:デフォルトの名無しさん
07/11/04 12:57:44
>>908
コピペで終わりじゃねーかw

911:デフォルトの名無しさん
07/11/04 14:17:19
ポインタが理解できたら入門者卒業

912:デフォルトの名無しさん
07/11/04 14:24:09
ポインタも
*****pくらいついたらわけがわかりません。

const ***p
***const p
とかなったらわかりません

関数ポインタなんて使ったことありません。なんで使うんですか。
GUIなんて作ったことなんてありません

でも入門卒業したいです><

913:デフォルトの名無しさん
07/11/04 14:57:08
関数ポインタの使い道が分からないなら卒業は無理だな

914:デフォルトの名無しさん
07/11/04 15:13:41
関数ポインタは便利だけど、あまり使うことはないですよ。
私は、DLLを使うときくらいですかね。

915:デフォルトの名無しさん
07/11/04 15:17:29
卒業せんでも中退すれば

916:デフォルトの名無しさん
07/11/04 15:37:45
typedef struct
{
char *str;
int (*myfunc)();
} object;

int main()
{
object obj;
obj->str = malloc(sizeof(char) * 256);
fgets(obj->str, 256, stdin);
obj->myfunc();
puts(obj->str);
free(obj->str);
return 0;
}

int myfunc()
{

}

917:915
07/11/04 15:38:59
>>916
×object obj;
○object *obj;

918:デフォルトの名無しさん
07/11/04 16:23:53
>>916
え、そんなやり方でmyfuncを呼び出せるの?
いつ関数ポインタをセットした?

919:916
07/11/04 16:28:35
>>918
あー忘れてた。
obj->myfunc();
の前に、
obj->myfunc = myfunc;
が必要だった。あかんね。

920:デフォルトの名無しさん
07/11/04 16:54:01
#include <stdio.h>
int main(void)
{
int c;
c = getchar();
printf("%d\n", c);
c = getchar();
printf("%d\n", c);
return 0;
}

このプログラムをBCC5.5.1でコンパイルして実行し、"a"と"Ctrl+Z"を1度に入力した場合、

D:\c\my\test2>test2.exe
a^Z
97
-1

このような結果になりました。
ところが、VC++2005EEのcl.exeでコンパイルし、同じように実行すると、

D:\c\my\test2>test2.exe
a^Z
97
26

このような結果になります。なぜVC++20005では正しい結果が得られないのでしょうか。

921:デフォルトの名無しさん
07/11/04 16:59:04
fflush(stdout);
これを main の return の前に。


922:デフォルトの名無しさん
07/11/04 17:06:20
*p (*p)
の違いがわかりません><

923:デフォルトの名無しさん
07/11/04 17:07:39
あっそ

924:デフォルトの名無しさん
07/11/04 17:08:00
教えてください><

925:デフォルトの名無しさん
07/11/04 17:10:33
>>920
stdinの初期設定が違うんじゃね?
^C が受けられるかどうかの設定もできるし。
あるいは ^D とか。

926:デフォルトの名無しさん
07/11/04 17:10:56
>>922
1+2 と (1+2) の違いを考えてみるといいよ

927:デフォルトの名無しさん
07/11/04 17:17:19
>>926
ありあとやんしたー

928:デフォルトの名無しさん
07/11/04 17:44:16
>>920
コレってマジ?
なんで今更こんなバグが残ってんの?

929:デフォルトの名無しさん
07/11/04 17:52:38
>>921
そのようにしても結果は変わりませんでした。

>>925
どうやって設定すればいいのでしょうか?

930:デフォルトの名無しさん
07/11/04 19:11:13
>>928
うちでも確認できた。これはひどいな。

931:デフォルトの名無しさん
07/11/04 19:22:15
そこは仕様では決められてない部分だから、
バグだとか、どっちが間違ってるとかでは無いんじゃね?


932:デフォルトの名無しさん
07/11/04 19:55:34
>>931
でも、stdio.hでは#define EOF (-1)となっているし、^Zだけならちゃんと-1と表示される。

933:デフォルトの名無しさん
07/11/04 20:55:14
そうじゃなくて EOF の前に一応 ^Z も発行するかどうか。

934:デフォルトの名無しさん
07/11/04 21:57:55
>>921
プログラム終了時に、標準入出力もふくめてファイルは全部フラッシュ&クローズされるから、
それは意味ない。

935:デフォルトの名無しさん
07/11/04 22:20:44
num[4][5]={
{0,1,2,3,4},
{5,6,7,8,9},
{10,11,12,13,14},
{15,16,17,18,19}
};
関数に配列の{5,6,7,8,9},とかを1行だけ送りたいのですが
どういう風にやればいいのでしょう?



936:デフォルトの名無しさん
07/11/04 22:41:03
送るだけならnum[1]とかを渡すだけ。

937:デフォルトの名無しさん
07/11/04 22:59:13
EOFについて。
MSのドキュメント調べたけど、記述が見つからなかったので仮説。
てゆーか、コマンドプロンプトの制御文字に関する仕様のドキュメントてどこさ?

行頭にある^Zはコマンドプロンプトが解釈してプログラムに「データ無し」を通知する。
行中にある^Zは文字「0x1A」をプログラムに通知する。
という理由で>>932な動きをするっぽい。
ここはコマンドプロンプトの仕様だと思うんだけど、文献ミツカンネ。

stdinをバイナリモードで開くと同じ挙動をするので、
テキストモードでのデータ中の0x1A(制御コードのEOF)の扱いの差のようだ。
バグというよりは仕様の範疇っぽいね。
0x1Aを含むファイルをテキストモードでオープンして読んでみれば
もうちょい分かりそうなんだけど、データ作るのめんどかったんで誰か頼む。

○ バイナリで開き直したサンプル
#include <stdio.h>
int main()
{
int c;
freopen("CON", "rb", stdin);
do{
printf("%d\n",c = getc(stdin));
}while(c != EOF);
return 0;
}


938:デフォルトの名無しさん
07/11/04 23:10:31
>>936
受け取る方を1次元にしてやれば、まとめて受け取られるんですね。
できました!ありがとうございました!

939:デフォルトの名無しさん
07/11/04 23:18:03
>>937
VC++2005でもファイル中の0x1Aは、getcで読んだらEOFが返ってきた。
やっぱバグじゃねーの?

940:デフォルトの名無しさん
07/11/04 23:24:30
Turbo Explorer
無償のアプリケーション開発環境
URLリンク(www.forest.impress.co.jp)

941:デフォルトの名無しさん
07/11/04 23:27:37
むぅ…すんません
今度は
num[4][5]={
{0,1,2,3,4},
{5,6,7,8,9},
{10,11,12,13,14},
{15,16,17,18,19}
};
の 0 5 10 15 の縦の列の総和を出したいだけなんですが
これも丸々関数に送りたいのです。。。
1次元配列にいっそのこと退避させたほうがいいんでしょうか

942:デフォルトの名無しさん
07/11/04 23:36:02
numごと渡して計算せよ

943:デフォルトの名無しさん
07/11/04 23:42:07
numがココでは小さいんですが
1000×1000と大きい場合を想定しています。
numごと渡したら、メモリがムダに消費されるとかポインタの解説ページとかで見た気がして



944:デフォルトの名無しさん
07/11/04 23:44:37
それならなお更numを渡せ

945:デフォルトの名無しさん
07/11/04 23:44:40
numごとって、本当に渡すわけじゃないぞ。アドレス渡すだけだから。

946:デフォルトの名無しさん
07/11/04 23:47:36
>>941
一次元配列を使うと32*1000バイト消費するが
そのままだと何も新規に生成しないから0バイトで済む

947:デフォルトの名無しさん
07/11/04 23:50:24
何かもう一個同じモノが作られるっぽいことが書いてあって、怖気づいてました。

ところでnumそのものの値も更新したい場合は、参照渡しするのですよね?

948:デフォルトの名無しさん
07/11/04 23:51:17
本当にnumそのものの値を変更したいのか?

949:デフォルトの名無しさん
07/11/04 23:53:11
C言語はふつうにやると配列は参照渡しになる
コピーを渡す方は難しい

950:デフォルトの名無しさん
07/11/04 23:57:50

クソの集まり創価学会

偽善者が政治活動、公明党

キチガイ集団が政治活動、公明党

池田狂信ネズミ講が政治活動、公明党

騙されバカ信者、池田犬作チョン大教祖様、さっさと死ねや

951:デフォルトの名無しさん
07/11/05 00:10:49
>>948-949
すみません。。。混乱してきてしまって。

今は総和だけ求めてるのですが
やりたいのは行で計算をして、その行の値を全部更新して
その更新された状態で列を計算して、全部の列を更新するということをしようとしてます。

配列と関数の関係をやりなおしてきます。

952:デフォルトの名無しさん
07/11/05 00:31:26
>>951
それならなおさらアドレス渡しでいいじゃん

953:デフォルトの名無しさん
07/11/05 00:38:26
>引数の引き渡しには,値渡しと参照渡しがある.
>C言語では,特別なことをしない限り,変数は値渡し,配列は参照渡しになる
今まで配列も値渡しだと思っていました…納得できました。

ありがとございましたー

954:920
07/11/05 11:44:12
検証してくださった方、ありがとうございます。

#include <stdio.h>
int main(void)
{
int c;
do {
printf("%d ", c = fgetc(stdin));
} while (c != EOF);
return 0;
}

このプログラムを実行し、
コマンドラインから"abc^Zdef"を入力すると、(^ZはCtrl+Z)

D:\c\my\test5>test5.exe
abc^Zdef
97 98 99 26

と表示され、入力待ちになります。

955:920
07/11/05 11:44:52
それに対して、

61 62 63 1A 64 65 66

の内容のテキストファイル(↑は16進ダンプしたもの)をstdinにリダイレクトして
実行してみたところ、

D:\c\my\test5>test5.exe < test.txt
97 98 99 -1

このような結果になり、プログラムが終了しました。

プログラムは全てVC++2005でコンパイルしました。
どうやらコマンドラインの行中にEOF(0x1A)があった場合、
stdio.hで定義されたEOF(-1)ではなく、
テキストファイルでのEOF(0x1A)として読み込むようです…。

BCC5.5.1ではどちらの場合も、

97 98 99 -1

と表示され、プログラムが終了します。

956:デフォルトの名無しさん
07/11/05 12:56:46
製品版では直ってるのかな。
一応、MSに報告したらどうだろ。

957:デフォルトの名無しさん
07/11/05 13:09:38
別にエラーじゃないのでは?
EOFのASCIIコードを表示しているのだから

958:デフォルトの名無しさん
07/11/05 13:11:09
EOFのアスキーコードじゃないよ。

959:デフォルトの名無しさん
07/11/05 13:12:35
EOF じゃなくて SUB だな。
つーか単なる ^Z だが。

960:920
07/11/05 13:52:32
>>958 >>959
Wikipedia項目リンク
>ASCIIでは例えば、改行文字は0x10、水平タブは0x09、EOF(End Of File、ファイル終端マーク)は0x1Aである。

961:920
07/11/05 14:01:52
>>957
コマンドラインの行頭にあるEOFや、リダイレクトされたファイルにあるEOFは
stdio.hに定義されたEOF(-1)と読み取ることを考えると、一貫性が無いように思えます。

962:デフォルトの名無しさん
07/11/05 14:12:08
ASCII に EOF なんて無いだろ。
それ書いたの誰だよ?

URLリンク(e-words.jp)
Wikipedia項目リンク
URLリンク(www.mew.org)


963:デフォルトの名無しさん
07/11/05 18:42:18
そりゃ、文字コード体系表見てもEOFなんてあるわけないわな。

964:デフォルトの名無しさん
07/11/05 21:29:32
c言語の除算はどのようなアルゴリズムで行っているんでしょうか?
わかる方がいればよろしくお願いします

965:デフォルトの名無しさん
07/11/05 22:03:20
>>964
コンパイラによるんじゃないでしょうか。
あと、たぶん、かなりアセンブラよりの質問だと思います。

966:デフォルトの名無しさん
07/11/05 22:04:36
CPUよりの話でしょう
浮動小数点や整数演算はハードでします

967:デフォルトの名無しさん
07/11/05 22:09:11
テキストモードでfopenしたら0x1aで終了だろ

968:デフォルトの名無しさん
07/11/05 22:13:31
いつまで、CP/Mの呪いをひきづらなければならないのやら。

969:迷い人
07/11/05 22:36:25
こんばんは。つぎのようなプログラムがわかりません
1を入力すると

2を入力すると

xx

4を入力すると

xx

xxx

xx

xxxx

xx

xxx

xx

となるようなものです。途中まではかけたのですが肝心なところがよくわかりませんでした。
途中までは次のようになりました


970:迷い人
07/11/05 22:37:28
#include <stdio.h>

void X_sequence (int n);
void X_triangle (int n);

main()
{
int n;
printf("Enter a number : ")
scanf("%d",&n);
X_triangle(n);
}

void X_sequence(int n)
{
int i;

for (i=1; i<=n; i++)
printf("X");
printf("\n");
}

void X_triangle(int n)
{・・・・こっからわかりません!

教えてくださいおねがいします

971:デフォルトの名無しさん
07/11/05 22:39:06
if (n > 2) X_triangle(n - 1);
X_sequence(n);
if (n > 2) X_triangle(n - 1);


972:デフォルトの名無しさん
07/11/05 22:39:22
printf("1を入力すると\n
x\n
2を入力すると\n
x\n
xx\n
x\n
4を入力すると\n
x\n
xx\n
x\n
xxx\n
x\n
xx\n
x\n
xxxx\n
x\n
xx\n
x\n
xxx\n
x\n
xx\n
x\n
");

973:デフォルトの名無しさん
07/11/05 22:39:38
if (n >= 2) X_triangle(n - 1);
X_sequence(n);
if (n >= 2) X_triangle(n - 1);


974:デフォルトの名無しさん
07/11/05 23:01:21
ふとオープンソースの午後のこーだのソース見てみたら、C言語だったけど、
どれくらい経験つめばこれくらい書けるようになるんですかね・・・

975:デフォルトの名無しさん
07/11/05 23:15:15
>>974
5000時間くらい。

976:デフォルトの名無しさん
07/11/06 00:19:12
すいません、質問させてください。
rand関数とか、abs関数とか、stdlib.hで宣言されてるってことを聞きました。
で、stdlib.hを探してみて実際に見たところ、確かに宣言されてたんですが、
これら関数の実体って、どのファイルに定義されているんでしょうか。
もしくは、ソースファイルはついてないんでしょうか?(当方Mac OSXです)
見てみたいな、と思いまして・・

977:デフォルトの名無しさん
07/11/06 00:27:18
>>976
そういう関数はライブラリに中身がある
ヘッダーには呼び出しかたが書いてある。
実体を見たいなら、使ってるコンパイラメーカーに聞くしかない。

978:976
07/11/06 00:31:18
>>977
早速のお返事ありがとうございました!了解です。

979:デフォルトの名無しさん
07/11/06 00:33:50
scanfを使って整数型の変数に1〜100の値を入力された時のみ
続く処理をして、それ以外は再度入力を促すプログラムを作りたいです。

入力された値のエラーのチェックにはどんな処理が必要か
わからないので教えてください。

あと数値を入力してEnterを押して実行したあとに
前回入力した数値が残ったままになっているのを
消すにはどうすればいいですか?

980:デフォルトの名無しさん
07/11/06 01:14:14
>>979
#include <stdio.h>
int main()
{
int num;
char buff[12] = {'\0'};
do{
puts("1〜100の数値入力");
fgets(buff,sizeof(buff),stdin);
sscanf(buff,"%d",&num);

}while(1 >= num || num >=100);

/*
1以上100以下の数値が入力されたときの
処理をここに書けば〜
*/
return 0;
}

981:980
07/11/06 01:35:49
訂正
while(1 >= num || num >=100);
       ↓
while(1 > num || num >100);

982:デフォルトの名無しさん
07/11/06 01:58:33
scanfっつてんだから、scanfを使うべきなんじゃなくて?

俺ならこう書くぞ
} while (!(1 <= num && num <= 100));

983:デフォルトの名無しさん
07/11/06 02:02:57
論理演算が苦手そうに見えてもいいならそれでいいんじゃね?

984:デフォルトの名無しさん
07/11/06 02:30:32
すいません。
!についての質問なんですが、

if ( !(fp = fopen("a.c","r")) )

もしこれでfpがNULLでない場合(どこかのメモリアドレスつまり正の整数値)、
if文が偽(つまりNULLつまり0)となると思うのですが、
何故、!(正の整数値)がNULLつまり0になるのですか??ANSIのそういう仕様ですか?


if ( !5 )は偽でNULLつまり0??

(ちなみに、現在ほとんどのstdio.hで #define NULL 0 となっている)

985:デフォルトの名無しさん
07/11/06 02:33:54
!5といえば 1も2も3も・・・負数でさえ含まれますよね。0ももちろん含まれますが

986:デフォルトの名無しさん
07/11/06 02:47:08
#include <stdio.h>

int main(void)
{
FILE *fp;
if ( fp = (FILE*)(NULL == NULL))
;
printf("fp = %p\n", fp);
printf("(int)fp = %d\n", (int)fp);
printf("!5 = %p\n", !5);
return 0;
}




結果
fp = 0x1
(int)fp = 1
!5 = 0x0

987:デフォルトの名無しさん
07/11/06 02:58:34
NULL == NULLは真だから1が返るな

988:デフォルトの名無しさん
07/11/06 05:52:39
>>985
!5は「5以外」じゃないよ。
5を真偽値として評価して真、その否定で結果偽。そんだけ。

989:デフォルトの名無しさん
07/11/06 09:23:16
5以外は !=5 だわな。

990:デフォルトの名無しさん
07/11/06 22:06:26
レビューってどうやってやってる・・・?
なんか自分で書いたソースなのに、うまく説明できなくて凹みまくってる・・・。
頭の中じゃどんな風に動くのかわかってるんだけど、説明となると10あるうち2しか伝える事ができない・・・

991:デフォルトの名無しさん
07/11/06 22:41:32
君が超天才ならしょうがないけど、
凡人なら説明できない程度の理解としか周囲には写らないだろうよ。

誰にでも納得できる説明をする必要があるの?
仕事でレビュア対象に説明するだけなら、レビュアに疑問点・不明点を上げてもらって
それに回答することで納得して貰うのじゃダメ?

レビューの経験を積めば、レビュアの傾向が見えて対策も打てるけど、
例え経験不足だとしてもレビューの前に事前にレビュアと調整しておくって手もある。
凹む前にもっと考えて行動すべきだと思うけどね。

992:979
07/11/06 22:43:34
>>980 >>981 >>982
色んなサンプルソース読んでみても
scanfはあまり使われてないみたいなので
gets,fgetsに変更してみようとおもいます。

なぜなのかは、ぐーぐるさんに聞いてきます
ありがとうございました

993:デフォルトの名無しさん
07/11/06 22:55:21
>>991
あ、えーと、
現状まだまだ勉強の段階なんですが、
int main(void) じゃあそのvoidってナニ?なんでここにintとかおいてるの?っていう
いわば基礎中の基礎の部分なので、プログラムに一切関係のない人でも理解できるように発言して欲しい=学べる
ということなのでやってるんですが、経験をとにかく詰むしかないんですかね・・・うーん。

994:デフォルトの名無しさん
07/11/06 23:20:34
自分で書いたソースを客観的に見ることができないなら、他人に見て貰うといい。
相談できる人が身近にいればその人に協力して貰うのがいい。
相談できる人がいない、探す気もないならここにソース晒してコメントを貰うのでもいいんじゃね?

995:デフォルトの名無しさん
07/11/06 23:50:36
少しお尋ねしたいのですが、
struct _TEST a
{
int num;
char ch;
}test[100] = {0};

こうした場合、test配列の中身が全て0クリアされる動作は保証されていますか?
testが配列でない場合は動作が保証されているとどこかで伺った気はしたのですが

996:デフォルトの名無しさん
07/11/07 00:10:58
保証されている。ついでに言えばtest[100] = { 1 };として場合であっても
1が入るのはtest[0].numだけであって、残りは全て0になる。

997:デフォルトの名無しさん
07/11/07 01:06:24
>>996
ありがとうございます!

998:デフォルトの名無しさん
07/11/07 04:38:52
URLリンク(www.geocities.jp)

こちらのプログラムをVS2005でそのまま実行しても'fopen'の宣言を確認してくださいと出ます。
同HP (URLリンク(www.geocities.jp)) にあるc1.wavやらを読み込ませて実行したいのです。
fopenで配列を読んでいるのですが、ここにどうやってc1.wavなどを読み込ませればいいのでしょうか?

999:デフォルトの名無しさん
07/11/07 05:12:45
調べてたらコマンドラインで実行すらしらなかったのでなんとかなるかもしれないので失礼します。すみません。

1000:デフォルトの名無しさん
07/11/07 05:29:05
1000げっつ

1001:1001
Over 1000 Thread
このスレッドは1000を超えました。
もう書けないので、新しいスレッドを立ててくださいです。。。


最新レス表示
スレッドの検索
類似スレ一覧
話題のニュース
おまかせリスト
▼オプションを表示
暇つぶし2ch

5157日前に更新/251 KB
担当:undef