# JAVA之旅(三十)——打印流PrintWriter,合并流,切割文件并且合并,對象的序列化Serializable,管道流,RandomAccessFile,IO其他類,字符編碼
* * *
> 三十篇了,又是一個陽光明媚的周末,一個又一個的周末,周而復始,不斷學習,前方的路你可曾看見?隨我一起走進技術的世界,流連忘返吧!
## 一.打印流PrintWriter
> 打印流有PrintWriter和PrintStream,他的特點可以直接操作輸入流還有文件
* 該流提供了打印方法,可以將各種數據類型原樣打印?
* file對象 File
* 字符串路徑 String
* 字節打印流
* 字符打印流
> 我們演示一遍大家就對此有所了解了
~~~
package com.lgl.hellojava;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
public class HelloJJAVA {
public static void main(String[] args) {
try {
BufferedReader bufr = new BufferedReader(new InputStreamReader(
System.in));
PrintWriter oWriter = new PrintWriter(System.out, true);
String line = null;
while ((line = bufr.readLine()) != null) {
if (line.equals("over")) {
break;
}
oWriter.write(line);
}
oWriter.close();
bufr.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 要想存到文件里,也是可以直接把文件給放進去的
## 二.合并流
> 我們來看看什么是合并流,在API文檔上說,他可以串聯流
~~~
package com.lgl.hellojava;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.SequenceInputStream;
import java.util.Enumeration;
import java.util.Vector;
public class HelloJJAVA {
public static void main(String[] args) {
try {
Vector<FileInputStream> v = new Vector<FileInputStream>();
v.add(new FileInputStream("1.txt"));
v.add(new FileInputStream("2.txt"));
Enumeration<FileInputStream> elements = v.elements();
SequenceInputStream sis = new SequenceInputStream(elements);
FileOutputStream fos = new FileOutputStream("3.txt");
byte[] buf = new byte[1024];
int len = 0;
while ((len = sis.read(buf)) != -1) {
fos.write(buf, 0, len);
}
fos.close();
sis.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 把1.txt和2.txt乃至add更多的內容合并到3.txt文件中,這就是流的合并
## 三.切割文件
> 合并文件可以,那肯定可以切割了,我們來看下具體是怎么去制作的
~~~
// 切割文件
public static void splitFile() {
try {
FileInputStream fis = new FileInputStream("1.jpg");
FileOutputStream fos = null;
byte[] buf = new byte[1024 * 1024];
int len = 0;
int count = 1;
while ((len = fis.read(buf)) != -1) {
fos = new FileOutputStream((count++) + ".patch");
fos.write(buf, 0, len);
fos.close();
}
fis.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
~~~
> 當運行結束之后,我們可以看到

> 已經成功切割了
>
> 切割完我們可以合并了
~~~
// 合并文件
public static void merge() {
ArrayList<FileInputStream> al = new ArrayList<>();
for (int i = 1; i <= 2; i++) {
try {
al.add(new FileInputStream(i + ".patch"));
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
Iterator<FileInputStream> iterator = al.iterator();
Enumeration<FileInputStream> en = new Enumeration<FileInputStream>() {
@Override
public boolean hasMoreElements() {
// TODO Auto-generated method stub
return iterator.hasNext();
}
@Override
public FileInputStream nextElement() {
// TODO Auto-generated method stub
return iterator.next();
}
};
try {
SequenceInputStream seq = new SequenceInputStream(en);
FileOutputStream fos = new FileOutputStream("2.jpg");
byte[] buf = new byte[1024];
int len = 0;
while ((len = seq.read(buf)) != -1) {
fos.write(buf, 0, len);
}
fos.close();
seq.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
~~~
> 這樣我們就把圖片拷貝出來了,可以看到

> 這段代碼是非常幫的,我們一定會用到的,希望能用的上
## 四.對象的序列化Serializable
> 其實就是可以直接操作對象的流,他會實現一個Serializable()接口,我們用代碼來看下他是怎么用的,我們直接寫讀取對象的類了
~~~
package com.lgl.hellojava;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
public class HelloJJAVA {
public static void main(String[] args) {
// writeObj();
readObj();
}
private static void readObj() {
try {
ObjectInputStream ois = new ObjectInputStream(new FileInputStream(
"obj.txt"));
Person p = (Person) ois.readObject();
System.out.println(p);
ois.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
private static void writeObj() {
try {
ObjectOutputStream oos = new ObjectOutputStream(
new FileOutputStream("obj.txt"));
oos.writeObject(new Person("zhangsan", 20));
oos.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class Person implements Serializable {
String name;
int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
}
~~~
> OK,自己去驗證一下
## 五.管道流
> 輸入輸出可以直接進行連接,通常結合線程使用

~~~
package com.lgl.hellojava;
import java.io.IOException;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;
public class HelloJJAVA {
public static void main(String[] args) {
PipedInputStream pi = new PipedInputStream();
PipedOutputStream po = new PipedOutputStream();
try {
pi.connect(po);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Read read = new Read(pi);
Write write = new Write(po);
new Thread(read).start();
new Thread(write).start();
}
}
class Read implements Runnable {
private PipedInputStream in;
public Read(PipedInputStream in) {
this.in = in;
}
@Override
public void run() {
try {
byte[] buf = new byte[1024];
int len = in.read(buf);
String s = new String(buf, 0, len);
System.out.println(s);
in.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class Write implements Runnable {
private PipedOutputStream out;
public Write(PipedOutputStream out) {
this.out = out;
}
@Override
public void run() {
try {
out.write("yes".getBytes());
out.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 現在就可以聯通了
## 六.RandomAccessFile
> 這是一個很特別的家伙,他繼承的是Object

* 該類不是IO體系中的子類
* 但是他是IO包中的成員,他同時具備讀寫功能
* 內部封裝了一個數組,而且通過指針對數組中的元素進行操作
* 可以通過getFilePointer或者指針位置同時可以通過seek改變指針的位置
> 其實完成讀寫的原理就是內部封裝了字節輸入,輸出流,通過構造函數可以看出該類具備操作文件的能力,而且操作文件還有模式
~~~
package com.lgl.hellojava;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.RandomAccessFile;
public class HelloJJAVA {
public static void main(String[] args) {
writeFile();
}
public static void writeFile() {
try {
RandomAccessFile raf = new RandomAccessFile("1.txt", "rw");
raf.write("zhangssan".getBytes());
raf.writeInt(99);
raf.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 這只是一個寫的過程,我們的特性還沒有體現出來,我們來看下他是怎么讀的
~~~
private static void ReadFile() {
try {
RandomAccessFile raf = new RandomAccessFile("1.txt", "r");
// 調整對象指針
raf.seek(8 * 0);
byte[] buf = new byte[1024];
raf.read(buf);
String s = new String(buf);
System.out.println(s);
raf.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
~~~
> 如果操作的文件不存在,就會自動創建,如果存在,直接覆蓋
## 七.IO其他類
> IO其他包里,還有一些其他的使用
* 操作基本數據類型
* 字節數組
* 字符數組
### 1.基本數據類型
> 我們先來看下基本數據類型的,我直接讀寫都寫出來
~~~
public static void readData() {
try {
DataInputStream dis = new DataInputStream(new FileInputStream(
"data.txt"));
int num = dis.readInt();
boolean d = dis.readBoolean();
System.out.println(num + "" + d + "");
dis.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
// 基本數據類型
public static void writeDatea() {
try {
DataOutputStream dos = new DataOutputStream(new FileOutputStream(
"data.txt"));
dos.writeInt(55);
dos.writeBoolean(true);
dos.close();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
~~~
> 讀取基本數據類型
## 二.字節數組
> 這個偶爾用下,但是我們還是要去學習一下的
* ByteArrayInputStream: 在構造的時候,需要接收數據源,而且數據源是一個字節數組
* ByteArrayOutputStream:在構造的時候,不用定義數據的目的,因為該對象中已經內部封裝了可變程度的字節數組
> 因為這兩個流的對象都操作的數組,并且沒有使用系統資源,所以,不用進行close關閉!
~~~
package com.lgl.hellojava;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
public class HelloJJAVA {
public static void main(String[] args) {
// 數據源
ByteArrayInputStream bis = new ByteArrayInputStream("123".getBytes());
// 數據目的
ByteArrayOutputStream bos = new ByteArrayOutputStream();
int ch = 0;
while ((ch = bis.read()) != -1) {
bos.write(ch);
}
System.out.println(bos.size());
System.out.println(bos.toString());
}
}
~~~
> 是不是比較簡單,字符流我就不說了,和字節流是類似的,但是他會衍生出一個新的內容就是字符編碼
## 八.字符編碼
> 這些亂七八糟的編碼說真的,東西還是挺多的,但是我并不想講那么多,因為本身我也不是很清楚,其次這玩意大致的知道一點就可以了(個人覺得),什么ASCLL,GBK,UTF-8之類的
~~~
String s = "hello java";
byte [] b1 = s.getBytes();
System.out.println(Arrays.toString(b1));
~~~
> 他得到的是什么?

> 現在我換一個編碼
~~~
byte [] b1 = s.getBytes("GBK");
~~~
> 我設置成GBK,他的值是一樣的,說明默認的是GBK
## 九.練習
> 寫完這個練習,我們的I/O流就GG了,感慨頗多,寫I/O寫了很多篇,但是仍然暈乎乎的,應該繼續加深印象的,我們來看下需求
* 有五個學生,每個學生有三門課程,從鍵盤輸入以下數據(包括姓名,三門課成績),輸入的格式,如zhangsan,30,60,96計算出總成績,并把學生的信息和計算出的總分數高低順序存入文件student.txt中去;
> 好了,開始擼代碼了
~~~
package com.lgl.hellojava;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Set;
import java.util.TreeSet;
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 邏輯:
* 1.通過獲取鍵盤錄入一行數據,然后取出數據封裝成學生對象
* 2.需要從高到低排列,需要排序,使用集合TreeSet就可以了
* 3.寫入文件
*/
try {
Set<Student> stus = StudentInfoTool.getStudents();
StudentInfoTool.writeFile(stus);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
// 比較數據,Comparable接口
class Student implements Comparable<Student> {
// 姓名
private String name;
// 三科分數
private int n1, n2, n3;
// 總分
private int sum;
// 構造函數
public Student(String name, int n1, int n2, int n3) {
this.name = name;
this.n1 = n1;
this.n2 = n2;
this.n3 = n3;
sum = n1 + n2 + n3;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getSum() {
return sum;
}
public void setSum(int sum) {
this.sum = sum;
}
@Override
public int compareTo(Student o) {
int num = new Integer(this.sum).compareTo(new Integer(o.sum));
if (num == 0) {
return this.name.compareTo(o.name);
}
return num;
}
@Override
public int hashCode() {
// TODO Auto-generated method stub
return name.hashCode() + sum * 78;
}
@Override
public boolean equals(Object obj) {
if (!(obj instanceof Student)) {
throw new ClassCastException("Type Error");
}
Student s = (Student) obj;
return this.name.equals(s.name) && this.sum == s.sum;
}
}
// 工具類
class StudentInfoTool {
public static Set<Student> getStudents() throws IOException {
BufferedReader bufr = new BufferedReader(new InputStreamReader(
System.in));
String line = null;
// 集合
Set<Student> stus = new TreeSet<>();
while ((line = bufr.readLine()) != null) {
if (line.equals("over")) {
break;
}
String[] info = line.split(",");
Student student = new Student(info[0], Integer.parseInt(info[1]),
Integer.parseInt(info[2]), Integer.parseInt(info[3]));
stus.add(student);
}
bufr.close();
return stus;
}
// 寫入文件
public static void writeFile(Set<Student> stus) {
try {
BufferedWriter bufw = new BufferedWriter(new FileWriter(
"student.txt"));
for (Student stu : stus) {
bufw.write(stu.getName());
bufw.write(stu.getSum() + "");
bufw.newLine();
bufw.flush();
}
bufw.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
~~~
> 輸出的結果,可以看到

> 總數,從小到大,如果你想從高到低的話,自己去實現下排序,看看你們學的怎么樣了
>
> 好的,我們本篇到這里就結束了,這篇也是憋了挺久的才別出來的,終于把IO寫完了
## 有興趣的可以加群:555974449
版權聲明:本文為博主原創文章,博客地址:http://blog.csdn.net/qq_26787115,未經博主允許不得轉載。
- 0-發現
- AndroidInterview-Q-A
- Android能讓你少走彎路的干貨整理
- LearningNotes
- temp
- temp11
- 部分地址
- 0-待辦任務
- 待補充列表
- 0-未分類
- AndroidView事件分發與滑動沖突處理
- Spannable
- 事件分發機制詳解
- 1-Java
- 1-Java-01基礎
- 未歸檔
- 你應該知道的JDK知識
- 集合框架
- 1-Java-04合集
- Java之旅0
- Java之旅
- JAVA之旅01
- JAVA之旅02
- JAVA之旅03
- JAVA之旅04
- JAVA之旅05
- JAVA之旅06
- JAVA之旅07
- JAVA之旅08
- JAVA之旅09
- java之旅1
- JAVA之旅10
- JAVA之旅11
- JAVA之旅12
- JAVA之旅13
- JAVA之旅14
- JAVA之旅15
- JAVA之旅16
- JAVA之旅17
- JAVA之旅18
- JAVA之旅19
- java之旅2
- JAVA之旅20
- JAVA之旅21
- JAVA之旅22
- JAVA之旅23
- JAVA之旅24
- JAVA之旅25
- JAVA之旅26
- JAVA之旅27
- JAVA之旅28
- JAVA之旅29
- java之旅3
- JAVA之旅30
- JAVA之旅31
- JAVA之旅32
- JAVA之旅33
- JAVA之旅34
- JAVA之旅35
- 1-Java-05辨析
- HashMapArrayMap
- Java8新特性
- Java8接口默認方法
- 圖解HashMap(1)
- 圖解HashMap(2)
- 2-Android
- 2-Android-1-基礎
- View繪制流程
- 事件分發
- AndroidView的事件分發機制和滑動沖突解決
- 自定義View基礎
- 1-安卓自定義View基礎-坐標系
- 2-安卓自定義View基礎-角度弧度
- 3-安卓自定義View基礎-顏色
- 自定義View進階
- 1-安卓自定義View進階-分類和流程
- 10-安卓自定義View進階-Matrix詳解
- 11-安卓自定義View進階-MatrixCamera
- 12-安卓自定義View進階-事件分發機制原理
- 13-安卓自定義View進階-事件分發機制詳解
- 14-安卓自定義View進階-MotionEvent詳解
- 15-安卓自定義View進階-特殊形狀控件事件處理方案
- 16-安卓自定義View進階-多點觸控詳解
- 17-安卓自定義View進階-手勢檢測GestureDetector
- 2-安卓自定義View進階-繪制基本圖形
- 3-安卓自定義View進階-畫布操作
- 4-安卓自定義View進階-圖片文字
- 5-安卓自定義View進階-Path基本操作
- 6-安卓自定義View進階-貝塞爾曲線
- 7-安卓自定義View進階-Path完結篇偽
- 8-安卓自定義View進階-Path玩出花樣PathMeasure
- 9-安卓自定義View進階-Matrix原理
- 通用類介紹
- Application
- 2-Android-2-使用
- 2-Android-02控件
- ViewGroup
- ConstraintLayout
- CoordinatorLayout
- 2-Android-03三方使用
- Dagger2
- Dagger2圖文完全教程
- Dagger2最清晰的使用教程
- Dagger2讓你愛不釋手-終結篇
- Dagger2讓你愛不釋手-重點概念講解、融合篇
- dagger2讓你愛不釋手:基礎依賴注入框架篇
- 閱讀筆記
- Glide
- Google推薦的圖片加載庫Glide:最新版使用指南(含新特性)
- rxjava
- 這可能是最好的RxJava2.x入門教程完結版
- 這可能是最好的RxJava2.x入門教程(一)
- 這可能是最好的RxJava2.x入門教程(三)
- 這可能是最好的RxJava2.x入門教程(二)
- 這可能是最好的RxJava2.x入門教程(五)
- 這可能是最好的RxJava2.x入門教程(四)
- 2-Android-3-優化
- 優化概況
- 各種優化
- Android端秒開優化
- apk大小優化
- 內存分析
- 混淆
- 2-Android-4-工具
- adb命令
- 一鍵分析Android的BugReport
- 版本控制
- git
- git章節簡述
- 2-Android-5-源碼
- HandlerThread 源碼分析
- IntentService的使用和源碼分析
- 2-Android-9-辨析
- LRU算法
- 什么是Bitmap
- 常見圖片壓縮方式
- 3-Kotlin
- Kotlin使用筆記1-草稿
- Kotlin使用筆記2
- kotlin特性草稿
- Kotlin草稿-Delegation
- Kotlin草稿-Field
- Kotlin草稿-object
- 4-JavaScript
- 5-Python
- 6-Other
- Git
- Gradle
- Android中ProGuard配置和總結
- gradle使用筆記
- Nexus私服搭建
- 編譯提速最佳實踐
- 7-設計模式與架構
- 組件化
- 組件化探索(OKR)
- 1-參考列表
- 2-1-組件化概述
- 2-2-gradle配置
- 2-3-代碼編寫
- 2-4-常見問題
- 2-9-值得一讀
- 8-數據結構與算法
- 0臨時文件
- 漢諾塔
- 8-數據-1數據結構
- HashMap
- HashMap、Hashtable、HashSet 和 ConcurrentHashMap 的比較
- 遲到一年HashMap解讀
- 8-數據-2算法
- 1個就夠了
- Java常用排序算法(必須掌握的8大排序算法)
- 常用排序算法總結(性能+代碼)
- 必須知道的八大種排序算法(java實現)
- 9-職業
- 閱讀
- 書單
- 面試
- 面試-01-java
- Java面試題全集駱昊(上)
- Java面試題全集駱昊(下)
- Java面試題全集駱昊(中)
- 面試-02-android
- 40道Android面試題
- 面試-03-開源源碼
- Android圖片加載框架最全解析(二),從源碼的角度理解Glide的執行流程
- 面試-07-設計模式
- 面試-08-算法
- 面試-09-其他
- SUMMARY
- 版權說明
- temp111