アプリ版:「スタンプのみでお礼する」機能のリリースについて

結果テキストファイルから特定の値のみ抽出するプログラムを作成しているのですが、思うように動いてくれません。どなたか教えてくださいませんか。お願いします。

<テキストファイルの形式>
様々な文字や記号slm,0.070000,-53458.000000様々な文字や記号

これが1行に4つ程含まれるものが10行ほどあるのですが、
2つ目のカンマの後の数値部分のみ抽出したいのです。

<プログラム>

char line[MAXLINE];
char a1[]="slm";
char *r;
char *s[2];

while (fgets(line, MAXLINE, fp) != NULL){
if(strstr(line, a1)!=NULL){
for (r = line ;r = strstr(r, a1); r += 27 ){
for(int p=0;p>2;p++){
s[p]=strtok( strstr(r, a1), "," );
printf("%s",s[3]);
}

}
printf(" \n");
i++;
}

}


そこでこのようなプログラムを作ってみたのですが、実行すると何も表示されません。
どなたか改善策を教えてください。本当に困っています。

A 回答 (7件)

> こちらのdouble型のaとbは何のための配列なのでしょうか?



配列と書いたのは

> これが1行に4つ程含まれるものが10行ほどあるのですが、

とあったので、それら全部を記憶しておく必要があるなら、配列とか用意するように修正してください。という意味で、a, b 自体配列でもなんでもありません。あくまでサンプルコードとして提示してみただけで、こちらとしては数値に変換していいのかどうかも知りません。

ですから文字列として処理する必要があるなら、全然見当外れとなります。
    • good
    • 0

> char *s[3];


> strcpy(s[p],strtok( NULL, ",") );

これは値不定のポインタ配列 s[3] に、いきなり strtok() の返り値のアドレスからコピーしようということです。本当に内容をコピーして保存したいなら s[] のそれぞれに実体を与えてからじゃなきゃダメです。

それに strtok() 使うなら
char *q;
for (q = r; q = strtok(q, ","); q = NULL) {
みたいに 2 回目以降の呼び出しを NULL にしなければなりません。

ただ最終的に数値として取り込みたいなら
double a, b;
for (r = line ;r = strstr(r, a1); r += i) {
if (sscanf(r, "slm ,%lf ,%lf%n", &a, &b, &i) < 2)
i = sizeof(a1)-1;

あとは配列に追加してくなり、その場で加工するなり。

この回答への補足

回答ありがとうございます。
double a, b;
for (r = line ;r = strstr(r, a1); r += i) {
if (sscanf(r, "slm ,%lf ,%lf%n", &a, &b, &i) < 2)
i = sizeof(a1)-1;

こちらのdouble型のaとbは何のための配列なのでしょうか?
何度もすみませんがご教授願います。

補足日時:2011/12/16 02:04
    • good
    • 0

とりあえず 1点:


strtok の使い方を確認してください.

この回答への補足

回答ありがとうございます。勉強してみます。

補足日時:2011/12/16 02:03
    • good
    • 0

それから、配列の基本もおわかりではないようですね。



>char *s[2];

という定義で、正しく参照できるのは、s[0]とs[1]の「2個」です。
定義時の*s[2]は、0番から始まる「2個の領域」という意味です。
したがって、

>printf("%s",s[2]);

s[2]は定義範囲外の領域です。また、ループの中で毎回出力しているのは意図が不明です。

この回答への補足

詳しい回答を何度もありがとうございます!
ご指摘くださった部分を改良して

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

#define MAXLINE 2094

int main(void){
FILE *fp;
int i=1;
char line[MAXLINE];
char a1[]="SIL";
char a2[]="acoustic=";
char *r;
char *s[3];

/* ファイルオープン */
if ((fp = fopen( "C:\\Users\\石原慎也\\Desktop\\N_best.result", "r" )) == NULL){
printf("ファイルオープンに失敗しました。B\n");
exit(1);
}

/* データ処理 */
while (fgets(line, MAXLINE, fp) != NULL){
if(strstr(line, a1)!=NULL){
for (r = line ;r = strstr(r, a1); r += 27 ){
for(int p=1;p<=2;p++){
strcpy(s[p],strtok( NULL, ",") );
printf("%s",s[2]);
}


}
printf(" \n");
i++;
}

}



}


という風にしてみたんですが、プログラムが動きません・・。
他にどこか悪いところありますか?
何度もごめんなさい。

補足日時:2011/12/15 21:35
    • good
    • 0

>for(int p=1;p>2;p++){



いや、ですからね、質問者さんはfor文の継続条件を勘違いされていると思うんですよ。

p>2
というのは、「pが2より大きい間」という意味です。
「pが2より大きくなるまで」という意味「ではありません」。
    • good
    • 0

strstr()の引数が、反対です。

ファイルポインタはどこで定義されていますか? 一部ではなく、プログラム全体を掲載してほしいです。

この回答への補足

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

#define MAXLINE 2094

int main(void){
FILE *fp;
int i=1;
char line[MAXLINE];
char a1[]="slm";
char *r;
char *s[2];

/* ファイルオープン */
if ((fp = fopen( "ファイルへのダイレクトパス", "r" )) == NULL){
printf("ファイルオープンに失敗しました。B\n");
exit(1);
}

/* データ処理 */
while (fgets(line, MAXLINE, fp) != NULL){
if(strstr(line, a1)!=NULL){
for (r = line ;r = strstr(r, a1); r += 27 ){
strcpy(s[0],strtok( strstr(r, a1), ",") );
for(int p=1;p>2;p++){
strcpy(s[p],strtok( NULL, ",") );
printf("%s",s[2]);
}


}
printf(" \n");
i++;
}

}



}


回答ありがとうございます。こちらが全体のプログラムなんですが・・・。少々変えました。

補足日時:2011/12/15 19:30
    • good
    • 0

for文の中の不等号の向きは正しいですか?


s[3] は、配列の定義範囲外の領域です。アクセスして大丈夫ですか?

この回答への補足

回答ありがとうございます。
すいません。間違えていました。
ですが直してもまだできません・・・。

補足日時:2011/12/15 19:30
    • good
    • 0

お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!