🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

Q&A

解決済

2回答

2930閲覧

C言語でxlsx形式の読み書き

otoma_yim

総合スコア6

C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

0グッド

0クリップ

投稿2020/01/04 14:54

初歩的な質問となりますが、よろしくお願いいたします。

前提・実現したいこと

C言語を用いてxlsx形式ファイルのコピーをしたいです。

発生している問題・エラーメッセージ

バイナリで読み書きして生成されたコピーファイルを開くとエラーになります。

'cpy.xlsx'の一部の内容に問題が見つかりました。可能な限り内容を回復しますか? ブックの発行元が信頼できる場合は、[はい]をクリックしてください。

[はい]を選択すると修復され、コピー元の内容も修復されます。

該当のソースコード

C言語

1#include <stdio.h> 2#include <stdlib.h> 3#include <string.h> 4void main() { 5FILE *rp, *wp; 6char str[256]; 7 8 *str = '\0'; 9 rp = fopen("test.xlsx", "rb"); 10 wp = fopen("cpy.xlsx", "wb"); 11 if ( rp == NULL || wp == NULL) { 12 fclose(rp); 13 fclose(wp); 14 return; 15 } 16 17 while (1) { 18 if ((fread(str, 1, 256, rp)) == 0 ) { 19 break; 20 } 21 fwrite(str, 1, 256, wp); 22 } 23 24 fclose(rp); 25 fclose(wp); 26 27 return; 28}

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

fread関数の戻り値は何を表しているのか、調べてみてはどうでしょう

投稿2020/01/04 23:39

y_waiwai

総合スコア88038

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

otoma_yim

2020/01/05 12:25

ご返答ありがとうございます。 正常時:読み込んだデータ個数 ファイル終了時及びエラー時: 読み込みデータ数より小さい値 ということでしょうか。 これをもとにwhile文のbreak条件を見直してみましたが結果が変わりませんでした。
y_waiwai

2020/01/05 12:48

255以下の数字が返ってきたときはどういう動作をするんでしょうか。
otoma_yim

2020/01/06 01:17

解決いたしました。 どうしようもない質問に回答頂きありがとうございました。 255以下の数字が返ってきた場合でも if ((fread(str, 1, 256, rp)) == 0) {break} としているので問題なくfwriteされるとだけ考えていましたが、そこではなく255以下の場合のゴミもfwriteされることが問題だったのですね。
guest

0

ベストアンサー

両ファイルサイズを比較してみてください.
読んだバイト数分だけを書き込むようにしなくては, ちゃんとコピーしたことにはならないでしょう.

投稿2020/01/04 15:06

jimbe

総合スコア13202

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

otoma_yim

2020/01/05 11:58

ご返答ありがとうございます。 一応ファイルサイズは同じで、修復すると内容もコピー元と一致しています。 ですが、開く度に本エラーメッセージが出てしまいます。
jimbe

2020/01/05 12:59

> ファイルサイズは同じ 1バイト単位まで同じなのでしょうか. y_waiwai さんが指摘されていますが, fread は必ずしも指定したバイト数分 str に読み込むわけではありません. そのため, 関数値として読み込んだバイト数を返し, str のそのバイト数以降は直前のままで有効な値ではありません. 例えば 100 バイトのファイルを fread(str, 1, 256, rp) として読んだ場合, うまくいっても 100 バイトしか読み込めませんので, str[0]~str[99] までにデータが入り, str[100]以降は有効では無く, fread の戻り値が 100 となります. その状態で fwrite(str, 1, 256, wp) とすると, 先頭 100 バイトがfread したデータ, 101から後が不定(0?) な 256 バイトのファイルが出来ると思います. そのため, ファイルサイズをバイト単位で確認し, ファイルサイズが異なり, 元のファイルが256バイト倍数で無いのにコピーしたファイルが256バイト単位になっていれば, 余計なデータが後ろについていると想像されます.
otoma_yim

2020/01/06 01:22

解決いたしました。 本当にどうしようもない質問に丁寧に回答頂きましてありがとうございました。 ご指摘のとおり255以下の読込時にゴミを含んだ状態でfwriteしていたため、コピーに失敗していました。 while (1) { rc = fread(str, 1, 256, rp); if (rc == 0) { break; } fwrite(str, 1, rc, wp); } としてfreadの戻り値をチェックし、fwriteのサイズ指定に使用することで解決しました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問