怎样用Java读写二进制文件

怎样用Java读写二进制文件,第1张

import java.util.*

import java.io.*

class SmallFile {

static final int HEADLEN = 24//头总长度

byte[] fileName = new byte[16]//列表文件名1: 长度128 想把它读到char[]里 它的编码方式不是Unicode。在不确定编码方式的时候,最好直接用byte[]来存放

int offset//列表文件地址1: 长度32 想把它读到int里

int length = -1//列表文件长度1: 长度32 想把它读到int里

byte[] content

public SmallFile() {

}

public SmallFile(byte[] fn, byte[] content) {

Arrays.fill(fileName, (byte) 0)

if (fn != null) {

if (fn.length <= 16) {

System.arraycopy(fn, 0, fileName, 0, fn.length)

}

else {

System.arraycopy(fn, 0, fileName, 0, 16)

}

}

this.content = content

if (content != null) {

this.length = content.length

}

else {

this.length = -1

}

}

}

public class ReadBinary {

static final int HEADLEN = 8//头总长度

private String filename

private byte[] filehead = new byte[4]//文件头: 长度32 想把它读到char[]里 它的编码方式不是Unicode

private int filecount = -1//列表长度: 长度32 想把它读到int里 假设他是3 就会有3个列表文件名

private List<SmallFile>files = null

public void setFilehead(byte[] fh) {

if (fh == null)

return

Arrays.fill(filehead, (byte) 0)

if (fh.length <= 4) {

System.arraycopy(fh, 0, filehead, 0, fh.length)

}

else {

System.arraycopy(fh, 0, filehead, 0, 4)

}

}

public ReadBinary(String filename) {

try {

readFromFile(filename)

}

catch (Exception ex) {

System.out.println(ex.getMessage())

System.out.println("在载入数据文件时失败,因此视同为新建一个数据文件!")

this.filename = filename

Arrays.fill(filehead, (byte) 0)

filecount = 0

files = new ArrayList<SmallFile>()

}

}

public void readFromFile(String filename) throws Exception {

BufferedInputStream bin = new BufferedInputStream(new FileInputStream(

filename))

this.filename = filename

DataInputStream in = new DataInputStream(bin)

in.read(filehead)//文件头: 长度32 想把它读到char[]里 它的编码方式不是Unicode

filecount = in.readInt()//列表长度: 长度32 想把它读到int里 假设他是3 就会有3个列表文件名

if (files == null) {

files = new ArrayList<SmallFile>()

}

else {

files.clear()

}

for (int i = 0i <filecounti++) {

SmallFile file = new SmallFile()

in.read(file.fileName)

file.offset = in.readInt()//列表文件地址1: 长度32 想把它读到int里

file.length = in.readInt()//列表文件长度1: 长度32 想把它读到int里

files.add(file)

}

}

public void writeToFile() throws Exception {

String temp = filename + ".tmp"//临时文件

boolean exists = false

RandomAccessFile raf = null

try {

raf = new RandomAccessFile(filename, "r")//文件存在则从文件读入

exists = true

}

catch (Exception ex) {

System.out.println("文件不存在,因此启用内存写入模式")

}

if (filecount != files.size()) {

throw new Exception("怪事,居然不相同??")

}

DataOutputStream out = new DataOutputStream(new BufferedOutputStream(new

FileOutputStream(temp)))

//1、写总文件头

out.write(filehead)

out.writeInt(filecount)

//2、写列表头

int sumlength = 0

for (int i = 0i <files.size()i++) {

SmallFile file = files.get(i)

out.write(file.fileName)

if (file.length <0) {

throw new Exception("怪事,文件长度怎么可能小于0?")

}

else {

out.writeInt(ReadBinary.HEADLEN + SmallFile.HEADLEN * filecount +

sumlength)

sumlength += file.length

out.writeInt(file.length)

}

}

//3、写文件内容

for (int i = 0i <files.size()i++) {

SmallFile file = files.get(i)

if (file.content != null &&(file.length == file.content.length)) {

out.write(file.content)

}

else if (exists) {

raf.seek(file.offset)

byte[] b = new byte[file.length]

raf.read(b)

System.out.println("b:" + new String(b))

out.write(b)

}

else {

throw new Exception("怪事,又不能从内存读,又不能从文件读。这活没法干了!")

}

}

out.close()

if (raf != null) {

raf.close()

raf = null

}

System.gc()

//把原先的文件删除

File f = new File(filename)

f.delete()

//再把临时文件改名到正式文件

File f2 = new File(temp)

f2.renameTo(f)

}

public void addFile(SmallFile file) {

if (files != null) {

filecount++

files.add(file)

}

}

public static void test1(){

ReadBinary rb = new ReadBinary("f:\\temp\\rb.dat")

rb.setFilehead("HEAD1234567890122222222222222222".getBytes())

SmallFile f = new SmallFile("第1个文件".getBytes(), "第1个文件的内容".getBytes())

rb.addFile(f)

try {

rb.writeToFile()

}

catch (Exception ex) {

ex.printStackTrace()

}

}

public static void test2(){

ReadBinary rb = new ReadBinary("f:\\temp\\rb.dat")

rb.setFilehead("HEA".getBytes())

SmallFile f = new SmallFile("第2个文件".getBytes(), "第2个文件的内容".getBytes())

rb.addFile(f)

try {

rb.writeToFile()

}

catch (Exception ex) {

ex.printStackTrace()

}

}

public static void main(String[] args) {

//test1()

test2()

}

}

Reko是一个C#项目,包含机器码二进制文件的反编译器。该项目可在GNU通用公共许可证下免费获得。该项目包括前端,核心反编译器引擎和后端,以帮助它实现其目标。在撰写本文时,存在命令行,Windows GUI和ASP.NET前端。反编译器引擎以单个可执行文件或反编译器项目文件的形式从前端接收输入。Reko项目文件包含有关二进制文件的其他信息,有助于反编译过程或格式化输出。然后,反编译器引擎继续分析输入二进制文件。

完整的文档,如下所示:

https://mp.weixin.qq.com/s/DIz8GcFYI0xA7aLJamohDA


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/tougao/8143604.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-13
下一篇 2023-04-13

发表评论

登录后才能评论

评论列表(0条)

保存