java io系列24之 BufferedWriter(字符缓冲输出流)

简介: 转载请注明出处:http://www.cnblogs.com/skywang12345/p/io_24.html 更多内容请参考:java io系列01之 "目录" BufferedWriter 介绍 BufferedWriter 是缓冲字符输出流。

转载请注明出处:http://www.cnblogs.com/skywang12345/p/io_24.html

更多内容请参考:java io系列01之 "目录"

BufferedWriter 介绍

BufferedWriter 是缓冲字符输出流。它继承于Writer。
BufferedWriter 的作用是为其他字符输出流添加一些缓冲功能。

BufferedWriter 函数列表

复制代码
// 构造函数
BufferedWriter(Writer out) 
BufferedWriter(Writer out, int sz) 
 
void close() // 关闭此流,但要先刷新它。 void flush() // 刷新该流的缓冲。 void newLine() // 写入一个行分隔符。 void write(char[] cbuf, int off, int len) // 写入字符数组的某一部分。 void write(int c) // 写入单个字符。 void write(String s, int off, int len) // 写入字符串的某一部分。
复制代码

 

BufferedWriter 源码分析(基于jdk1.7.40)

复制代码
  1 package java.io;
  2 
  3 public class BufferedWriter extends Writer {  4  5 // 输出流对象  6 private Writer out;  7  8 // 保存“缓冲输出流”数据的字符数组  9 private char cb[];  10  11 // nChars 是cb缓冲区中字符的总的个数  12 // nextChar 是下一个要读取的字符在cb缓冲区中的位置  13 private int nChars, nextChar;  14  15 // 默认字符缓冲区大小  16 private static int defaultCharBufferSize = 8192;  17  18 // 行分割符  19 private String lineSeparator;  20  21 // 构造函数,传入“Writer对象”,默认缓冲区大小是8k  22 public BufferedWriter(Writer out) {  23 this(out, defaultCharBufferSize);  24  }  25  26 // 构造函数,传入“Writer对象”,指定缓冲区大小是sz  27 public BufferedWriter(Writer out, int sz) {  28 super(out);  29 if (sz <= 0)  30 throw new IllegalArgumentException("Buffer size <= 0");  31 this.out = out;  32 cb = new char[sz];  33 nChars = sz;  34 nextChar = 0;  35  36 lineSeparator = java.security.AccessController.doPrivileged(  37 new sun.security.action.GetPropertyAction("line.separator"));  38  }  39  40 // 确保“BufferedWriter”是打开状态  41 private void ensureOpen() throws IOException {  42 if (out == null)  43 throw new IOException("Stream closed");  44  }  45  46 // 对缓冲区执行flush()操作,将缓冲区的数据写入到Writer中  47 void flushBuffer() throws IOException {  48 synchronized (lock) {  49  ensureOpen();  50 if (nextChar == 0)  51 return;  52 out.write(cb, 0, nextChar);  53 nextChar = 0;  54  }  55  }  56  57 // 将c写入到缓冲区中。先将c转换为char,然后将其写入到缓冲区。  58 public void write(int c) throws IOException {  59 synchronized (lock) {  60  ensureOpen();  61 // 若缓冲区满了,则清空缓冲,将缓冲数据写入到输出流中。  62 if (nextChar >= nChars)  63  flushBuffer();  64 cb[nextChar++] = (char) c;  65  }  66  }  67  68 // 返回a,b中较小的数  69 private int min(int a, int b) {  70 if (a < b) return a;  71 return b;  72  }  73 74 // 将字符数组cbuf写入到缓冲中,从cbuf的off位置开始写入,写入长度是len。 75 public void write(char cbuf[], int off, int len) throws IOException { 76 synchronized (lock) { 77 ensureOpen(); 78 if ((off < 0) || (off > cbuf.length) || (len < 0) || 79 ((off + len) > cbuf.length) || ((off + len) < 0)) { 80 throw new IndexOutOfBoundsException(); 81 } else if (len == 0) { 82 return; 83 } 84 85 if (len >= nChars) { 86 /* If the request length exceeds the size of the output buffer, 87 flush the buffer and then write the data directly. In this 88 way buffered streams will cascade harmlessly. */ 89 flushBuffer(); 90 out.write(cbuf, off, len); 91 return; 92 } 93 94 int b = off, t = off + len; 95 while (b < t) { 96 int d = min(nChars - nextChar, t - b); 97 System.arraycopy(cbuf, b, cb, nextChar, d); 98 b += d; 99 nextChar += d; 100 if (nextChar >= nChars) 101 flushBuffer(); 102 } 103 } 104 } 105 106 // 将字符串s写入到缓冲中,从s的off位置开始写入,写入长度是len。 107 public void write(String s, int off, int len) throws IOException { 108 synchronized (lock) { 109 ensureOpen(); 110 111 int b = off, t = off + len; 112 while (b < t) { 113 int d = min(nChars - nextChar, t - b); 114 s.getChars(b, b + d, cb, nextChar); 115 b += d; 116 nextChar += d; 117 if (nextChar >= nChars) 118 flushBuffer(); 119 } 120 } 121 } 122 123 // 将换行符写入到缓冲中 124 public void newLine() throws IOException { 125 write(lineSeparator); 126 } 127 128 // 清空缓冲区数据 129 public void flush() throws IOException { 130 synchronized (lock) { 131 flushBuffer(); 132 out.flush(); 133 } 134 } 135 136 public void close() throws IOException { 137 synchronized (lock) { 138 if (out == null) { 139 return; 140 } 141 try { 142 flushBuffer(); 143 } finally { 144 out.close(); 145 out = null; 146 cb = null; 147 } 148 } 149 } 150 }
复制代码

说明: BufferedWriter的源码非常简单,这里就BufferedWriter的思想进行简单说明:BufferedWriter通过字符数组来缓冲数据,当缓冲区满或者用户调用flush()函数时,它就会将缓冲区的数据写入到输出流中。

 

示例代码

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

复制代码
 1 import java.io.BufferedWriter;
 2 import java.io.File;  3 import java.io.OutputStream;  4 import java.io.FileWriter;  5 import java.io.IOException;  6 import java.io.FileNotFoundException;  7 import java.lang.SecurityException;  8 import java.util.Scanner;  9 10 /** 11  * BufferedWriter 测试程序 12  * 13  * @author skywang 14 */ 15 public class BufferedWriterTest { 16 17 private static final int LEN = 5; 18 // 对应英文字母“abcdefghijklmnopqrstuvwxyz” 19 //private static final char[] ArrayLetters = "abcdefghijklmnopqrstuvwxyz"; 20 private static final char[] ArrayLetters = new char[] {'a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','w','x','y','z'}; 21 22 public static void main(String[] args) { 23  testBufferedWriter() ; 24  } 25 26 /** 27  * BufferedWriter的API测试函数 28 */ 29 private static void testBufferedWriter() { 30 31 // 创建“文件输出流”对应的BufferedWriter 32 // 它对应缓冲区的大小是16,即缓冲区的数据>=16时,会自动将缓冲区的内容写入到输出流。 33 try { 34 File file = new File("bufferwriter.txt"); 35 BufferedWriter out = 36 new BufferedWriter( 37 new FileWriter(file)); 38 39 // 将ArrayLetters数组的前10个字符写入到输出流中 40 out.write(ArrayLetters, 0, 10); 41 // 将“换行符\n”写入到输出流中 42 out.write('\n'); 43 44  out.flush(); 45 //readUserInput() ; 46 47  out.close(); 48 } catch (FileNotFoundException e) { 49  e.printStackTrace(); 50 } catch (SecurityException e) { 51  e.printStackTrace(); 52 } catch (IOException e) { 53  e.printStackTrace(); 54  } 55  } 56 57 /** 58  * 读取用户输入 59 */ 60 private static void readUserInput() { 61 System.out.println("please input a text:"); 62 Scanner reader=new Scanner(System.in); 63 // 等待一个输入 64 String str = reader.next(); 65 System.out.printf("the input is : %s\n", str); 66  } 67 }
复制代码

运行结果
生成文件“bufferwriter.txt”,文件的内容是“abcdefghij”。

相关文章
|
16天前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
66 23
|
20天前
|
存储 传感器 缓存
java变量与数据类型:整型、浮点型与字符类型
### Java数据类型全景表简介 本文详细介绍了Java的基本数据类型和引用数据类型,涵盖每种类型的存储空间、默认值、取值范围及使用场景。特别强调了`byte`、`int`、`long`、`float`、`double`等基本类型在不同应用场景中的选择与优化,如文件流处理、金融计算等。引用数据类型部分则解析了`String`、数组、类对象、接口和枚举的内存分配机制。
49 15
|
2月前
|
自然语言处理 Java
Java中的字符集编码入门-增补字符(转载)
本文探讨Java对Unicode的支持及其发展历程。文章详细解析了Unicode字符集的结构,包括基本多语言面(BMP)和增补字符的表示方法,以及UTF-16编码中surrogate pair的使用。同时介绍了代码点和代码单元的概念,并解释了UTF-8的编码规则及其兼容性。
121 60
|
1月前
|
缓存 网络协议 Java
JAVA网络IO之NIO/BIO
本文介绍了Java网络编程的基础与历史演进,重点阐述了IO和Socket的概念。Java的IO分为设备和接口两部分,通过流、字节、字符等方式实现与外部的交互。
|
4月前
|
Java
java 中 IO 流
Java中的IO流是用于处理输入输出操作的机制,主要包括字节流和字符流两大类。字节流以8位字节为单位处理数据,如FileInputStream和FileOutputStream;字符流以16位Unicode字符为单位,如FileReader和FileWriter。这些流提供了读写文件、网络传输等基本功能。
81 10
|
5月前
|
Java
Java 中 IO 流的分类详解
【10月更文挑战第10天】不同类型的 IO 流具有不同的特点和适用场景,我们可以根据具体的需求选择合适的流来进行数据的输入和输出操作。在实际应用中,还可以通过组合使用多种流来实现更复杂的功能。
112 0
|
27天前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
160 60
【Java并发】【线程池】带你从0-1入门线程池
|
23天前
|
Java 调度
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
94 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
|
1月前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
136 14
|
1月前
|
安全 Java 程序员
Java 面试必问!线程构造方法和静态块的执行线程到底是谁?
大家好,我是小米。今天聊聊Java多线程面试题:线程类的构造方法和静态块是由哪个线程调用的?构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节有助于掌握Java多线程机制。下期再见! 简介: 本文通过一个常见的Java多线程面试题,详细讲解了线程类的构造方法和静态块是由哪个线程调用的。构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节对掌握Java多线程编程至关重要。
61 13

热门文章

最新文章