Java IO流

Java IO流

基本

进制转化

十进制转其它进制

Integer.toBinaryString(112)
Integer.toOctalString(112)
Integer.toHexString(112)

其它进制转十进制

Integer.parseInt("1110000", 2)
Integer.parseInt("27", 8)
Integer.parseInt("A0", 16)

字节

数据类型转字节
例如int类型转字节
Java IO流_第1张图片
字符串转为字节数组
Java IO流_第2张图片

int类型转为字节数组

    public static byte[] int2Bytes(int id){
        byte[] arr = new byte[4];
        for (int i = 0; i < arr.length; i++) {
            arr[i] = (byte)((id >> i*8) & 0xff);
        }
        return arr;
    }

    public static int bytes2Int(byte[] arr){
        int result = 0;
        for (int i = 0; i < arr.length; i++) {
            result += (int)((arr[i] & 0xff) << i*8);
        }
        return result;
    }

文件编码

utf-8,中文占用3个字节,英文占用1个字节
gbk编码中文占用2个字节,英文占用1个字节
utf-16be,中文占用2个字节,英文占用2个字节

        String s = "慕课ABC";
        byte[] bytes1 = s.getBytes("utf-8");
        for (byte b : bytes1) {
            System.out.print(Integer.toHexString(b & 0xff) + " ");
        }

File

java.io.File类用于表示文件(目录)
File类只用于表示文件(目录)的信息(名称、大小等),不能用于文件内容的访问。
mkdir()创建文件夹
mkdirs()创建多级目录
delete()删除文件/文件夹
File.separator设置分隔符
file.isDirectory()是否是一个目录
file.isFile()是否是一个文件
创建一个文件

        //File file2 = new File("/Users/Miller/Movies/img/imooc/日记.txt");
        File file2 = new File("/Users/Miller/Movies/img/imooc", "日记.txt");
        if (!file2.exists()) { try { file2.createNewFile(); } catch (IOException e) { e.printStackTrace(); }
        }else{ file2.delete(); }

遍历目录

list()方法用于列出当前目录下的子目录和文件,返回的是字符串数组。
listFiles()返回的是直接目录和文件的抽象。

    /** * 列出指定目录下(包括子目录)的所有文件 * @param dir * @throws IOException */
    public static void listDirectory(File dir) throws IOException
    {
        if(!dir.exists()){
            throw new IllegalArgumentException("目录:"+dir+"不存在");
        }
        if (!dir.isDirectory()) {
            throw new IllegalArgumentException(dir+"不是目录");
        }
        /* //list()方法用于列出当前目录下的子目录和文件 String[] filenames = dir.list(); for (String filename : filenames) { System.out.println(dir+File.separator+filename); } */

        File[] files = dir.listFiles();
        if (files != null && files.length > 0) {
            for (File file : files) {
                if (file.isDirectory()) {
                    //递归
                    listDirectory(file);
                }else{
                    System.out.println(file);
                }
            }
        }


    }

RandomAccessFile

java提供的对文件内容的访问,既可以读文件,也可以写文件。RandomAccessFile支持随机访问文件,可以访问文件的任意位置。
java文件模型
在硬盘上的文件是byte byte byte存储的,是数据的集合。
打开文件
有两种模式,”rw”读写,”r”只读

RandomAccessFile raf = new RandomAccessFile(file, "rw");

它内部还包括一个文件指针,打开文件时 pointer=0;

写方法
raf.write(int)–>只写一个字节(后8位),同时指针指向下一个位置。

读方法
int b = raf.read()–>读一个字节

文件读写完成以后一定要关闭。

例子
写文件
用write方法只能写一个字节,如果需要把i写进去就得写4次

        int i = 0x7fffffff;
        //用write方法只能写一个字节,如果需要把i写进去就得写4次
        raf.write(i >>> 24);//高8位
        raf.write(i >>> 16);
        raf.write(i >>> 8);
        raf.write(i);
        System.out.println(raf.getFilePointer());

也可以直接写一个int

raf.writeInt(i);

直接写入字节数组

        String s = "中";
        byte[] gbk = s.getBytes("gbk");
        raf.write(gbk);
        System.out.println(raf.getFilePointer());

读文件
读文件,必须把指针移到头部

        String s = "中";
        byte[] gbk = s.getBytes("gbk");
        raf.write(gbk);
        System.out.println(raf.getFilePointer());

        raf.seek(0);
        byte[] buf = new byte[(int)raf.length()];
        raf.read(buf);
        System.out.println(Arrays.toString(buf));
        String s1 = new String(buf, "gbk");
        System.out.println(s1);

        raf.close();

字节流

InputStream/OutputStream
InputStream抽象了应用程序读取数据的方式
OutputStream抽象了应用程序写出数据的方法
EOF= end, 读到-1就读到结尾
输入流基本方法,主要是读。
int b = in.read();读取一个字节无符号填充到int低八位。-1是EOF
in.read(byte[] buf) 读取数据填充到字节数组buf
in.read(byte[] buf, int start, int size)读取数据到字节数组buf,从buf的start位置开始,存放size长度的数据

输出流基本方法,主要是写。
out.write(int b)写一个byte到流,b的低8位
out.write(byte[] but) 将but字节数组都写入到流
out.write(byte[] but, int start, int size) 字节数组but从start位置开始写size长度的字节到流

FileInputStream

FileInputStream–>具体实现了在文件上读取数据
单字节读取不适合大文件,大文件效率很低

    /** * 对于指定文件内容,按照16进制输出到控制台 * 并且每输出10个byte换行 * @param fileName */
    public static void printHex(String fileName) throws IOException {

        //把文件作为字节流进行流操作
        FileInputStream in = new FileInputStream(fileName);
        int b;
        int i = 1;
        while((b=in.read()) != -1){
            if(b < 0xf){
                //单位数前面补0
                System.out.print("0");
            }
            System.out.print(Integer.toHexString(b)+" ");
            if(i++%10 ==0){
                System.out.println();
            }
        }
        in.close();

    }

也可以通过byte来实现。批量字节读取,对大文件而言,效率高,也是最常用的。

    public static void printHexByByteArray(String fileName) throws IOException
    {
        FileInputStream in = new FileInputStream(fileName);
        byte[] buf = new byte[20*1024];
        //从in中批量读取字节,放入buf字节数组中,从0开始放,最多放buf.length
        //返回的是读到的字节的个数
        //一次性读完,说明字节数组足够大
        /* int bytes = in.read(buf, 0, buf.length); int j = 1; for (int i = 0; i < bytes; i++) { if ((buf[i] & 0xff) <= 0xf) { System.out.print("0"); } System.out.print(Integer.toHexString(buf[i] & 0xff)+" "); if (j++ % 10 == 0) { System.out.println(); } } */

        int bytes = 0;
        int j = 0;
        while((bytes = in.read(buf,0,buf.length)) != -1){
            for(int i= 0; i < bytes; i++){
                System.out.print(Integer.toHexString(buf[i] & 0xff)+" ");
                if (j++ % 10 == 0) {
                    System.out.println();
                }
            }
        }

        in.close();
    }

FileOutputStream

FileOutputStream实现了向文件中写出byte数据的方法。

        //如果文件不存在,则直接创建,如果存在,删除后创建
        FileOutputStream out = new FileOutputStream("demo/out.dat");
        //如果文件不存在,则直接创建,如果存在,则在后面添加内容
        //FileOutputStream out = new FileOutputStream("demo/out.dat", true);
        out.write('A');//写出A字符的低八位
        out.write('b');//写出b字符的低八位
        int a = 10;//write只能写八位,那么些int需要写4次
        out.write(a >>> 24);
        out.write(a >>> 16);
        out.write(a >>> 8);
        out.write(a);

        //写字节数组
        byte[] gbk = "中国".getBytes("gbk");
        out.write(gbk);
        out.close();

文件的copy操作

    public static void copyFile(File srcFile, File destFile) throws IOException
    {
        if (!srcFile.exists()) {
            throw new IllegalArgumentException("文件:"+srcFile+"不存在");
        }
        if(!srcFile.isFile()){
            throw new IllegalArgumentException(srcFile+"不是文件");
        }
        FileInputStream in = new FileInputStream(srcFile);
        FileOutputStream out = new FileOutputStream(destFile);
        byte[] buf = new byte[8*1024];
        int b;
        while((b = in.read(buf,0, buf.length)) != -1){
            out.write(buf, 0, b);
            out.flush();
        }
        in.close();
        out.close();

    }

DataOutputStream/DataInputStream

是对流功能的扩展,可以更加方便的读取int、long,字符等类型数据。
DataOutputStream
writeInt()/writeDouble()/writeUTF()

BufferedInputStream&BufferedOutputStream

这两个流类为IO提供了带缓冲区的操作,一般打开文件进行写入或读取操作时,都会加上缓冲,这种流模式提高了IO的性能。

PrintStream

转载自:java io系列16之 PrintStream(打印输出流)详解
PrintStream 是打印输出流,它继承于FilterOutputStream。
PrintStream 是用来装饰其它输出流。它能为其他输出流添加了功能,使它们能够方便地打印各种数据值表示形式。
与其他输出流不同,PrintStream 永远不会抛出 IOException;它产生的IOException会被自身的函数所捕获并设置错误标记, 用户可以通过 checkError() 返回错误标记,从而查看PrintStream内部是否产生了IOException。
另外,PrintStream 提供了自动flush 和 字符集设置功能。所谓自动flush,就是往PrintStream写入的数据会立刻调用flush()函数。

示例代码
关于PrintStream中API的详细用法,参考示例代码(PrintStreamTest.java):

import java.io.PrintStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;

/** * PrintStream 的示例程序 * * @author skywang */
public class PrintStreamTest {

    public static void main(String[] args) {

        // 下面3个函数的作用都是一样:都是将字母“abcde”写入到文件“file.txt”中。
        // 任选一个执行即可!
        testPrintStreamConstrutor1() ;
        //testPrintStreamConstrutor2() ;
        //testPrintStreamConstrutor3() ;

        // 测试write(), print(), println(), printf()等接口。
        testPrintStreamAPIS() ;
    }

    /** * PrintStream(OutputStream out) 的测试函数 * * 函数的作用,就是将字母“abcde”写入到文件“file.txt”中 */
    private static void testPrintStreamConstrutor1() {
        // 0x61对应ASCII码的字母'a',0x62对应ASCII码的字母'b', ...
        final byte[] arr={0x61, 0x62, 0x63, 0x64, 0x65 }; // abced
        try {
            // 创建文件“file.txt”的File对象
            File file = new File("file.txt");
            // 创建文件对应FileOutputStream
            PrintStream out = new PrintStream(
                    new FileOutputStream(file));
            // 将“字节数组arr”全部写入到输出流中
            out.write(arr);
            // 关闭输出流
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    /** * PrintStream(File file) 的测试函数 * * 函数的作用,就是将字母“abcde”写入到文件“file.txt”中 */
    private static void testPrintStreamConstrutor2() {
        final byte[] arr={0x61, 0x62, 0x63, 0x64, 0x65 };
        try {
            File file = new File("file.txt");
            PrintStream out = new PrintStream(file);
            out.write(arr);
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    /** * PrintStream(String fileName) 的测试函数 * * 函数的作用,就是将字母“abcde”写入到文件“file.txt”中 */
    private static void testPrintStreamConstrutor3() {
        final byte[] arr={0x61, 0x62, 0x63, 0x64, 0x65 };
        try {
            PrintStream out = new PrintStream("file.txt");
            out.write(arr);
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    /** * 测试write(), print(), println(), printf()等接口。 */
    private static void testPrintStreamAPIS() {
        // 0x61对应ASCII码的字母'a',0x62对应ASCII码的字母'b', ...
        final byte[] arr={0x61, 0x62, 0x63, 0x64, 0x65 }; // abced
        try {
            // 创建文件对应FileOutputStream
            PrintStream out = new PrintStream("other.txt");

            // 将字符串“hello PrintStream”+回车符,写入到输出流中
            out.println("hello PrintStream");
            // 将0x41写入到输出流中
            // 0x41对应ASCII码的字母'A',也就是写入字符'A'
            out.write(0x41);
            // 将字符串"65"写入到输出流中。
            // out.print(0x41); 等价于 out.write(String.valueOf(0x41));
            out.print(0x41);
            // 将字符'B'追加到输出流中
            out.append('B');

            // 将"CDE is 5" + 回车 写入到输出流中
            String str = "CDE";
            int num = 5;
            out.printf("%s is %d\n", str, num);

            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

运行上面的代码,会在源码所在目录生成两个文件“file.txt”和“other.txt”。
file.txt的内容如下:

abcde

other.txt的内容如下:

hello PrintStream
A65BCDE is 5

你可能感兴趣的:(java,IO流)