DataOutputStream 源码分析
来源:互联网 发布:mac book air i5和i7 编辑:程序博客网 时间:2024/06/05 04:00
转自:http://www.fengfly.com/plus/view-214065-1.html
DataOutputStream 是数据输出流。它继承于FilterOutputStream。
DataOutputStream 是用来装饰其它输出流,将DataOutputStream和DataInputStream输入流配合使用,“允许应用程序以与机器无关方式从底层输入流中读写基本 Java 数据类型”。
public class DataOutputStream extends FilterOutputStream implements DataOutput { // “数据输出流”的字节数 protected int written; // “数据输出流”对应的字节数组 private byte[] bytearr = null; // 构造函数 public DataOutputStream(OutputStream out) { super(out); } // 增加“输出值” private void incCount(int value) { int temp = written + value; if (temp < 0) { temp = Integer.MAX_VALUE; } written = temp; } // 将int类型的值写入到“数据输出流”中 public synchronized void write(int b) throws IOException { out.write(b); incCount(1); } // 将字节数组b从off开始的len个字节,都写入到“数据输出流”中 public synchronized void write(byte b[], int off, int len) throws IOException { out.write(b, off, len); incCount(len); } // 清空缓冲,即将缓冲中的数据都写入到输出流中 public void flush() throws IOException { out.flush(); } // 将boolean类型的值写入到“数据输出流”中 public final void writeBoolean(boolean v) throws IOException { out.write(v ? 1 : 0); incCount(1); } // 将byte类型的值写入到“数据输出流”中 public final void writeByte(int v) throws IOException { out.write(v); incCount(1); } // 将short类型的值写入到“数据输出流”中 // 注意:short占2个字节 public final void writeShort(int v) throws IOException { // 写入 short高8位 对应的字节 out.write((v >>> 8) & 0xFF); // 写入 short低8位 对应的字节 out.write((v >>> 0) & 0xFF); incCount(2); } // 将char类型的值写入到“数据输出流”中 // 注意:char占2个字节 public final void writeChar(int v) throws IOException { // 写入 char高8位 对应的字节 out.write((v >>> 8) & 0xFF); // 写入 char低8位 对应的字节 out.write((v >>> 0) & 0xFF); incCount(2); } // 将int类型的值写入到“数据输出流”中 // 注意:int占4个字节 public final void writeInt(int v) throws IOException { out.write((v >>> 24) & 0xFF); out.write((v >>> 16) & 0xFF); out.write((v >>> 8) & 0xFF); out.write((v >>> 0) & 0xFF); incCount(4); } private byte writeBuffer[] = new byte[8]; // 将long类型的值写入到“数据输出流”中 // 注意:long占8个字节 public final void writeLong(long v) throws IOException { writeBuffer[0] = (byte)(v >>> 56); writeBuffer[1] = (byte)(v >>> 48); writeBuffer[2] = (byte)(v >>> 40); writeBuffer[3] = (byte)(v >>> 32); writeBuffer[4] = (byte)(v >>> 24); writeBuffer[5] = (byte)(v >>> 16); writeBuffer[6] = (byte)(v >>> 8); writeBuffer[7] = (byte)(v >>> 0); out.write(writeBuffer, 0, 8); incCount(8); } // 将float类型的值写入到“数据输出流”中 public final void writeFloat(float v) throws IOException { writeInt(Float.floatToIntBits(v)); } // 将double类型的值写入到“数据输出流”中 public final void writeDouble(double v) throws IOException { writeLong(Double.doubleToLongBits(v)); } // 将String类型的值写入到“数据输出流”中 // 实际写入时,是将String对应的每个字符转换成byte数据后写入输出流中。 public final void writeBytes(String s) throws IOException { int len = s.length(); for (int i = 0 ; i < len ; i++) { out.write((byte)s.charAt(i)); } incCount(len); } // 将String类型的值写入到“数据输出流”中 // 实际写入时,是将String对应的每个字符转换成char数据后写入输出流中。 public final void writeChars(String s) throws IOException { int len = s.length(); for (int i = 0 ; i < len ; i++) { int v = s.charAt(i); out.write((v >>> 8) & 0xFF); out.write((v >>> 0) & 0xFF); } incCount(len * 2); } // 将UTF-8类型的值写入到“数据输出流”中 public final void writeUTF(String str) throws IOException { writeUTF(str, this); } // 将String数据以UTF-8类型的形式写入到“输出流out”中 static int writeUTF(String str, DataOutput out) throws IOException { //获取String的长度 int strlen = str.length(); int utflen = 0; int c, count = 0; // 由于UTF-8是1~4个字节不等; // 这里,根据UTF-8首字节的范围,判断UTF-8是几个字节的。 for (int i = 0; i < strlen; i++) { c = str.charAt(i); if ((c >= 0x0001) && (c <= 0x007F)) { utflen++; } else if (c > 0x07FF) { utflen += 3; } else { utflen += 2; } } if (utflen > 65535) throw new UTFDataFormatException( "encoded string too long: " + utflen + " bytes"); // 新建“字节数组bytearr” byte[] bytearr = null; if (out instanceof DataOutputStream) { DataOutputStream dos = (DataOutputStream)out; if(dos.bytearr == null || (dos.bytearr.length < (utflen+2))) dos.bytearr = new byte[(utflen*2) + 2]; bytearr = dos.bytearr; } else { bytearr = new byte[utflen+2]; } // “字节数组”的前2个字节保存的是“UTF-8数据的长度” bytearr[count++] = (byte) ((utflen >>> 8) & 0xFF); bytearr[count++] = (byte) ((utflen >>> 0) & 0xFF); // 对UTF-8中的单字节数据进行预处理 int i=0; for (i=0; i<strlen; i++) { c = str.charAt(i); if (!((c >= 0x0001) && (c <= 0x007F))) break; bytearr[count++] = (byte) c; } // 对预处理后的数据,接着进行处理 for (;i < strlen; i++){ c = str.charAt(i); // UTF-8数据是1个字节的情况 if ((c >= 0x0001) && (c <= 0x007F)) { bytearr[count++] = (byte) c; } else if (c > 0x07FF) { // UTF-8数据是3个字节的情况 bytearr[count++] = (byte) (0xE0 | ((c >> 12) & 0x0F)); bytearr[count++] = (byte) (0x80 | ((c >> 6) & 0x3F)); bytearr[count++] = (byte) (0x80 | ((c >> 0) & 0x3F)); } else { // UTF-8数据是2个字节的情况 bytearr[count++] = (byte) (0xC0 | ((c >> 6) & 0x1F)); bytearr[count++] = (byte) (0x80 | ((c >> 0) & 0x3F)); } } // 将字节数组写入到“数据输出流”中 out.write(bytearr, 0, utflen+2); return utflen + 2; } public final int size() { return written; } }
0 0
- DataOutputStream 源码分析
- bytearrayoutputstream dataoutputstream分析
- DataInputStream和DataOutputStream源码理解
- Java8 I/O源码-DataInputStream与DataOutputStream
- 10 DataOutputStream
- DataOutputStream、DataInputStream
- DataInputStream/DataOutputStream
- DataOutputStream/RandomAccessFile
- 数据流DataOutputStream
- Java_io体系之DataInputStream、DataOutputStream简介、走进源码及示例——08
- java io系列15之 DataOutputStream(数据输出流)的认知、源码和示例
- java io系列15之 DataOutputStream(数据输出流)的认知、源码和示例
- 源码分析
- 源码分析
- 源码分析
- 源码分析
- 源码分析
- 源码分析
- codeforces 224 B. Number Busters
- 二叉树的遍历非递归算法
- pb 中keydown(keycode)用法
- 关于Unix/Linux下查看命令行历史记录(history/.bash_history)
- 使用Eval()绑定数据时使用三元运算符
- DataOutputStream 源码分析
- SQL Server2005 表分区(第四章)
- vc 整形转字符串型和字符串转整形
- PB中DATAWINDOW实时滚动实时检索的问题
- Ruby命令行参数介绍
- 简单的并发编程中犯2的一个小例子--CAS使用时一定要考虑下是否有必要做轮询
- pb 快捷键
- 游戏服务器开发中的一点值得注意的地方
- 36条互联网创业建议之软件工程:精简研发团队