Java-IO流篇-DataOutputStream

2023-11-02

DataOutputStream

DataOutputStreams是OutputStream的子类。
是数据输出流,此类继承自FillterOutputStream类,同时实现DataOutput接口。在DataOutput接口定义了一系列写入各种数据的方法。
从类 java.io.FilterOutputStream 继承的方法有:close,write
从接口 java.io.DataOutput 继承的方法有:write
字节处理流,提供了多个很常用的方法。
是对普通流的功能的一个扩展,可以更加方便地读取int,long,字符等类型数据 。

数据流: DataInputStream 和 DataOutputStream
  • 1). DataInputStream 数据的字节输入流; DataOutputStream 数据的字节输出流。
  • 2). 功能: 实现八种基本类型数据的输入/输出。 同时,也可实现字符串的输入/输出。
  • 3). 特点: 八种基本类型的数据在输入/输出时,会保持类型不变。
  • 因此,这种流特别适合在网络上实现基本类型数据和字符串的传递。
  • 4). 方法:
  • readByte() writeByte()
  • readShort() writeShort();
  • readInt() writeInt();
  • readLong() writeLong();
  • readFloat() writeFloat();
  • readDouble() writeDouble();
  • readBoolean() writeBoolean();
  • readChar() writeChar();
  • readUTF(); writeUTF();
  • 5). 数据流属于处理流,它必须套接在节点流。
  • 6). 注意: 数据流在读取与存储时的顺序要一致。否则,读取数据会失真。
对DataOutputStream源码的分析
package java.io;
public class DataOutputStream extends FilterOutputStream implements DataOutput {
     //字节数,该变量用于记录实际写入数据的字节数,当写入的数据超过int型数据的表达范围时,该值将被赋予Integer.MAX_VALUE
    protected int written;
    
     //对应的字节数组
    private byte[] bytearr = null;

     //构造函数:一个带一个参数的构造方法,传入的参数是一个OutputStream对象,内部调用FilterOutputStream对象的构造参数。
    public DataOutputStream(OutputStream out) {
        super(out);
    }

     //增加written
    private void incCount(int value) {
        int temp = written + value;
        if (temp < 0) {//这里的temp<0的情况其实就是int型变量数据溢出了。
            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();
    }

    public final void writeBoolean(boolean v) throws IOException {
        out.write(v ? 1 : 0);
        incCount(1);
    }

    public final void writeByte(int v) throws IOException {
        out.write(v);
        incCount(1);
    }

     // 注意:short占2个字节
    public final void writeShort(int v) throws IOException {
        out.write((v >>> 8) & 0xFF);// 写入 short高8位 对应的字节
        out.write((v >>> 0) & 0xFF); // 写入 short低8位 对应的字节
        incCount(2);
    }

      // 注意:char占2个字节
    public final void writeChar(int v) throws IOException {
        out.write((v >>> 8) & 0xFF);
        out.write((v >>> 0) & 0xFF);
        incCount(2);
    }

      // 注意: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占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 {
        int strlen = str.length();//获取String的长度
        int utflen = 0;
        int c, count = 0;

		// 由于UTF-8是1~4个字节不等;这里,根据UTF-8首字节的范围,判断UTF-8是几个字节的。
        /* use charAt instead of copying String to char array */
        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)//(65535)10=(1111111111111111)2
            throw new UTFDataFormatException(
                "encoded string too long: " + utflen + " bytes");

        byte[] bytearr = null;// 新建“字节数组bytearr”
        //定义了一个字节数组,用于充当向流中写入数据时的临时缓存。

		//先判断传入的out对象是否是DataOuputStream类或者其子类,如果是,就使用其内部定义的bytearr数组,否则就新建一个byte数组赋值给bytearr。
		//如果使用内置的数组,会先对内置数组进行检测,检测其是否为null和容量是否满足需求,如果不满足也要重新创建。
        if (out instanceof DataOutputStream) {
            DataOutputStream dos = (DataOutputStream)out;
            if(dos.bytearr == null || (dos.bytearr.length < (utflen+2)))
				//这里就是内置数组为null或者容量不够时,重新创建数组对象,新建的容量会根据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中的单字节数据进行预处理,如果是utf8中的单字节数据,则直接装换成byte型写入数据缓存中。
        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);
		//最终返回以utf8格式写入流中的字节数
        return utflen + 2;
    }

     //返回当前写入流中数据的字节总数
    public final int size() {
        return written;
    }
}

为了方便理解,这里附上一幅utf8编码的一些小格式:
UTF-8

最后附上一个小例子:
import java.io.*;

/**
 * @Author: lishi
 * @Description:
 * @Date: Create in 18:40 2019/3/25
 */
public class DataOutputStream0325 {
    public static void main(String args[]) throws Exception{    // 所有异常抛出
        DataOutputStream dos = null ;            // 声明数据输出流对象
        File f = new File("C:\\Users\\lishi\\Desktop\\test.txt") ; // 文件的保存路径
        dos = new DataOutputStream(new FileOutputStream(f)) ;    // 实例化数据输出流对象
        String names[] = {"衬衣","手套","围巾"} ;    // 商品名称
        float prices[] = {98.3f,30.3f,50.5f} ;        // 商品价格
        int nums[] = {3,2,1} ;    // 商品数量
        for(int i=0;i<names.length;i++){    // 循环输出
            dos.writeChars(names[i]) ;    // 写入字符串,注意,这边少数writeChars(),不是writechar()。
            dos.writeChar('\t') ;    // 写入分隔符,这边是读取writechar()。
            dos.writeFloat(prices[i]) ; // 写入价格
            dos.writeChar('\t') ;    // 写入分隔符
            dos.writeInt(nums[i]) ; // 写入数量
            dos.writeChar('\n') ;    // 换行
        }
        dos.close() ;    // 关闭输出流


        DataInputStream dis = null ;        // 声明数据输入流对象
  //      File f = new File("d:" + File.separator + "order.txt") ; // 文件的保存路径
        dis = new DataInputStream(new FileInputStream(f)) ;    // 实例化数据输入流对象
        String name = null ;    // 接收名称
        float price = 0.0f ;    // 接收价格
        int num = 0 ;    // 接收数量
        char temp[] = null ;    // 接收商品名称
        int len = 0 ;    // 保存读取数据的个数
        char c = 0 ;    // '\u0000'
        try{
            while(true){
                temp = new char[200] ;    // 开辟空间
                len = 0 ;
                while((c=dis.readChar())!='\t'){    // 接收内容,因为直到读取到'\t'才完成了读取一个字符串,未读取到表示还有内容。
                    temp[len] = c ;
                    len ++ ;    // 读取长度加1
                }
                name = new String(temp,0,len) ;    // 将字符数组变为String
                price = dis.readFloat() ;    // 读取价格
                dis.readChar() ;    // 读取\t
                num = dis.readInt() ;    // 读取int
                dis.readChar() ;    // 读取\n
                System.out.printf("名称:%s;价格:%5.2f;数量:%d\n",name,price,num) ;
            }
        }catch(Exception e){}
        dis.close() ;
    }
}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Java-IO流篇-DataOutputStream 的相关文章

  • 解决pyodbc.Error: (‘HY000‘, ‘[HY000] [Microsoft][ODBC Microsoft Access Driver]常见错误 无法打开注册表项“Temporary

    经过尝试 问题主要是数据库读写权限的问题 假如需要读写的数据库存放在系统C盘 就会因为权限的问题 pyodbc Error HY000 HY000 Microsoft ODBC Microsoft Access Driver 常见错误 无法
  • Node.js 开发常用到的库和插件工具,同事看到后也悄悄收藏了……

    Node js是一个功能强大 并且非常流行的 JavaScript 运行时环境 使开发人员能够高效率的构建高性能应用程序 下面介绍了8个常见的应用程序开发中用到的库和函数 可以用于缓存数据 操作日期 处理图像 发送电子邮件 发出 HTTP
  • MLCC电容为什么会啸叫?怎么让它闭嘴!

    原文来自公众号 工程师看海 公众号回复获取资料 电源 随着笔记本电脑 手机等设备的普及 由电容器振动所产生的 啸叫 问题越来越多的受到人们的关注 如何优化各电源架构的电容啸叫 让电容闭嘴 是一个有趣的问题 MLCC电容器发生啸叫主要是由陶瓷
  • 【AI绘画】我以Midjourney为主学习AI绘画效果咋样?

    上一篇博客链接 ChatGPT ChatGPT掀起AIGC与AI浪潮 山楂山楂丸的博客 CSDN博客 这周 我加入了新星计划 涉及的领域是我感兴趣以及对未来规划有帮助的 AI绘画 文章目录 前言 一 AI绘画是什么 二 AI绘画进阶 三 我
  • 误差和误差限

    误差和误差限 在数值计算中使用的数通常有两种类型 一种是能够准确反映客观事物数量关系的精确数 如班级有30人 有 1 2 frac 1 2 21 为男生 另一种是近似反映客观事物数量关系的近似数 如书重0 15kg 如果改变观测方法 提高测
  • Python串口通信模块PySerial使用教程(CH340 USB TTL转接芯片)

    CONTENTS 1 CH340 USB TTL介绍 2 PySerial教程 1 CH340 USB TTL介绍 TTL 一般是从单片机或者芯片中发出的电平 高电平为 5V 51单片机 或者 3 3V STM32 USB 转 TTL 模块
  • BP批量导入程序

    REPORT zfic cust batch upload 数据定义 TABLES sscrfields CONSTANTS cn flag TYPE char1 VALUE X DATA smp dyntxt TYPE smp dyntx
  • yolov3训练自己的数据集(MMDetection)

    用FasterRcnn训练了自己标注的数据集Voc格式 现在想用yolo来训练一下 修改了yolo文件内容 打算直接用yolo训练voc格式的数据 出现了一点问题 因为比较着急 就没有再详细研究 MMDetection中大多训练模型为coc
  • web视频播放

    一 3大视频直播协议 一 RTMP 全称 Real Time Messageing Protocol 实时消息传送协议 出身 由Adobe公司基于 Flash Player 播放器对应的音视频flv封装格式提出的一种传输协议 优点 1 延迟
  • ESP32开发:从入门到精通的全面指南

    ESP32开发 从入门到精通的全面指南 ESP32 是一款功能强大的微控制器 它可以帮助开发者构建各种物联网 IoT 应用 ESP32 拥有 WiFi 和蓝牙无线连接功能 并且可以轻松地与各种传感器和设备进行通信 本文将介绍 ESP32 的
  • 【框架1-springboot整合Knife4j框架-API文档框架】

    一 简介 图示 编辑 二 框架使用 1 引入依赖 2 配置config 3 配置application yml 启动knife4j 4 设置端口 5 进入doc html 三 框架配置 1 菜单项设置 2 菜单子项设置 3 请求参数设置 四
  • AcWing600.仰视奶牛(单调栈)

    输入样例 6 3 2 6 1 1 2 输出样例 3 3 0 6 6 0 include
  • 微信小程序接入微信支付流程

    一 基本介绍 1 支付场景 点击支付按钮唤起微信支付弹窗 输入正确密码后完成支付 2 基本流程 点击支付按钮首先生成一个订单 然后在后端调用微信api接口进行统一下单 将接口返回的数据回传到前端拉起支付操作 然后异步通知支付结果 二 配置信
  • Inno Setup打包教程

    简述 Inno Setup 是一个免费的 Windows 安装程序制作软件 第一次发表是在 1997 年 现在已经更新到Inno Setup 5了 Inno Setup是一个十分简单实用的打包小工具 可以按照我们自己的意愿设置功能 稳定性也
  • shell中的变量 $VAR 与 ${VAR}区别

    VAR 最好VAR在定义的时候 采用VAR abcdefg 这种带引号的定义 VAR 这种写法比较规范 VAR在定义的时候 采用VAR abcdefg 这种不带引号的定义也没有关系
  • 微信抓包您的连接不是私密连接解决办法

    问题 在对微信公众号进行抓包时出现以下问题 您的连接不是私密连接 攻击者可能会试图从 x x x x 窃取您的信息 例如 密码 通讯内容或信用卡信息 了解详情 NET ERR CERT INVALID 将您访问的部分网页的网址 有限的系统信
  • 面向 Android* 设备的英特尔® USB 驱动程序

    作者 Tao Wang Intel 利用英特尔 Android USB 驱动程序包 您能够将您基于 Windows 的机器连接至安装了英特尔 凌动 处理器的 Android 设备 注 英特尔 USB 驱动程序包 版本 1 1 5 增加了对
  • el-input 只能输入数字并限制长度

    在上一个博客中 有关于限制长度的使用 本文介绍限制只能输入数字的方法 el input 代码如下
  • 为什么要用数据库视图?

    视图的定义 视图 View 是一种虚拟的表 其结构和数据来自于一个或多个基本表 可以被当作普通表一样进行查询操作 但实际上不存储任何数据 在数据库中 视图可以被看作是一种数据访问的方式 它可以隐藏底层表的复杂性 提供简洁易懂的数据访问接口

随机推荐