1.InputStream

1.1InputStream是所有输入流的超类:

  • int read()

    * 读取下一个字节,并返回字节(0-255)

    * 如果已读到末尾,返回-1

    * read()方法是阻塞(blocking)的,必须等待read()方法返回才能执行下一行代码
  • int read(byte[]):读取若干字节并填充到byte[]数组,返回读取的字节数
  • int read(byte[], int off, int len):指定byte[]数组的偏移量和最大填充术数。
  • void close():关闭输入流
  • 使用try(resource)可以保证InputStream正确关闭

代码一,使用close关闭文件:如果运行时发生IO错误,文件不能正确关闭,资源不能得到及时的释放

public class Main {
public static void main(String[] args) throws IOException {
InputStream input = new FileInputStream("./src/main/java/com/testList/Person.txt");
int n;
while((n=input.read())!= -1){
System.out.println(n);
}
input.close();
}
}

代码二:

public class Main {
public static void main(String[] args) throws IOException {
InputStream input = null;
try{
input= new FileInputStream("./src/main/java/com/testList/Person.txt");
int n;
while((n=input.read())!= -1){
System.out.println(n);
}
}finally {
if (input != null) {
input.close();
}
}
}
}

代码三,使用try(resource)保证InputStream正确关闭,推荐

public class Main {
public static void main(String[] args) throws IOException {
try(InputStream input = new FileInputStream("./src/main/java/com/testList/Person.txt")){
int n;
while((n=input.read())!= -1){
System.out.println(n);
}
}//自动关闭InputStream
}
}

代码四,利用缓冲区一次读取多个字节

public class Main {
public static void main(String[] args) throws IOException {
try(InputStream input = new FileInputStream("./src/main/java/com/testList/Person.txt")){
byte[] buffer = new byte[10];
int n;
while((n=input.read(buffer))!= -1){
System.out.println(Arrays.toString(buffer));
}
}
}
}

1.2常用InputStream

1.2.1 FileInputStream

FileInputStream是InputStream的实现类,可以从文件获取输入流。

    try(InputStream input = new FileInputStream("./src/main/java/com/testList/Person.txt")){
byte[] buffer = new byte[10];
int n;
while((n=input.read(buffer))!= -1){
System.out.println(Arrays.toString(buffer));
}

1.2.2 ByteArrayInputStream

ByteArrayInputStream可以在内存中模拟一个InputStream。用的不多,可以测试的时候构造InputStream

public class Main {
public static void main(String[] args) throws IOException {
byte[] data = {-26, -103, -82, -23, -128, -102, -27, -83, -105, -25,-84, -90, -28, -72, -78, 10, 99, 111, 109, 46,116, 101, 115, 116, 76, 105, 115, 116, 46, 77,97, 105, 110, 64, 50, 98, 49, 57, 51, 102,50, 100, 10, 64, 50, 98, 49, 57, 51, 102};
try(InputStream input = new ByteArrayInputStream(data)){
byte[] buffer = new byte[10];
int n;
while((n=input.read(buffer))!= -1){
System.out.println(Arrays.toString(buffer));
}
}
}
}

1.3总结:

  • InputStream定义了所有输入流的超类
  • FileInputStream实现了文件输入
  • ByteArrayInputStream在内存中模拟一个字节流输入
  • 使用try(resource)保证InputStream正确关闭

2.OutputStream

2.1 java.io.OutPutStream是所有输出流的超类:

  • abstract write(int b):写入一个字节
  • void write(byte[] b):写入byte数组的所有字节
  • void write(byte[] b, int off, int len):写入byte[]数组指定范围的字节
  • write()方法是阻塞的,必须等待write方法执行完毕返回后才能执行下一行代码
  • void close():关闭输出流
  • 使用try(resource)可以保证OutputStream正确关闭
  • void flush() :将缓冲区的内容输出

    为什么需要flush呢?

    因为像磁盘、网络写入数据的时候,出于效率的考虑,很多时候,并不是输出1个字节就立即写入。而是先把输出的字节放在内存缓冲区里,等到缓冲区满了之后,再一次性写入。对于很多设备来说,一次写入1个字节和写入1000个字节话费的时间是一样的。所以Output Stream有一个flush方法,能够强制把缓冲区的内容输出。通常情况下,我们不需要调用这个方法,因为缓冲区在满的时候,会自动调用flush。我们在调用close方法关闭OutputStream时,也会调用flush方法。

代码一:如果写入过程中发生IO错误,OutputStream不能正常关闭

public class Main {
public static void main(String[] args) throws IOException {
OutputStream output = new FileOutputStream("./src/main/java/com/testList/output.txt");
output.write(72);//1次写入1个字节
output.write(101);
output.write(108);
output.write(108);
output.write(111);
output.close();
}
}

代码二:通过try(resource)自动关闭文件

public class Main {
public static void main(String[] args) throws IOException {
try(OutputStream output = new FileOutputStream("./src/main/java/com/testList/output.txt")){
output.write(72);
output.write(101);
output.write(108);
output.write(108);
output.write(111);
}
}
}

代码三:一次传入多个字节

public class Main {
public static void main(String[] args) throws IOException {
try(OutputStream output = new FileOutputStream("./src/main/java/com/testList/output.txt")){
byte[] b = "hello,张三".getBytes("UTF-8");
output.write(b,3,9);
}
}
}

代码四:一次性写入

public class Main {
public static void main(String[] args) throws IOException {
try(OutputStream output = new FileOutputStream("./src/main/java/com/testList/output.txt")){
byte[] b = "hello,张三".getBytes("UTF-8");
output.write(b);
}
}
}

2.2 常用OutPutStream:

2.2.1 FileOutStream

FileOutStream可以输出到文件

2.2.2 ByteArrayOutPutStream

ByteArrayOutputStream可以在内存中模拟一个OutputStream

public class Main {
public static void main(String[] args) throws IOException {
try(ByteArrayOutputStream output = new ByteArrayOutputStream()){
output.write("Hello".getBytes("utf-8"));
output.write("world!".getBytes("utf-8"));
byte[] data = output.toByteArray();
System.out.println(Arrays.toString(data));
}
}
}

2.3 总结

  • OutputStream定义了所有输出流的超类
  • FileOutputStream实现了文件流输出
  • ByteArrayOutputStream在内存中模拟一个字节流的输出
  • 使用try(resource)保证OutputStream正确关闭

3.Input/OutPut练习

FileInputStream可以从文件读取数据,FileOutputStream可以把数据写入文件。

如果我们一边从一个文件读取数据,一边把数据写入到另一个文件,就完成了文件的拷贝。

请编写一个程序,接收两个命令行参数,分别表示源文件和目标文件,然后用InputSream/OutputStream把源文件复制到目标文件。

复制后,请检查源文件和目标文件是否相同(文件长度相同,内容相同),分别用文本文件、图片文件和zip文件测试。

使用FileInputStream读取文件

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException; public class Main {
public static void main(String[] args) throws IOException {
File f = new File("./src/main/java/com/testList/Person.java");
System.out.println(f.length());
//创建字节输入流
FileInputStream fis = new FileInputStream("./src/main/java/com/testList/Person.java");
//创建竹筒
byte[] bbuf = new byte[100];
//保存实际读取的字节数
int hasRead = 0;
//使用循环重复取水过程
while((hasRead = fis.read(bbuf))>0){
//取出竹筒中的水滴即字节,将字节数组转换成字符串输入
System.out.println(new String(bbuf,0,hasRead));
}
//关闭字节流
fis.close();
}
}

使用FileReader读取文件

import java.io.File;
import java.io.FileReader;
import java.io.IOException; public class Main {
public static void main(String[] args) throws IOException {
File f = new File("./src/main/java/com/testList/Person.java");
System.out.println(f.length());
try(
//创建字符输入流
FileReader fr = new FileReader("./src/main/java/com/testList/Person.java")
){
//创建一个长度为100的竹筒
char[] cbuf = new char[100];
//hasRead用于保存实际读取的字符数
int hasRead = 0;
while((hasRead =fr.read(cbuf))>0){//使用循环重复取水过程
//取出竹筒中的水滴即字符,将字符数组转换为字符串输入
System.out.println(new String(cbuf,0,hasRead));
}
}catch (IOException ex){
ex.printStackTrace();
}
}
}
import java.io.*;

public class Main {
public static void main(String[] args) throws IOException {
try(
//创建字节输入流
FileInputStream fis = new FileInputStream("./src/main/java/com/testList/Person.java");
//创建字节输出流
FileOutputStream fos = new FileOutputStream("./src/main/java/com/testList/Person.txt")
){
byte[] bbuf = new byte[300];
int hasRead = 0;
//循环从输入流中取出数据
while ((hasRead = fis.read(bbuf))>0){
//取出1次,写入1次
fos.write(bbuf,0,hasRead);
}
}catch (IOException ex){
ex.printStackTrace();
}
}
}
import java.io.*;

public class Main {
public static void main(String[] args) throws IOException {
try(
//创建字节输出流
FileWriter fw = new FileWriter("./src/main/java/com/testList/Person.txt")
){
fw.write("于易水送人 - 骆宾王\r\n");
fw.write("此地别燕丹,壮士发冲冠。\r\n");
fw.write("昔时人已没,今日水犹寒。\r\n");
}catch (IOException ex){
ex.printStackTrace();
}
}
}
import java.io.*;

public class Main {
public static void main(String[] args) throws IOException {
try(
FileOutputStream fos = new FileOutputStream("./src/main/java/com/testList/Person.txt");
PrintStream ps = new PrintStream(fos)
){
ps.println("普通字符串");
ps.println(new Main());
}catch (IOException ex){
ex.printStackTrace();
}
}
}

廖雪峰Java6IO编程-2input和output-1inputStream的更多相关文章

  1. 廖雪峰Java6IO编程-1IO基础-1IO简介

    1.IO简介 IO是指Input/Output,即输入和输出: Input指从外部读取数据到内存,例如从磁盘读取,从网络读取. * 为什么要把数据读到内存才能处理这些数据呢? * 因为代码是在内存中运 ...

  2. 廖雪峰Java15JDBC编程-3JDBC接口-5JDBC连接池

    1. JDBC连接池 1.1 JDBC连接池简介 线程池可以复用一个线程,这样大量的小任务通过线程池的线程执行,就可以避免反复创建线程带来的开销. 同样JDBC可以复用一个JDBC连接 JDBC的连接 ...

  3. 廖雪峰Java15JDBC编程-3JDBC接口-4JDBC事务

    1 数据库事务:Transaction 1.1 定义 若干SQL语句构成的一个操作序列 要么全部执行成功 要么全部执行不成功 1.2 数据库事务具有ACID特性: Atomicity:原子性 一个事务 ...

  4. 廖雪峰Java15JDBC编程-3JDBC接口-3JDBC更新

    使用update语句的时候,需要通过JDBC实现update语句的执行,这个时候仍然通过PreparedStatement对象来使用,直接传入update语句,然后通过setObject传入占位符的值 ...

  5. 廖雪峰Java15JDBC编程-3JDBC接口-2JDBC查询

    我们可以使用JDBC查询来执行select语句. 1. Statement try(Connection conn = DriverManager.getConnection(JDBC_URL, JD ...

  6. 廖雪峰Java15JDBC编程-3JDBC接口-1JDBC简介

    JDBC:Java DataBase Connectivity Java程序访问数据库的标准接口 使用Java程序访问数据库的时候,Java代码并不是直接通过TCP连接去访问数据库,而是通过JDBC接 ...

  7. 廖雪峰Java15JDBC编程-2SQL入门-2insert/select/update/delete

    1. INSERT用于向数据库的表中插入1条记录 insert into 表名 (字段1,字段2,...) values (数据1,数据2,数据3...) 示例 -- 如果表存在,就删除 drop t ...

  8. 廖雪峰Java15JDBC编程-2SQL入门-1SQL介绍

    1.SQL:结构化查询语言 Structured Query Language 针对关系数据库设计 各种数据库基本一致 允许用户通过SQL查询数据而不关心数据库底层存储结构 1.1 SQL使用: 可以 ...

  9. 廖雪峰Java15JDBC编程-1关系数据库基础-1关系数据库简介

    1.数据库 1.1 定义 数据库是按照数据结构来组合.存储和管理数据的软件. 1.2 数据库模型 数据库有层次模型.网状模型.关系模型三种模型. 2 关系数据库 关系数据库是建立在关系模型上的数据库, ...

随机推荐

  1. (22)Ajax的基本使用(实现登录功能和局部刷新以及防止跨站请求伪造攻击)

    Ajax的作用 前后端分离的项目,需要交互,就要通过Ajax来完成交互 AJAX(Asynchronous Javascript And XML)翻译成中文就是“异步Javascript和XML”.即 ...

  2. (21)模型层 -ORM之msql 聚合查询,F和Q(与、或、非查询)、分组查询

    什么是聚合查询,就是使用聚合函数做计算 from django.db.models import Count,Avg,Max,Min   #聚合函数要从模块中导入 from django.db.mod ...

  3. 树莓派ssh服务

    从官网下载的镜像更新raspberry pi 3 B,但默认是不支持SSH的,即不可外部通过SSH登陆到板子里. 解决办法很简单,在SD卡的根目录下创建一个"ssh"的文件夹即可.

  4. python------软件目录结构规范

    一. 目录结构 www.cnblogs.com/alex3714/articles/5765046.html print(__file__) 获得相对路径 import osprint(os.path ...

  5. Map 按Key排序 和 按Value排序

    https://www.cnblogs.com/binz/p/6671917.html 一.根据value排序 通用方法 public class MapUtil { public static &l ...

  6. ionic中 ng-repeat下使用ng-model获取不到选中数据问题:

    在这种情况下点击了单选框并不能获取$scope.unitcode的值: <ion-list> <ion-radio ng-repeat="item in userInfos ...

  7. 使用ipns 为ipfs 系统自定义域名

    ipns 可以帮助我们进行寻址操作,但是默认的hashid 还是太长,不好记忆,ipns 同时也支持 基于域名的解析,我们添加txt 记录就可以方便的解决ipfs 文件访问地址难记的问题,使用的是 一 ...

  8. SPA项目中,404页面 和 登陆页面 对应的路由,应该怎样控制?

    SPA项目中,404页面 和 登陆页面 对应的路由,应该怎样控制? 可以这样做: 登陆之前,所有页面跳到 登陆页面:包括随便输入的路由地址. 登陆后,跳到相应页面:随便输入的.不存在的路由地址,才跳到 ...

  9. django使用session报错:no such table: django_session

    Django版本:1.11.15 使用session的代码:request.session['key'] = value 运行后报错:no such table: django_session 解决办 ...

  10. 开源自然语言处理工具包hanlp中CRF分词实现详解

     CRF简介 CRF是序列标注场景中常用的模型,比HMM能利用更多的特征,比MEMM更能抵抗标记偏置的问题. [gerative-discriminative.png] CRF训练 这类耗时的任务,还 ...