Java深入淺出說流的使用

一、File類的使用

1.File類的一個對象,代表一個文件或一個文件目錄(俗稱:文件夾)
2.File瞭聲明在java.io包下
3.File類中涉及到關於文件或文件目錄的創建、刪除、重命名、修改時間、文件大小等方法。
並涉及到寫入或讀取文件內容的操作。入宮需要讀取或寫入文件內容,必須使用IO流來完成。
4.後續File類的對象常會作為參數傳遞到流的構造器中,指明讀取或寫入的”終點”。

A.常用構造器

 

B.路徑分隔符

/*
    1.如何創建File類的實例
      File(String filePath)
      File(String parentPath,String,childPath)
      File(File parentFile,String childPath)
    2.相對路徑:相較於某個路徑下,知名的路徑。
      絕對路徑:包含盤符在內的未見或文件目錄的路徑
    3.路徑分隔符
      windows:\\
      unix:/
     */
    public void test1() {
        //構造器1
        File file1 = new File("hello.txt");//相對於module
        File file2 = new File("E:\\zxdym\\IDEA\\code\\JavaSenior\\2021_08\\he.txt");
        System.out.println(file1);
        System.out.println(file2);
        //構造器2
        File file3 = new File("E:\\zxdym\\IDEA\\code", "JavaSenior");
        System.out.println(file3);
        //構造器3
        File file4 = new File(file3, "hi.txt");
        System.out.println(file4);
    }

C.常用方法

 
    public void test2() {
        File file1 = new File("hello.txt");
        File file2 = new File("d:\\io\\hi.txt");
        System.out.println(file1.getAbsolutePath());
        System.out.println(file1.getPath());
        System.out.println(file1.getName());
        System.out.println(file1.getParent());
        System.out.println(file1.length());
        System.out.println(new Date(file1.lastModified()));
        System.out.println();
        System.out.println(file2.getAbsolutePath());
        System.out.println(file2.getPath());
        System.out.println(file2.getName());
        System.out.println(file2.getParent());
        System.out.println(file2.length());
        System.out.println(file2.lastModified());
    }
    public void test3() {
        File file = new File("E:\\\\zxdym\\\\IDEA\\\\code\\\\JavaSenior");
        String[] list = file.list();
        for (String s : list) {
            System.out.println(s);
        }
        System.out.println();
        File[] files = file.listFiles();
        for (File f : files) {
            System.out.println(f);
        }
    }
/*
    public boolean removeTo(File dest):把文件重命名為指定的文件路徑
      比如:file1.renameTo(file2)為例:
            要想保證返回true,需要file1在硬盤中是存在的,且file2不能在硬盤中存在。
     */
    @Test
    public void test4() {
        File file1 = new File("hello.txt");
        File file2 = new File("D:\\io\\hi.txt");
        boolean renameTo = file1.renameTo(file2);
        System.out.println(renameTo);
    }
 
    @Test
    public void test5() {
        File file1 = new File("hello.txt");
        System.out.println(file1.isDirectory());
        System.out.println(file1.isFile());
        System.out.println(file1.exists());
        System.out.println(file1.canRead());
        System.out.println(file1.canWrite());
        System.out.println(file1.isHidden());
    }
 @Test
    public void test6() throws IOException {
        File file1 = new File("hi.txt");
        if (!file1.exists()) {
            file1.createNewFile();
            System.out.println("創建成功");
        } else {//文件存在
            file1.delete();
            System.out.println("刪除成功");
        }
    }
    @Test
    public void test7(){
        //文件目錄的創建
        File file1 = new File("e:\\io\\io1\\io3");
        boolean mkdir = file1.mkdir();
        if(mkdir){
            System.out.println("創建成功1");
        }
 
        File file2 = new File("e:\\io\\io1\\io3");
        boolean mkdir1 = file1.mkdirs();
        if(mkdir1){
            System.out.println("創建成功2");
        }
    }

D.註意點

二、流的分類及其體系


開發中,用緩沖流,效率比節點流高(藍色框中的表示重要的、常用的)

輸入、輸出的標準化過程

1.輸入過程

A.創建File類的對象,指明讀取的數據的來源。(要求此文件一定要存在)

B.創建相應的輸入流,將File類的對象作為參數,傳入流的構造器中

C.具體的讀入過程:

創建相應的byte[] 或 char[]

D.關閉流資源

說明:程序中出現的異常需要使用try-catch-finally處理。

2.輸出過程

A.創建File類的對象,指明寫出的數據的來源。(要求此文件一定要存在)

B.創建相應的輸出流,將File類的對象作為參數,傳入流的構造器中

C.具體的寫出過程:

write(char[]/byte[] buffer,0,len)

D.關閉流資源

說明:程序中出現的異常需要使用try-catch-finally處理。

public class FileReaderWriterTest {
    public static void main(String[] args) {
        File file = new File("hello.txt");//相較於前工程
        System.out.println(file.getAbsolutePath());
        File file1 = new File("2021_08\\hello.txt");
        System.out.println(file1.getAbsolutePath());
    }
    /*
    將day09下的hello.txt文件內容讀入程序中,並輸出到控制臺
    說明點:
    1.read()的理解,返回讀入的一個字符,如果達到文件末尾,返回-1
    2.異常的處理:為瞭保證流資源一定可以執行關閉操作。需要使用try-catch-finally處理
    3.讀入的文件一定要存在,否則就會報FileNotFoundException.
     */
    @Test
    public void testFileReader(){
        FileReader fr = null;
        try {
            //1.實例化File類的對象,指明要操作的文件
            File file = new File("hello.txt");//相較於當前Module
            //2.提供具體的流
            fr = new FileReader(file);
            //3.數據的讀入
            //read():返回讀入的一個字符,如果達到文件末尾,返回-1
            //方式一:
//        int data = fr.read();
//        while(data != -1){
//            System.out.print((char)data);
//            data = fr.read();
//        }
            //方式二:語法上針對方式一的修改
            int data;
            while((data = fr.read()) != -1){
                System.out.print((char)data);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //4.流的關閉操作
            try {
                if(fr != null)
                fr.close();
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
            }
        }
    }
    @Test
    public void testFileReader1(){
        FileReader fr = null;
        try {
            //1.File類的實例化
            File file = new File("hello.txt");
            //2.FileReader流的實例化
            fr = new FileReader(file);
            //3.讀入的操作
            //read(char[] cbuf):返回每次讀入cbuf數組中的字符的個數,如果達到文件末尾,返回-1
            char[] cbuf = new char[5];
            int len;
            while(( len = fr.read(cbuf)) != -1){
                //方式一:
                //錯誤的寫法//知識點難點:數組元素的覆蓋
//                for(int i = 0;i < cbuf.length;i++){
//                    System.out.print(cbuf[i]);
//                }
                //正確的寫法
//                for(int i = 0;i < len;i++){
//                    System.out.print(cbuf[i]);
//                }
                //方式二:
                //錯誤的寫法,對應著方式一的錯誤的寫法
//                String str = new String(cbuf);
//                System.out.println(str);
                //正確的寫法
                String str = new String(cbuf, 0, len);
                System.out.print(str);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if(fr != null){
            //4.資源的關閉
            try {
                fr.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            }
        }
    }
    /*
    從內存中寫出數據到硬盤的文件裡
    說明:
    1.輸出操作,對應的File可以不存在的,並不會報異常
    2.
      File對應的硬盤中的文件如果不存在,在輸出的過程中,會自動創建此文件
      File對應的硬盤中的文件如果存在:
              如果流使用的構造器是:FileWriter(file,false) / FileWriter(file):對原有文件的覆蓋
              如果流使用的構造器是:FileWriter(file,true):不會對原有文件覆蓋,而是在原有文件基礎上追加內容
     */
    @Test
    public void testFileWriter() {
        FileWriter fw = null;
        try {
            //1.提供File類的對象,指明寫出到的文件
            File file = new File("hello1.txt");
            //2.提供FileWriter的對象,用於數據的寫出
            fw = new FileWriter(file,false);
            //3.寫出的操作
            fw.write("I have a dream!\n");
            fw.write("you need to have a dream!");
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //4.流資源的關閉
            if(fw != null){
                try {
                    fw.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
    @Test
    public void testFileReaderFileWriter(){
        FileReader fr = null;
        FileWriter fw = null;
        try {
            //1.創建File類的對象,指明讀入和寫出的文件
            File srcFile = new File("hello.txt");
            File destFile = new File("hello2.txt");
 
            //不能使用字符流來處理圖片等字節數據
//            File srcFile = new File("愛情與友情.jpg");
            File destFile = new File("愛情與友情1.jpg");
            //2.創建數據入流和輸出流的對象
            fr = new FileReader(srcFile);
            fw = new FileWriter(destFile);
            //3.數據的讀入和寫出操作
            char[] cbuf = new char[5];
            int len;//記錄每次讀入到cbuf數組中的字符的個數
            while((len = fr.read(cbuf)) != -1){
                //每次寫出len個字符
                fw.write(cbuf,0,len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
//            //4.關閉流資源
//            //方式一:
//            try {
//                fw.close();
//            } catch (IOException e) {
//                e.printStackTrace();
//            }finally {
//                try {
//                    fr.close();
//                } catch (IOException e) {
//                    e.printStackTrace();
//                }
//            }
            //方式二:
            try {
                fw.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            try {
                fr.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

三、流的詳細介紹

1.字節流和字符流

測試FileInputStream和FileOutputStream的使用
 
public class FileInputOutputStreamTest {
    //使用字節流FileInputOutputStream處理文本文件,可能出現亂碼
    @Test
    public void testFileInputStream(){
        FileInputStream fis = null;
        try {
            //1.造文件
            File file = new File("hello.txt");
            //2.造流
            fis = new FileInputStream(file);
            //3.讀數據
            byte[] buffer = new byte[5];
            int len;//記錄每次讀取的字節的個數
            while((len = fis.read(buffer)) != -1){
                String str = new String(buffer, 0, len);
                System.out.print(str);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //4.關閉資源
            try {
                fis.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    /*
    實現對圖片的復制操作
     */
    @Test
    public void testFileInputOutputStream(){
        FileInputStream fis = null;
        FileOutputStream fos = null;
        try {
            File srcFile = new File("愛情與友情.jpg");
            File destFile = new File("愛情與友情2.jpg");
            fis = new FileInputStream(srcFile);
            fos = new FileOutputStream(destFile);
            //復制的過程
            byte[] buffer = new byte[5];
            int len;
            while((len = fis.read(buffer)) != -1){
                fos.write(buffer,0,len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if(fos != null){
                try {
                    fos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if(fis != null){
                try {
                    fis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
    //指定路徑下文件的復制
    public void copyFile(String srcPath,String destPath){
        FileInputStream fis = null;
        FileOutputStream fos = null;
        try {
            File srcFile = new File(srcPath);
            File destFile = new File(destPath);
            fis = new FileInputStream(srcFile);
            fos = new FileOutputStream(destFile);
            //復制的過程
            byte[] buffer = new byte[5];
            int len;
            while((len = fis.read(buffer)) != -1){
                fos.write(buffer,0,len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if(fos != null){
                try {
                    fos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if(fis != null){
                try {
                    fis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
    @Test
    public void testCopyFile(){
        long start = System.currentTimeMillis();
        String srcPath = "C:\\Users\\Administrator\\Desktop\\01-視頻.avi";
        String destPath = "C:\\Users\\Administrator\\Desktop\\02-視頻.avi";
        copyFile(srcPath,destPath);
        long end = System.currentTimeMillis();
        System.out.println("復制操作花費的時間為:" + (end - start));
    }
}

結論:
1.對於文本文件(.txt,.java,.c,.cpp),使用字符流處理
2.對於非文本文件(.jpg,.mp3,.mp4,.avi,.doc,.ppt,…),使用字節流處理

2.節點流和處理流(重點)

處理流之一:緩沖流的作用
1.緩沖流:
BufferedInputStream
BufferedOutputStream
BufferedReader
BufferedWriter
2.作用:提高流的讀取,寫入的速度
提高讀寫速度的原因:內部提供瞭一個緩沖區,默認情況是8kb
3.處理流:就是”套接”在已有流的基礎上

public class BufferedTest {
    @Test
    public void BufferedStreamTest(){
        BufferedInputStream bis = null;
        BufferedOutputStream bos = null;
        try {
            //1.造文件
            File srcFile = new File("愛情與友情.jpg");
            File destFile = new File("愛情與友情3.jpg");
            //2.造流
            //2.1造節點流
            FileInputStream fis = new FileInputStream(srcFile);
            FileOutputStream fos = new FileOutputStream(destFile);
            //2.2造緩沖流
            bis = new BufferedInputStream(fis);
            bos = new BufferedOutputStream(fos);
            //3.復制的細節:讀取、寫入
            byte[] buffer = new byte[10];
            int len;
            while((len = bis.read(buffer)) != -1){
                bos.write(buffer,0,len);
//                bos.flush();//刷新緩沖區
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //4.資源關閉
            //要求:先關閉外層的流,再關閉內層的流
            if(bos != null){
                try {
                    bos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if(bis != null){
                try {
                    bis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            //說明:關閉外層流的同時,內層流也會自動的進行關閉,關於內層流的關閉,可以省略
//        fos.close();
//        fis.close();
        }
 
    }
    @Test
    public void testCopyFileWithBuffered(){
        long start = System.currentTimeMillis();
        String srcPath = "C:\\Users\\Administrator\\Desktop\\01-視頻.avi";
        String destPath = "C:\\Users\\Administrator\\Desktop\\03-視頻.avi";
        copyFileWithBuffered(srcPath,destPath);
        long end = System.currentTimeMillis();
        System.out.println("復制操作花費的時間為:" + (end - start));
    }
    //實現文件復制的方法
    public void copyFileWithBuffered(String srcPath,String destPath) {
        BufferedInputStream bis = null;
        BufferedOutputStream bos = null;
        try {
            //1.造文件
            File srcFile = new File(srcPath);
            File destFile = new File(destPath);
            //2.造流
            //2.1造節點流
            FileInputStream fis = new FileInputStream(srcFile);
            FileOutputStream fos = new FileOutputStream(destFile);
            //2.2造緩沖流
            bis = new BufferedInputStream(fis);
            bos = new BufferedOutputStream(fos);
            //3.復制的細節:讀取、寫入
            byte[] buffer = new byte[10];
            int len;
            while ((len = bis.read(buffer)) != -1) {
                bos.write(buffer, 0, len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //4.資源關閉
            //要求:先關閉外層的流,再關閉內層的流
            if (bos != null) {
                try {
                    bos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (bis != null) {
                try {
                    bis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            //說明:關閉外層流的同時,內層流也會自動的進行關閉,關於內層流的關閉,可以省略
//        fos.close();
//        fis.close();
        }
    }
    /*
    使用BufferedReader和BufferedWriter實現文本文件的復制
     */
    @Test
    public void testBufferedReaderBufferedWriter(){
        BufferedReader br = null;
        BufferedWriter bw = null;
        try {
            //創建文件和相應的流
            br = new BufferedReader(new FileReader(new File("dpcp.txt")));
            bw = new BufferedWriter(new FileWriter(new File("dpcp1.txt")));
            //讀寫操作
            //方式一:使用char[]數組
//            char[] cbuf = new char[1024];
//            int len;
//            while((len = br.read(cbuf)) != -1){
//                bw.write(cbuf,0,len);
//            }
            //方式二:使用String
            String data;
            while((data = br.readLine()) != null){
                //方法一:
                bw.write(data + "\n");//data中不包含換行符
                //方法二:
                bw.write(data);//data中不包含換行符
                bw.newLine();//提供換行的操作
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //關閉資源
            if(bw != null){
                try {
                    bw.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if(br != null){
                try {
                    br.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

處理流之二:轉換流的使用(重點)
1.轉換流:屬於字符流
InputStreamReader:將一個字節的輸入流轉換為字符的輸入流
OutputStreamWriter:將一個字符的輸出流轉換為字節的輸出流


2.作用:提供字節流與字符流之間的轉換
3.解碼:字節、字節數組 —> 字符數組、字符串
編碼:字符數組、字符串—> 字節、字節數組

說明:編碼決定瞭解碼的方式
4.字符集

說明:文件編碼的方式(比如:GBK),決定瞭解析時使用的字符集(也隻能是GBK)

public class InputStreamReaderTest {
    /*
    此時處理異常的話,仍然應該使用try-catch-finally
     */
    @Test
    public void test1() throws IOException{
        FileInputStream fis = new FileInputStream("dbcp.txt");
//        InputStreamReader isr = new InputStreamReader(fis);//使用系統默認的字符集
        //參數2指明瞭字符集,具體使用那個字符集,取決於文件dbcp.txt保存時使用的字符集
        InputStreamReader isr = new InputStreamReader(fis, "UTF-8");
        char[] cbuf = new char[20];
        int len;
        while((len = isr.read(cbuf)) != -1){
            String str = new String(cbuf,0,len);
            System.out.print(str);
        }
        isr.close();
    }
    /*
    此時處理異常的話,仍然應該使用try-catch-finally
    綜合使用InputStreamReader和OutputStreamWriter
     */
    @Test
    public void test2() throws Exception{
        //1.造文件、造流
        File file1 = new File("dbcp.txt");
        File file2 = new File("dbcp_gbk.txt");
        FileInputStream fis = new FileInputStream(file1);
        FileOutputStream fos = new FileOutputStream(file2);
        InputStreamReader isr = new InputStreamReader(fis, "UTF-8");
        OutputStreamWriter osw = new OutputStreamWriter(fos, "gbk");
        //2.讀寫過程
        char[] cbuf = new char[20];
        int len;
        while((len = isr.read(cbuf)) != -1){
            osw.write(cbuf,0,len);
        }
        //3.關閉資源
        isr.close();
        osw.close();
    }
}

3.其他流

1.標準的輸入、輸出流

1.1  System.in:標準的輸入流,默認從鍵盤輸入

System.out:標準的輸入流,默認從控制臺輸出

1.2  System類的setIn() / setOut()方式重新指定輸入和輸出的流。

修改默認的輸入和輸出行為:

System類的setIn(InputStream is) / setOut(PrintStream ps)方式重新指定輸入和輸出的流。

1.3  練習:
從鍵盤輸入字符串,要求將讀取道德整行字符串轉換成大寫輸出。然後繼續進行輸入操作。
直至當輸入”e”或”exit”時,退出程序
方法一:使用Scanner實現,調用next()返回一個字符串
方法二:使用System.in實現。System.in —> 轉換流 –>BufferedReader的readLine()

 public static void main(String[] args) {
        BufferedReader br = null;
        try {
            InputStreamReader isr = new InputStreamReader(System.in);
            br = new BufferedReader(isr);
            while(true){
                System.out.println("請輸入字符串:");
                String data = br.readLine();//調用此方法讀取一行數據
                if("e".equalsIgnoreCase(data) || "exit".equalsIgnoreCase(data)){//避免空指針的寫法,之前有
                    System.out.println("程序結束");
                    break;
                }
                String upperCase = data.toUpperCase();
                System.out.println(upperCase);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if(br != null){
                try {
                    br.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

對象流 (重點)

對象流的使用

1.ObjectInputStream和ObjectOutputStream
2.作用:用於存儲和讀取基本數據類型或對象的處理流。
3.要想一個Java對象是可序列化的,需要滿足相應的要求。見Person.java
4.序列化機制:(重點!!!!)
對象序列化機制允許把內存中的Java對象轉換成平臺無關的二進制流,從而允許把這種二進制流持久的保存在磁盤上,或通過網絡將這種
二進制流傳輸到另一個網絡節點,當其他程序獲取瞭這種二進制流,就可以恢復成原來的Java對象

 
    序列化過程:將內存中的java對象保存到磁盤中或通過網絡傳輸出去
    使用ObjectOutputStream實現
    @Test
    public void testObjectOutputStream(){
        ObjectOutputStream oos = null;
        try {
            oos = new ObjectOutputStream(new FileOutputStream("object.dat"));
            oos.writeObject(new String("我愛北京天安門"));
            oos.flush();//刷新操作
            oos.writeObject(new Person("王銘",23));
            oos.flush();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if(oos != null){
                try {
                    oos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
    反序列化:將磁盤文件中的對象還原為內存中的一個Java對象
    使用ObjectInputStream來實現
    @Test
    public void testObjectInputStream(){
        ObjectInputStream ois = null;
        try {
            ois = new ObjectInputStream(new FileInputStream("object.dat"));
            Object obj = ois.readObject();
            String str = (String) obj;
            Person p = ois.readObject();
            System.out.println(str);
            System.out.println(p);
        }catch(IOException e){
            e.printStackTrace();
        }catch(ClassNotFoundException e){
            e.printStackTrace();
        }finally {
            if(ois != null){
                ois.close;
            }
        }
    }

Person類

Person需要滿足如下的要求,方可序列化
1.需要實現接口:Serializable
2.當前類提供一個全局常量:serialVersionUID
3.除瞭當前Person類需要實現Serializable接口之外,還必須保證其內部所有屬性也必須是可序列化的(默認情況下,基本數據類型、String:本身是可序列化的)
補充:ObjectOutputStream 和 ObjectInputStream不能序列化static和transient修飾的成員變量
eg:輸出結果:Person{name=’null’,age=0,id=0,acct=null}

public class Person implements Serializable{
    public static final long serialVersionUID = 397497937034L;
    private String name;
    private int age;
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public Person() {
    }
}

對象的序列化機制

隨機存取文件流(瞭解)

 
 RandomAccessFile
 1.RandomAccessFile直接繼承於java.lang.Object類,實現瞭DataInput和DataOutput接口
 2.RandomAccessFile既可以作為一個輸入流,又可以作為一個輸出流
 3.如果RandomAccessFile作為輸出流時,寫出到的文件如果不存在,則在執行過程中自動創建
   如果寫出到的文件存在,則會對原有文件內容進行覆蓋,(默認情況下,從頭覆蓋)
 4.可以通過相關的操作,實現RandomAccessFile"插入"數據的效果
 
public abstract class RandomAccessFileTest {
    @Test
    public void test1(){
        RandomAccessFile raf1 = null;
        RandomAccessFile raf2 = null;
        try {
            raf1 = new RandomAccessFile(new File("愛情與友情.jpg"), "r");
            raf2 = new RandomAccessFile(new File("愛情與友情1.jpg"), "rw");
            byte[] buffer = new byte[1024];
            int len;
            while((len = raf1.read(buffer)) != -1){
                raf2.write(buffer,0,len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if(raf1 != null){
                try {
                    raf1.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if(raf2 != null){
                try {
                    raf2.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
    @Test
    public void test2()throws IOException{
        RandomAccessFile raf1 = new RandomAccessFile("hello.txt","rw");
        raf1.seek(3);//將指針調到角標為3的位置
        raf1.write("xyz".getBytes());//
        raf1.close();
    }
    
    使用RandomAccessFile實現數據的插入效果
    
    @Test
    public void test3() throws IOException{
        RandomAccessFile raf1 = new RandomAccessFile("hello.txt", "rw");
        raf1.seek(3);//將指針調到角標為3的位置
        //保存指針3後面的所有數據到StringBuilder中
        StringBuilder builder = new StringBuilder((int) new File("hello.txt".length()));
        byte[] buffer = new byte[20];
        int len;
        while((len = raf1.read(buffer)) != -1){
            builder.append(new String(buffer,0,len));
        }
        //調回指針,寫入"xyz"
        raf1.seek(3);
        raf1.write("xyz".getBytes());
        //將StringBuilder中的數據寫入到文件中
        raf1.write(builder.toString().getBytes());
        raf1.close();
    }
}

Java中的NIO

到此這篇關於Java深入淺出說流的使用的文章就介紹到這瞭,更多相關Java流內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: