Java - IO 記憶體流和列印流

来源:https://www.cnblogs.com/wangyuyang1016/archive/2019/07/12/11178592.html
-Advertisement-
Play Games

IO 記憶體流和列印流操作 字元編碼 電腦中所有的信息組成都是二進位數據,所有能夠描述的中文文字都是經過處理後的結果;所有的語言文字都會使用編碼來進行描述,例如:ASCII碼 常見編碼 GBK/GB2312: 中文的國標編碼 GBK包含有簡體中文與繁體中文兩種,而GB2312只包含簡體中文 ISO ...


IO 記憶體流和列印流操作

字元編碼

  • 電腦中所有的信息組成都是二進位數據,所有能夠描述的中文文字都是經過處理後的結果;所有的語言文字都會使用編碼來進行描述,例如:ASCII碼

常見編碼

GBK/GB2312:

  • 中文的國標編碼
  • GBK包含有簡體中文與繁體中文兩種,而GB2312只包含簡體中文

ISO-8859-1:

  • 國際編碼
  • 可以描述任何的文字信息

UNICODE:

  • 十六進位編碼
  • 任何文字信息都用十六進位表示,會導致無用數據過多

UTF-8:*

  • 融合ISO8859-1和UNICODE兩種編碼的特點

字元亂碼

本質:

  • 編碼與解碼的字元集不統一

列出系統的所有環境變數

public class TestDemo {
    public static void main(String [] args) throws IOException {
        System.getProperties().list(System.out);
    }
}
  • 輸出結果
file.encoding=UTF-8

代表系統環境預設的字元集編碼為:UTF-8

public class TestDemo {
    public static void main(String [] args) throws IOException {
        File file = new File("F:" + File.separator + 
                "demo" + File.separator +
                "demo.txt");
        OutputStream out = new FileOutputStream(file);
        out.write("華為".getBytes());// getBytes() > 無轉碼
        out.close();
    }
}

上述是以預設的系統編碼方式進行輸出

華為

可以看出來,利用系統預設的編碼方式(不轉碼)編碼輸出,在用系統的編碼方式解碼,不會出現亂碼現象

public class TestDemo {
    public static void main(String [] args) throws IOException {
        File file = new File("F:" + File.separator + 
                "demo" + File.separator +
                "demo.txt");
        OutputStream out = new FileOutputStream(file);
        out.write("華為".getBytes("ISO8859-1"));// getBytes() > 無轉碼
        out.close();
    }
}

我們利用getBytes()方法將進行轉碼為ISO8859-1編碼方式輸出

??

由結果看出,系統使用GB2312進行解碼,而文件是使用ISO8859-1進行編碼,編碼和解碼的字元集不同由此導致了 亂碼現象 的出現

記憶體操作流

  • 在不產生新文件的情況下;利用記憶體流來實現輸入與輸出的操作

位元組記憶體流:

public class ByteArrayInputStream
extends InputStream
public class ByteArrayOutputStream
extends OutputStream

ByteArrayInputStream

  • 構造方法
public ByteArrayInputStream(byte [] buf)

將要操作的數據設置到記憶體輸入流

ByteArrayOutputStream

  • 構造方法
public ByteArrayOutputStream()

記憶體輸出流(輸出數據 )

toByteArray()*

public byte [] toByteArray()
  • 將所有保存在記憶體中的位元組數據變為位元組數組存在
  • 將兩個文件利用 toByteArray() 進行合併輸出*
public class TestDemo {
    public static void main(String [] args) throws IOException {
        File fileA = new File("F:" + File.separator + "demo" + File.separator + "demo.txt");
        File fileB = new File("F:" + File.separator + "demo" + File.separator + "data.txt");
        InputStream inA = new FileInputStream(fileA);
        InputStream inB = new FileInputStream(fileB);
        ByteArrayOutputStream output = new ByteArrayOutputStream();
        int temp = 0 ;
        while((temp = inA.read()) != -1) { //讀取A數據
            output.write(temp);
        }
        while((temp = inB.read()) != -1) { //讀取B數據
            output.write(temp);
        }
        // 讀取A,B文件結束後,將記憶體中的所有位元組數據轉為位元組數組
        byte [] data = output.toByteArray();
        inA.close();
        inB.close();
        output.close();
        System.out.println(new String(data));
    }
}

實例

public class TestDemo {
    public static void main(String [] args) throws IOException {
        String str = "Hello,World!";
        InputStream in = new ByteArrayInputStream(str.getBytes());
        // 將所有要讀取的數據設置大記憶體輸入流中
        OutputStream out = new ByteArrayOutputStream();
        // 記憶體輸出流
        int temp = 0 ;// 讀取到的每一個位元組數據
        while ((temp = in.read()) != -1) { // 每次讀取一個位元組數據
            out.write(Character.toUpperCase(temp));//位元組輸出流
            //   temp數據轉大寫並輸出到記憶體輸出流當中
        }
        System.out.println(out);
        in.close();
        out.close();
    }
}

字元記憶體流:

public class CharArrayReader
extends Reader
public class CharArrayWriter
extends Writer

列印流

接觸列印流

如果使用OutputStream,輸出String字元串數據,就需要將String變為位元組數組輸出getBytes(),同理boolean也需要變為Byte數據輸出……

package helloworld;

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

class PrintUtil {
    private OutputStream out; // 輸出依靠 OutputSteam類
    public PrintUtil(OutputStream out) {
        this.out = out ; //確定OutputStream的對象是File……或者ByteArray……
    }
    public void print(int x) throws IOException {
        this.print(String.valueOf(x));//將x轉為String型
    }
    public void print(String x) throws IOException {
        this.out.write(x.getBytes());
    }
    public void print(double x) throws IOException {
        this.print(String.valueOf(x));//將x轉為String型
    }
    public void println(int x) throws IOException {
        this.println(String.valueOf(x));
    }
    public void println(String x) throws IOException {
        this.print(x.concat("\r\n"));
        //在String字元串結尾添加字元[concat()]
    }
    public void println(double x) throws IOException {
        this.println(String.valueOf(x));
    }
    public void close() throws IOException {//關閉輸出流
        this.out.close();
    }
}

public class TestDemo {
    public static void main(String [] args) throws IOException {
        // 調用PrintUtil類的構造方法,實例化對象
        PrintUtil pu = new PrintUtil(
                new FileOutputStream(
                        new File("F:" 
                                    + File.separator + "demo" 
                                    + File.separator + "demo.txt")));
        pu.print("Hello,");
        pu.println("World!");
        pu.println(1+1);
        pu.println(1.1+1.1);
        pu.close();
    }
}

PrintUtil類,則是為了方便列印而設計的一個工具類,在類中,我們通過調用print方法,可以將當前的數據轉為String後在轉為Byte型數據,可以方便我們的數據輸出;避免我們在代碼編寫過程中浪費時間來設計數據類型轉換為Byte位元組輸出。

列印流

  • 為瞭解決上述的數據輸出時的功能不足問題,java.io包提供了一套專門用於輸出數據的類:PrintStream(列印位元組流)PrintWriter(列印字元流)

PrintStream:位元組列印流

  • 繼承結構
java.lang.Object
    java.io.OutputStream
        java.io.FileOutputStream
            java.io.PrintStream
  • 構造方法
PrintStream(OutputStream out)

在PrintStream類中提供了一系列和上述 PrintUtil 類相似的print()、println()方法;這些方法都可支持各種數據類型的輸出,同理:使用了PrintStream,那麼就可以不用去使用write()位元組流輸出方法了。

PrintStream類的實現本質上是基於OutputStream類實現的;這類的設計在Java中稱為 裝飾設計模式 相當於將一個功能不足的操作類,通過某些類的包裝實現成功能健全的更好的操作類。

  • 實現 PrintStream
public class TestDemo {
    public static void main(String [] args) throws IOException {
        // 調用PrintStream類的構造方法,實例化對象
        PrintStream pu = new PrintStream(
                new FileOutputStream(
                        new File("F:" 
                                    + File.separator + "demo" 
                                    + File.separator + "demo.txt")));
        pu.print("Hello,");
        pu.println("World!");
        pu.println(1+1);
        pu.println(1.1+1.1);
        pu.close();
    }
}

將PrintUtil類刪除,使用PrintStream類代替,可以看出操作方法如出一轍。

PrintWrite:字元列印流


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 背景 最近工作上有個類似需求是: 現有約3億條數據詞典存在於一個csv文件A中,作為數據源。對於 用戶輸入的任意單詞M,需要快速的在A中匹配M單詞是否存在。(A文件約3G大小左右,總行數三億) 拿到這個需求,你的第一想法怎麼做呢? 正常思路可能是: 上面的方式有個明顯的缺點是:慢! 3億多行的數據, ...
  • 鍵盤錄入兩個數據,返回兩個數中的較大值: 結果: ...
  • 按照Laravel文檔的安裝方式在windows上安裝Laravel時,執行composer global require "laravel/installer"。 然後命令行就顯示Changed current directory to C:/Users/Administrator/AppData ...
  • 花下貓語:本文是學習群內 櫻雨樓 小姐姐的投稿。之前已發佈過她的一篇作品《 "當談論迭代器時,我談些什麼?" 》,大受好評。本文依然是對比 C++ 與 Python,來探討編程語言中極其重要的概念。祝大家讀有所獲,學有所成! 櫻雨樓 | 原創作者 豌豆花下貓 | 編輯潤色 本文原創並首發於公眾號【 ...
  • 1.方法優勢: 提高代碼的復用性 2.什麼是方法: 完成特定功能的代碼塊。 3.方法的格式: 修飾符 返回值類型 方法名(參數類型 參數名1,參數類型 參數名2...) { 方法體語句; return 返回值; } 4.方法的格式說明: 修飾符:目前就用 public static。後面我們再詳細的 ...
  • 基礎數據類型補充以及二次編碼 一.基礎數據類型補充 1.字元串(str) 首字母大寫 每個單詞首字母大寫 大小寫反轉 居中 填充 查找(從左向右,之查找一個) 拼接 格式化 2.列表 定義方式 list(可迭代對象) + 元素都是共用的 排序(預設升序) 降序 反轉 面試題 3.元組 4.字典 定義 ...
  • 方法之兩個整數的求和案例: 結果: ...
  • 異常原因: 對cookie 0版本標準而言,不能直接使用逗號這種特殊符號作為cookie的內容。而新版本的Cookie(參見RFC 2109)目前還不被Javax.servlet.http.Cookie包所支持。 補充: Cookie Version 0中,某些特殊的字元,例如:空格,方括弧,圓括弧 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...