C言語なら俺に聞け( ..
[2ch|▼Menu]
701:デフォルトの名無しさん
07/04/10 23:41:08
>>689
それはつまり1人用の麻雀ゲームを作ろうというわけですね
それなら簡単なGUIと思考ルーチンが必要です
思考ルーチンは現在の状況, つまり引いた牌と持ち牌(っていうのか?もれは麻雀やらないから言い方違うかも)
と相手の打った牌などから次にどの牌を打つかを決める一連の思考の手順のことです
これは一般的に一種類あるいは数種類の思考パターンを用意し, これらのような決まった処理のことを
アルゴリズムと言う
アルゴリズム(思考手順)そのものは言語に依存しないので(もちろん言語によって書き方は異なるかもしれないけど)
すでに知っている言語があったら新たにCとかC++を勉強しなくてもOK
思考手順そのものを研究するのが目的ならVBやJavaなどの簡単にGUIが作れる言語が(個人的には)おすすめ
これを足がかりにほかのゲームや実用ソフトを作りたいなら将来にわたって使っていけるような言語を選ぶのもあり(CとかC++)
もしLinuxなどのUNIX系OSや, プラットフォーム非依存をターゲットとするならJavaが一番手っ取り早い
GTKやwxWidgetなどのクロスプラットフォームのツールキットを使うこともできる
(こういったツールキットは大抵複数の言語に対応している)
あるいはPythonなどのスクリプト言語を使ってみるのもあり

でも, ひとこと言わせてもらうと
> 自分の打ち方がどれぐらい強いのか試そうと思って
の意図はいまいちよくわからないが
作ったプログラムと自分で対戦すると言うなら
それは一人二役でゲームをするのと同じだからちょっと無意味っぽいと(個人的に)思う
また自分の思考をプログラミングしてほかの人やプログラムと戦わせたいのなら相当の覚悟が必要
人の思考は想像以上に複雑で, 無意識や勘なども作用してくるから
完全に自分の思考回路をプログラミングすることは(現実には)不可能に近い
したがって出来上がったプログラムは君自身より弱い
また, 何手か先を読むような(ありふれた)機能を盛り込む場合でも
計算しなければいけない手数は爆発的に増えるから(計算量とメモリーの増加), それをいかにして押さえるかとか
いろいろとテクニックも必要になってくる

ここまで考えるとかなり手間がかかる
もれは大学の卒論がチェスのゲーム理論だったからよくわかる

702:デフォルトの名無しさん
07/04/10 23:44:16
>>699
do
{
    if( !A )
    {
        if( !B )
        {
            break ;
        }
    }
    C ;
}while(0);

703:デフォルトの名無しさん
07/04/10 23:48:24
>>699
なぜそうするのかはわからんが
単純にフラグをつけるのもいいんじゃない?
Aが真ならflag = TRUE;してBのところでflag == TRUE;ならCを実行しないとか?

704:デフォルトの名無しさん
07/04/10 23:49:25
>>699
if( A )
{
  C ;
}
else if( B )
{
  C ;
}

これじゃダメなのか?

705:704
07/04/10 23:52:50
>>699
あー1回だけの記述か


706:デフォルトの名無しさん
07/04/10 23:54:18
>>699
if(A) {
 goto l;
} else if(B) {
 l: C;
}

707:デフォルトの名無しさん
07/04/11 00:01:42
>>699
if(A) goto label1;
if(B) goto label1;
goto label2;
label1: C;
label2: ;

708:デフォルトの名無しさん
07/04/11 00:18:16
>>699人気だなww

709:デフォルトの名無しさん
07/04/11 00:30:03
で、本当にこれ宿題なのか?
だとしたらこんな腐った問題を出す教師ってどんなやつなんだ。

710:デフォルトの名無しさん
07/04/11 00:35:24
Q.腐った問題を出す教師自体が腐ってる確率を求めるプログラムを作りなさい

711:デフォルトの名無しさん
07/04/11 00:40:44
>>710
int main()
{
printf("腐った問題を出す教師自体が腐ってる確率=%d%%",100);
return 0;
}

712:デフォルトの名無しさん
07/04/11 01:01:48
>>701
えっとですね、あんまり必要ないかと思って省いてしまったんですけど。
東風荘っていうネット雀荘みたいなものがあるんですけど、そのゲームの画面の状況から変数を作るdllを見つけてですね。
上がりや聴牌の判断もそのdllの関数がやってくれるので私はその変数の扱いを設定するだけで麻雀の代うちソフトが作れるようになるわけなんです。
だから、相手は人だし麻雀というゲームを遊ぶための土台の部分は配布されているので単純にアルゴリズムを考えるだけでOKってことなんですよね。

テクニックとしては、プログラムのテクニックよりは麻雀のテクニックのほうが重要だとおもうので簡単という言葉を使わせてもらってます。

713:デフォルトの名無しさん
07/04/11 03:44:18
そんな面白そうなものがあるのか。
ちょっと探してこよう……と思ったら一発で見つかった。

714:701
07/04/11 05:22:28
>>712
ほほー便利だなー
それなら自分がどんな風に思考しているのかと
どうやってそれをアルゴリズムにするのかを悩むだけでいいのかw


715:デフォルトの名無しさん
07/04/11 07:56:24
if (A) {
if (B) {
C;
}
}


716:デフォルトの名無しさん
07/04/11 08:00:29
>>715
っ A==1 B==0

717:デフォルトの名無しさん
07/04/11 08:07:36
>>715
おちつけ
それは、A||Bは(又は)ではなくてA&&B(喝)だ

718:デフォルトの名無しさん
07/04/11 08:07:36
実際こういう場合はどうしたらいいの?
素直に判別式に論理演算使えってこと?

719:デフォルトの名無しさん
07/04/11 08:17:18
switch (x) {
case A:
case B:
C;
}

720:デフォルトの名無しさん
07/04/11 08:22:28
逆に論理演算を使いたくない理由ってのがわからない

721:デフォルトの名無しさん
07/04/11 08:22:53
というかその為に論理演算子があるんだから使ってやれよ…
論理演算子が泣いてるぞ。

722:デフォルトの名無しさん
07/04/11 08:23:43
>>719
なんじゃそりゃーーー!(松田優作風に

723:デフォルトの名無しさん
07/04/11 08:23:56
AまたはBならCをする。
=> AでないかつBでないならCはしない。

というド・モルガンの法則を理解するための課題であった。

なわけはないか。


724:デフォルトの名無しさん
07/04/11 09:22:50
>>719
caseは定数しかとらないので
A, Bのような式はだめ

725:デフォルトの名無しさん
07/04/11 09:23:38
>>699
void f(void){
 C;
}
int main(void){
 if(A){
  f();
 }else if(B){
  f();
 }
}

726:デフォルトの名無しさん
07/04/11 09:28:07
#define SHE C
void main(void){
 if(A){
  SHE;
 }else if(B){
  SHE;
 }
}

727:デフォルトの名無しさん
07/04/11 09:29:03
↑訂正
×void main(void){
○int main(void){

728:デフォルトの名無しさん
07/04/11 09:52:06
>>723
setjmp&longjmp関数の有用性を教えるための課題

729:デフォルトの名無しさん
07/04/11 10:08:09
invalid digit "8" in octal constant
invalid digit "9" in octal constant
このコンパイルエラーって何ですか?教えてください!

730:デフォルトの名無しさん
07/04/11 10:15:07
>>729
8進数なんだから0〜7までで表せよ、
8や9なんてねーよバーカ

っていうエラー。整数の頭に0をつけてるだろ。

731:デフォルトの名無しさん
07/04/11 10:15:55
>>729
#include<stdio.h>
int main(void){
printf("%d\n", 012); /* 数値の頭に 0 が付くと 8進数 */
printf("%d\n", 12);
return 0;
}

732:729
07/04/11 10:22:40
>730さん
>731さん
ありがとうございます!
参考になりました。

733:695
07/04/11 10:35:06
整数やら文字列が入った構造体を、ひとつの文字列にするいい方法はないですか?><

734:デフォルトの名無しさん
07/04/11 10:44:08
>>733
関数を作る

735:733
07/04/11 10:48:45
整数やら文字列が入った構造体を、ひとつの文字列にする関数を作りたいのですが、いい方法はないですか?><

736:デフォルトの名無しさん
07/04/11 10:51:23
>>735
構造体晒せ

737:デフォルトの名無しさん
07/04/11 10:52:16
いつまでこのアホタレを甘やかす気ですか?><

738:735
07/04/11 11:02:14
struct test {
 int i;
 char c[10];
  struct kouzoutai {
  int seisuu;
  double kazu;
  int suu;
  char moji[10];
 }kou;
};

739:デフォルトの名無しさん
07/04/11 11:04:39
>>738
釣り乙
死ね

740:735
07/04/11 11:04:41
全部snprintfとかstrcatで繋ぐのはめんどくさいです
要するにバイト列と文字列を相互変換できる方法が知りたいのです><

741:735
07/04/11 11:05:18
自己解決しました><

742:753
07/04/11 11:06:53
爆釣ありがとうございました><

743:740
07/04/11 11:07:14
>>741
してないです><
とりあえず16進数でずらずらと数字を並べてみることにするです><

744:743
07/04/11 11:08:48
できました><

745:743
07/04/11 11:21:14
>>744
まだやってません><

ところで、
sprintf(text, "ABC");

sscanf("ABC", "%s", text);
って
strcpy(text, "ABC");
と同じですか?

746:743
07/04/11 11:39:57
隣の後輩が教えてくれました><
ありがとうございました><

747:745
07/04/11 11:44:37
>>746
><

なんとか自分でやってみます。ありがとうございませんでした><

748:デフォルトの名無しさん
07/04/11 12:20:11
カオスすぎです><

749:747
07/04/11 12:26:13
初心者なのですいません><

750:749
07/04/11 14:57:11
構造体へのポインタをunsigned char*型にキャストして
1バイトずつ2桁の16進数に直したら
構造体と文字列の相互変換ができました><
unsigned char *p;
unsigned char text[1024], buf[1024];

strcmp(text, "");
p = (char *)&test1.kou;
for(i=0; i<sizeof(test1.kou); i++)
{
 sprintf(buf, "%02x", (unsigned char)p[i]);
 strncat(text, buf, 2);
}

751:749
07/04/11 14:59:26
p = (char *)&test2.kou;
for(i=0; i<sizeof(test2.kou); i++)
{
 strncpy(buf, &text[i*2], 2);
 sscanf(buf, "%02x", &((unsigned char)p[i]));
}

752:749
07/04/11 15:00:31
以上オナニーでしたありがとうございました><

753:749
07/04/11 15:01:26
また初心者を装った釣りに来ますのでよろしくです><

754:デフォルトの名無しさん
07/04/11 15:02:25
もうこなくていいよ

755:751
07/04/11 15:04:22
>>752-754
ほんとに初心者なのにひどいです><

756:735
07/04/11 15:06:39
すみませんなんか偽物がレスして話進んでるんですけど><
>>738から全部偽物です><

757:755
07/04/11 15:09:48
なら>>756も偽者ですね><

758:デフォルトの名無しさん
07/04/11 17:04:36
>>735
1バイトづつ16進2桁の文字列にしてしまうとか、あるいはコントロールコード
などの表示不能な文字はエスケープして表示可能な文字にして、表示可能な
文字はそのままにするとか、あるいは表示する必要が全くないのなら気にせず
全部 write() や fwrite() で出力してしまうとか、方法は色々あります。


759:757
07/04/11 17:43:19
>>758
>表示不能な文字をエスケープして表示可能な文字にして
文字列←→構造体で相互に変換したいので、文字列から戻す時に
もともと表示可能な文字だったのかエスケープして表示可能になった文字なのか
分からなくなってしまうと思うのでこの場合は使えないです><すいません

ソケットプログラムでsendするときunsigned char型を使うので、
構造体を文字列にしたかったのですが後は何とか自分でがんばってみるです><

760:デフォルトの名無しさん
07/04/11 18:01:36
バイナリを可視なASCII列にするにゃ Base64 エンコが手っ取り早いような気がしないでもない

が、構造体の中身をそのまんまエンコして送るのも抵抗があるな。
受け側のアライメントとエンディアン、基本型のサイズの全てが一致してないといけないしね。

761:デフォルトの名無しさん
07/04/11 18:01:41
>>759
エンコードした文字列は当然、デコードできる。
例えば不可読文字に遭遇したら%に引き続く二文字を16進の文字コードになるようエンコードする。
デコードするときは当然、その逆をやればいい。注意すべきは、%に遭遇した場合もエンコードしておく必要があることだけだ。
Ex.
source: SOH "abc%" LF "def" ETX 20(4バイト整数) EOT
encoded: %01abc%25%0adef%03%00%00%00%14%04


762:759
07/04/11 18:06:10
>>760-761
ありがとうございます><すごく勉強になります
%をつける方法だと文字列の長さが一定にならないから送受信のとき大変そうですけど
なんとか考えてやってみます

763:デフォルトの名無しさん
07/04/11 18:11:31
>>762
ソケットの送受信は長さ決め打ちして読むと泣きを見るわけだが…
プロトコルを自前で規定してるか?

# fgets 等を使わずに fread と fwrite で同じことを記述できるようにしてから、
# ソケットへ移行したほうが良いと思うぞ。

764:デフォルトの名無しさん
07/04/11 18:12:28
バイナリ値主体なら>760、テキスト主体なら>761だな。
#どちらも定番だね。

765:762
07/04/11 18:43:32
つい最近ソケットプログラムとかを勉強し始めたばかりなのでわからないことばかりで…。
とりあえずBase64というのをやってみることにします><;すいませんでした。

766:765
07/04/11 18:58:09
今は文字列と数値を全部snprintfでそのまま繋げてやっていますが
構造体内の変数名とかを全部書かないといけないので
ソースが汚いうえ無駄が多いです><

767:デフォルトの名無しさん
07/04/11 18:59:58
どういう風にやっているのかは知らないが、
そうやって単なる文字列にするのも一般的だと思うぞ。

768:デフォルトの名無しさん
07/04/11 19:03:30
構造体のシリアライズは一大テーマで、
ここで扱うには余白が狭すぎるとは思うんだよなあw

769:デフォルトの名無しさん
07/04/11 20:11:51
>>713-714
亀ですけど、もし興味わいたなら最強の麻雀のアルゴリズム作ろうぜスレで一緒に最強のアルゴリズムつくりませんか?


770:デフォルトの名無しさん
07/04/11 21:12:05
亀がしゃべった!

771:デフォルトの名無しさん
07/04/11 21:34:43
C始めて3日目の俺がきましたよ。
研修でダンプ出力するPGを作成する課題やってるんだけど、

ファイル開く→開いたファイルから16バイト読み込む
→読み込んだデータの16進数変換を行いHEXに格納する

までは出来たんだが、その後の置換処理がさっぱり分からん('A`)

置換処理はHEXに格納されたデータの表示できないコードを
ピリオドに置換してCHARに格納するんだが、検討もつかないし_| ̄|○

助けてくだしあ

772:デフォルトの名無しさん
07/04/11 21:40:26
>>771
isprint関数を使う

isprint(c)? c: '.';

773:デフォルトの名無しさん
07/04/11 21:43:06
>>770
亀レスって言葉をご存知?

774:デフォルトの名無しさん
07/04/11 21:44:03
>HEXに格納された
意味が不明すぎる
char配列に16進数の文字を入れてるって事?数値?

775:デフォルトの名無しさん
07/04/11 21:50:02
>>773
く、くまー
ネタにマジレ(ry

776:771
07/04/11 22:20:05
参考書片手に悩んでた。

>>772
レストンクス
ちと考えてみるわ。

>>774
HEXには
sprintf(HEX,"%X",ss)
でss内の文字列を16進表示で入れたんだが。
おそらく数値?

777:デフォルトの名無しさん
07/04/11 22:25:04
>>768
浮動小数点数を有効桁いっぱいいっぱいまで伝達しようとすると悩ましいね

778:714
07/04/11 23:14:51
>>769
いやオレはもうチェスで懲り懲りだよww
プログラム部はだいたい解決策が決まってるけど
思考の部分は定石と言えるような方法がない
(まあ思考とプログラムも明確に分けれるもんじゃないけどね)
これ以上はスレ違いなんでそのスレの住民と一緒にがんばれ!

779:771
07/04/11 23:39:22
>>774
自分でも問題の意味を分かってなかったらしい。すまぬ。

HEXDATA          CHARADATA
| ̄| ̄| ̄| ̄| ̄| ̄| ̄| ̄| | ̄| ̄| ̄| ̄| ̄| ̄| ̄| ̄|
*|54|45|53|54| | | | | *|T|E|S|T|. |. |. |. |
 ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄  ̄
見づらくてスマヌ('A`)
こんな感じで格納したいわけで。

置換処理でisprint使うっていうのは大体分かったけど、for文で回してなにかしろ
みたいなことを講師が言っていた希ガス。何回したらいいの(;´Д`)??


780:771
07/04/11 23:40:09
なんてこったい。
ステキにずれてらっしゃる_| ̄|○

781:デフォルトの名無しさん
07/04/11 23:43:15
数値をCHARADATAにそのまま書けばいいじゃん

782:デフォルトの名無しさん
07/04/11 23:52:39
回すものといえば皿ぐらいしか

783:771
07/04/12 00:06:20
レストンクス。
>>781
CHARADATAに書き込むってisprint使ってですかい?

>>782
俺の頭の中で星が回ってるよ。

784:デフォルトの名無しさん
07/04/12 00:10:22
>>783
isprintで表示できないコードを判別して表示できないのならピリオドを、
表示できるのなら、そのコードを代入

785:771
07/04/12 00:15:09
>>783
なるほど分かってきた。
isprintで表示できるって判断したときに返す値って何?
表示できないってときは多分0返してくるんだよね?

786:771
07/04/12 00:17:48
784の安価ミス

787:デフォルトの名無しさん
07/04/12 00:42:50
>>785
そこまでヒントもらってるんだからマニュアルみなよ。ぐぐってもいいしさ。

788:771
07/04/12 00:49:03
ある程度分かったから参考書片手に頑張ってみるわ。
おまいらありがとう

789:デフォルトの名無しさん
07/04/12 08:32:40
>>785
Cの真偽値の基本は、0でない=真、0=偽
真の場合、0でない値として必ず1を返すケースがいくつか定義されているが、
明言されていない場合は、0でないことだけが保証される。


790:デフォルトの名無しさん
07/04/12 11:41:47
なんで*(s+1)とs[1]の二つの書き方をつくったのよ>K&R
*s++は*にしかできない書き方だし
*(*(s+1)+2)は読みにくいからs[1][2]と書くんだけど
なんか一貫性に欠けてるぞオイ
なんでアメ人は人工言語にこんな汚い要素を持たせるのよ
完璧主義で潔癖症なオレはポインタと配列を使うとき、毎回迷うじゃん

791:デフォルトの名無しさん
07/04/12 11:53:05
>>790
*を使わなければいい。
*s++なんてロジックは却って最適化を阻害しかねない。
s[offset++]でいいジャマイカ。

792:デフォルトの名無しさん
07/04/12 11:56:03
読み込むファイルにひらがなあったら駄目だけどね
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#define BUF_SIZE 16
int main()
{
FILE *fp;
char filename[1024] = {'\0'};
char buff[BUF_SIZE] = {'\0'};
char *temp = NULL;
size_t len = BUF_SIZE;
fgets(filename,sizeof(filename),stdin);
if(temp = strchr(filename, '\n'))
*temp = '\0';
fp = fopen(filename, "r+b");
while(len == BUF_SIZE){
unsigned int i;
len = fread(buff, 1,BUF_SIZE , fp);
for(i = 0; i < len; i++)
printf("%02x ",(unsigned char)buff[i]);
for(; i < BUF_SIZE; i++)
printf("-- ");
for(i = 0; i < len; i++){
int c = buff[i];
if(isspace(c)) c = ' ';
putchar(c);
} putchar('\n'); } fclose(fp);
return 0;
}

793:デフォルトの名無しさん
07/04/12 12:47:04
>>791
そうなんだけど
*s++ = *t++みたいな記述は確かに便利だし
s[i]は添え字の分のメモリーも必要だし
ちと読みにくくなるけど
*(*(s+1)+2)でいくか

794:デフォルトの名無しさん
07/04/12 13:01:03
>>793
>s[i]は添え字の分のメモリーも必要だし
ダウト。
今時のコンパイラはどっちで書いても同じコードを吐いてくれる(ことが期待できる)。

795:さいきち
07/04/12 13:05:57
教えてください。C言語を勉強し始めたひよっこです。
Cコンパイラ(Borland C++)をPCにインストールして、コンパイルしようとしたら
以下のようなエラーがでてウマく行きません。何が原因なのでしょうか?

エラー E2209 sample1.c 5: インクルードファイル 'stdio.h' をオープンできない
警告 W8065 sample1.c 10: プロトタイプ宣言のない関数 'printf' の呼び出し(関数 main )

10日でおぼえるC言語入門 という本を見て勉強を始めたのですが勉強を進めきれず
困っています。
どなたか教えてください。

796:デフォルトの名無しさん
07/04/12 13:06:10
* は直接参照する時しか使わないな。
インデックスを使うときは必ず [ ] を使う。
その方が読みやすい。

全てのループは if と goto で書けるが
そう書かないのと同じ事だ。
分かりやすく書くのが至上。
後の事は最近のカシコイコンパイラが最適化してくれる。

797:デフォルトの名無しさん
07/04/12 13:11:11
>>795
このあたりから setbcc とかを参考にしてみるといいかもしれない。
URLリンク(lacc.biz)

798:さいきち
07/04/12 13:17:18
大変申し訳ありません。795で質問しました さいきち です。
わたくしはルール違反をしてしまいました。過去ログを調べていません。調べて見ます。
795の質問は『無視』でお願いいたします。

799:デフォルトの名無しさん
07/04/12 15:49:17
>>798
つい最近まったく同じ質問があって、インストールしたbccがアップデート版だった
というのがあったな。

800:デフォルトの名無しさん
07/04/12 16:20:45
argv[1]とargv[1][0]ってどう違うのですか?
どちらも一つ目のオプションを指してると思うんですが
printf( "%p %p\n", argv[1], argv[1][0] );
で表示させると違う値になります

801:デフォルトの名無しさん
07/04/12 16:24:04
>>791>>794>>796
コンパイラがちゃんと最適化してくれるのか 知らなかった
これからは[]で統一する
やっぱ*(*(s+1)+2)なんて書かれたらウザいよねw

802:デフォルトの名無しさん
07/04/12 16:25:44
>>800
argv[1] はchar*
argv[1][0] は char

803:デフォルトの名無しさん
07/04/12 16:27:14
argv[0]は&argv[0][0]と同じってことですね

804:766
07/04/12 16:36:19
構造体のなかのintをcharに全部変えたらbase64の文字列が短くなって通信ができるようになりました
>>760さんありがとうございました。><

805:デフォルトの名無しさん
07/04/12 18:42:57
入力した数字が0ではない間ループするってやつで
int aho[100];
int i=0;
while(aho[i]!=0){
scanf("%d",&aho[i]);
i++;}
ってやると0入力してもおわらんのだけどなんでじゃ?


806:デフォルトの名無しさん
07/04/12 18:51:52
>>805
標準入力からaho[0]に数値が入力されてるのに、while文の判定ではaho[1]
見てるからじゃね?

807:デフォルトの名無しさん
07/04/12 18:54:45
>>805
「入力した数字」なのだから、入力前に検査しちゃダメだろう。
do {
scanf("%d", & aho[i]);
++i;
} while (aho[i - 1] != 0);
or
int tmp = 0;
do {
scanf("%d", & tmp);
aho[i] = tmp;
++i;
} while (tmp != 0);
いずれにしても、scanf()をそのまま使うのはお勧めできないが。

808:デフォルトの名無しさん
07/04/12 20:37:29
>>806>>807
ほんとだw
ありがとう

809:デフォルトの名無しさん
07/04/12 21:05:32
>>800
別に配列にしたからって、より最適化してくれるってわけでもないがな。


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

5385日前に更新/183 KB
担当:undef