みんなの「教えて(疑問・質問)」にみんなで「答える」Q&Aコミュニティ

こんにちはゲストさん。会員登録(無料)して質問・回答してみよう!

解決済みの質問

Fortranの'read'のVBへの書き換え方

Fortranのread/writeをVBへの書き換え方を教えて下さい。

他の方の作られたfortran.90のprogramをvisual basicに書きなおす作業をしています。その中の下記のfileのaccessが上手出来ません。このなかに有るbinary,とかbig_endianが原因のようです。
(これの無いfile accessの箇所は問題なく出来ました。)

program内でdataをfileに書込み保存して蓄積しているようで、後日このdataをreadして使うようになっていますので新しくfileを作り直すわけにもいきません。

いろいろとcodeを入れてみたが、結果は文字化けばかりで10日も費してしまいました。

binary,bigendianなどの理屈が判っていないのでどうしょうもないと感じました。
どうか下記のfortranをVisual Basicへの書き変え方をご教授お願いします。

REAL ShipDim(17)
'読み込みの時
  filename = 'c:\pbcal\pbcal\pushdata\'//'push04'
open (3,file= filename, status='old', &
& form='BINARY', convert='BIG_ENDIAN',err=930)
DO 10 J= 1,17
read(3) shipDim(j)
10 CONTINUE
     (中略)  '見やすいように簡略化しています。
CLOSE (3,STATUS='KEEP')
 
'書込みの時はopenは下記で、後readをwriteに変えただけです。
open (3,file= filename, status=pstatus,form='BINARY', convert='BIG_ENDIAN',err=940)

VBでTRYしたことの一例
    Dim dat() As Byte = New Byte(-1) {}
  Dim sr As System.IO.Stream = Nothing
  Dim br As System.IO.BinaryReader = Nothing
Dim strf As String = filepath & "push04" 'filepathは適当にしている
   sr = System.IO.File.Open(strf, _
       System.IO.FileMode.Open, System.IO.FileAccess.Read)
br = New System.IO.BinaryReader(sr)
ReDim dat(sr.Length - 1)
dat = br.ReadBytes(sr.Length)

     dat に=572個の2-3桁の数字が入っています。encodingなどでためしてみたが文字化け
    ばかりです。
   (System.Text.Encoding.GetEncoding("****").GetString(dat) など)
     
  どうかよろしくお願いします。

投稿日時 - 2014-06-24 15:46:25

QNo.8651601

困ってます

質問者が選んだベストアンサー

FORTRANのOPEN命令の引数の意味が判らないのであれば、先ずそのFORTRANの取説あるいは文法書を見ないことには始まらない。
昔使っていたFORTRAN(VAX FORTRAN)には「form='BINARY'」や「convert='BIG_ENDIAN'」といったオプションはなかったので、
正確には判らないが、私の経験から

◎FORTRAN側
(1) FORTRANでのREAL変数は、4バイト実数です。(8バイト、又は16バイトの可能性もある)
(2) ファイルを'BINARY'指定でOPENしているので、ファイルに書き込まれているデータはバイナリ形式(メモリ内部形式)と推測できる。
(3) ファイルを'BIG_ENDIAN'指定でOPENしているので、ファイルに書き込まれているバイナリデータはビッグエンディアン形式と推測できる。
※ これは非常に厄介な状態です。なぜならWindows系の言語ではリトルエンディアンが普通なので、バイトの順番を入れ替えて読まないといけない。

◎Windows側
(1) バイナリファイルなのでエンコードは関係ない。(エンコードはテキストファイルの時のみ使う)
(2) ファイルから1変数分の4バイトをByte配列に読込む。(FORTRANのREALのサイズにより8または16の可能性あり)
(3) 読みこんだ4バイトの順番を入れ替える。
(4) 4バイトのByte配列を実数に変換する。
(5) 2~4をデータの個数分繰り返す。

※ ただし、FORTRANのREALのデータ形式と、VB.NETのSingleのデータ形式が同じとは限らないので正しく変換できないかもしれない。
※ また、ファイルサイスが572バイトだと、読み込もうとしているデータとサイズが合わないのも気になる。

サンプル(ファイル名等は合わせてください)
  Dim fs As New FileStream("c:\test.bin", FileMode.Open, FileAccess.Read)
  Dim buf(3) As Byte ' データ格納用配列(REALが8バイトならbuf(7)にする)
  Dim fWork As Single ' 実数データ(REALが8バイトならDoubleにする)
  Dim readSize As Integer ' Readメソッドで読み込んだバイト数
  Dim i As Integer

  For i = 0 To 16
    readSize = fs.Read(buf, 0, buf.Length)
    Array.Reverse(buf)
    fWork = BitConverter.ToSingle(buf, 0)
    Console.WriteLine(fWork.ToString())
  Next i
  fs.Close()

投稿日時 - 2014-06-24 21:17:24

お礼

ありがとうございました。ご回答通りにcodingして文字化けしない数字が出た時は感激しました。(10日もあれやこれやとやっていたことが馬鹿みたいに思えました。) これに倣いwriteのほうもcodingしtestまで出来ました。
なを572bytesですが長くなるので、後のread 126回を省略していました。17+126=143 143*4=572で合いました。(余計な気がかりをさしてすみませんでした)
助かりました。 深謝!!!

投稿日時 - 2014-06-25 20:54:34

このQ&Aは役に立ちましたか?

10人が「このQ&Aが役に立った」と投票しています

回答(2)

ANo.1

変換はこちら
http://www.atmarkit.co.jp/fdotnet/dotnettips/045getbytes/getbytes.html
バイト列は、"1""2""3"と言った「文字」で表現しているわではないので、テキストで変換しても無駄です。

投稿日時 - 2014-06-24 20:27:38

あなたにオススメの質問