DataOutputStream 源码分析

转自: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
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
主要特性Java 语言是简单的:Java 语言的语法与 C 语言和 C++ 语言很接近,使得大多数程序员很容易学习和使用。另一方面,Java 丢弃了 C++ 中很少使用的、很难理解的、令人迷惑的那些特性,如操作符重载、多继承、自动的强制类型转换。特别地,Java 语言不使用指针,而是引用。并提供了自动分配和回收内存空间,使得程序员不必为内存管理而担忧。Java 语言是面向对象的:Java 语言提供类、接口和继承等面向对象的特性,为了简单起见,只支持类之间的单继承,但支持接口之间的多继承,并支持类与接口之间的实现机制(关键字为 implements)。Java 语言全面支持动态绑定,而 C++语言只对虚函数使用动态绑定。总之,Java语言是一个纯的面向对象程序设计语言。Java语言是分布式的:Java 语言支持 Internet 应用的开发,在基本的 Java 应用编程接口中有一个网络应用编程接口(java net),它提供了用于网络应用编程的类库,包括 URL、URLConnection、Socket、ServerSocket 等。Java 的 RMI(远程方法激活)机制也是开发分布式应用的重要手段。Java 语言是健壮的:Java 的强类型机制、异常处理、垃圾的自动收集等是 Java 程序健壮性的重要保证。对指针的丢弃是 Java 的明智选择。Java 的安全检查机制使得 Java 更具健壮性。Java语言是安全的:Java通常被用在网络环境中,为此,Java 提供了一个安全机制以防恶意代码的攻击。除了Java 语言具有的许多安全特性以外,Java 对通过网络下载的类具有一个安全防范机制(类 ClassLoader),如分配不同的名字空间以防替代本地的同名类、字节代码检查,并提供安全管理机制(类 SecurityManager)让 Java 应用设置安全哨兵。Java 语言是体系结构中立的:Java 程序(后缀为 java 的文件)在 Java 平台上被编译为体系结构中立的字节码格式(后缀为 class 的文件),然后可以在实现这个 Java 平台的任何系统中运行。这种途径适合于异构的网络环境和软件的分发。Java 语言是可移植的:这种可移植性来源于体系结构中立性,另外,Java 还严格规定了各个基本数据类型的长度。Java 系统本身也具有很强的可移植性,Java 编译器是用 Java 实现的,Java 的运行环境是用 ANSI C 实现的。Java 语言是解释型的:如前所述,Java 程序在 Java 平台上被编译为字节码格式,然后可以在实现这个 Java 平台的任何系统中运行。在运行时,Java 平台中的 Java 解释器对这些字节码进行解释执行,执行过程中需要的类在联接阶段被载入到运行环境中。Java 是高性能的:与那些解释型的高级脚本语言相比,Java 的确是高性能的。事实上,Java 的运行速度随着 JIT(Just-In-Time)编译器技术的发展越来越接近于 C++。Java 语言是多线程的:在 Java 语言中,线程是一种特殊的对象,它必须由 Thread 类或其子(孙)类来创建。通常有两种方法来创建线程:其一,使用型构为 Thread(Runnable) 的构造子类将一个实现了 Runnable 接口的对象包装成一个线程,其二,从 Thread 类派生出子类并重写 run 方法,使用该子类创建的对象即为线程。值得注意的是 Thread 类已经实现了 Runnable 接口,因此,任何一个线程均有它的 run 方法,而 run 方法中包含了线程所要运行的代码。线程的活动由一组方法来控制。Java 语言支持多个线程的同时执行,并提供多线程之间的同步机制(关键字为 synchronized)。Java 语言是动态的:Java 语言的设计目标之一是适应于动态变化的环境。Java 程序需要的类能够动态地被载入到运行环境,也可以通过网络来载入所需要的类。这也有利于软件的升级。另外,Java 中的类有一个运行时刻的表示,能进行运行时刻的类型检查。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值