zoukankan      html  css  js  c++  java
  • Java IO(下)

    1,字符编码

    在Java程序的开发中最常见的是ISO8859-1,GBK/GBK2312,unicode,UTF编码.

    ISO8859-1:属于单字节编码,最多只能表示0-255的字符范围,主要在英文上应用.

    GBK/GB2312:中文的国际编码,专门用来表示汉字,是双字节编码,如果在此编码中出现中文,则使用ISO8859-1编码,GBK可以表示简体中文和繁体中文,而GB2312只能表示简体中文,GBK兼容GB2312

    unicode:Java中使用此编码方式,是最标准的一种编码,使用十六进制进行编码,但此编码不兼容ISO8859-1编码

    UTF:它兼容ISO8859-1编码,同时也可以用来表示所有的语言字符,不过UTF编码是不定长编码,每一个字符的长度为1-6个字节不等,一般在中文网页中使用此编码,可以节约空间

    在IO体系中,使用到转化流的时候,我们可以对输入或者输出的内容指定其特定的编码表

    12

    2,内存操作流

    在IO中,是使用ArrayByteInputStream和ArrayByteOutputStream来完成内存的输入和输出,由于数据是在内存中进行操作的,不涉及到具体的文件,所以在输出的时候,

    ArrayByteOutputStream这个类只需要构造无参的构造方法.

    内存中数据的存在方式都是字节,所以在ArrayByteInputStream中的构造函数传入的是byte类型的数组.

    12

    package cn.bytearray;
    
    import java.io.ByteArrayInputStream;
    import java.io.ByteArrayOutputStream;
    import java.io.IOException;
    
    public class ByteArrayDemo {
        public static void main(String[] args) throws IOException {
            String str = "abcsw";
            ByteArrayInputStream bis = new ByteArrayInputStream(str.getBytes());
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            System.out.println("available:"+bis.available());
            int len = 0;
            byte[] buf = new byte[1024];
            while((len=bis.read(buf))!=-1){
                String s = new String(buf,0,len);
                System.out.println(s);
            }
            bis.close();
            bos.close();
        }
    }

    关于内存操作流的使用

    内存操作流一般在生成临时信息的时候才会使用到,如果将这些临时信息保存在文件中,当代码执行完毕的时候,在将储存临时信息的文件删除,这样操作起来会比较麻烦,这个时候使用内存操作流是最合适的.

    3,数据操作流

     

    4,跟踪行号的缓冲字符输入流(LineNumberReader)

    该类在对读到的字符输入流的时候,可以为该文本文件上加上行号,它里面也有读取文本行的方法,还有一个可以获取当前行号的方法

    12

    package cn.linenumberread;
    
    import java.io.FileNotFoundException;
    import java.io.FileReader;
    import java.io.IOException;
    import java.io.LineNumberReader;
    
    public class LineNumberReadDemo {
        public static void main(String[] args) throws IOException {
            FileReader fr = new FileReader("各种需求.txt");
            LineNumberReader lnr = new LineNumberReader(fr);
            String lin = null;
            while((lin=lnr.readLine())!=null){
                System.out.println(lnr.getLineNumber()+":"+lin);
            }
            fr.close();
            lnr.close();
        }    
    
    }

    输出结果的实例,该输出结果的左侧就具备了行号

    捕获

    5,关于window中的'\r'和'\n'的知识点

    在Java中\r的阿斯科码值是13,\n的阿斯科码值是10.在window中的换行是有两个阿斯科码值13和10组成的,其实在window中的enter键,这个键按一下,它是实现了2个动作的,

    \r和\n,所以在window中的换行是/r/n.

    package cn.io;
    
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.FileInputStream;
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.InputStreamReader;
    import java.io.OutputStreamWriter;
    
    public class ReadKeyBoard {
        public static void main(String[] args) throws IOException {
            //ReadKey_1();
            ReadKey_2();
        }
    
        public static void ReadKey_2() throws IOException {
            BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));
            BufferedWriter bufw = new BufferedWriter(new OutputStreamWriter(System.out));
            String temp = null;
            while((temp=bufr.readLine())!=null){
                if("over".equals(temp))
                    break;
                System.out.println(temp.toUpperCase());
                bufw.newLine();
            }
            bufr.close();
            bufw.close();
        }
    
        public static void ReadKey_1() throws IOException {
            StringBuilder sb = new StringBuilder();
            InputStream in = System.in;
            int len = 0;
            while((len=in.read())!=-1){
                if(len=='\r')
                    continue;
                if(len=='\n'){
                    String str = sb.toString();
                    if("over".equals(str))
                        break;
                    System.out.println(str.toUpperCase());
                    sb.delete(0, sb.length());
                }
                else
                sb.append((char)len);
            }
        }
    
    }

    在以上代码中,ReadKey_1这个方法是原始的在控制台上打印出内容,在window中,打印的时候,当一句话打完,就回按一下enter键,这个时候对于程序来说,会产生两个char,\r和\n,这个时候就必须对程序进行判断,当是/r的时候,我们不记录字符,当/n的时候,表示一句话完成了,就得记录打印的数据,然后再把它显示在控制台上去

    那么每次记录的字符该如何处理,这个时候就得想到容器的存在,而由于容器一定义就会一直存在在程序中,就是每次打印结果都是把上次记录的数据也打印出来,这个时候,没记录一次数据,打印数据之后,就得删除上次打印的内容,集合清空一般使用clear方法,StringBuffer和StringBuilder用delete方法

     

    6,对象序列化(ObjectOutputStream和ObjectInputStream)

    对象序列化就是将一个对象变成二进制的数据流的一种方法,,如果一个类需要被序列化,那么对象所在的类必须实现java.io.Serializable接口,该接口中没有定义任何的方法,它只是一个标示接口,标示这一个类具有被序列化的能力.

    使用对象输出流输出序列化对象的步骤叫做序列化,反之叫做反序列化

     

     

     

     

     

     

     

     

     

    对象序列化和对象反序列化操作时的版本兼容的问题

    在对象序列化和反序列化的时候,JDK版本的不统一的话就有可能会造成异常,所以在序列化操作的时候引入一个serialVersionUID的常量,例如

    private static final long serialVersionUID = 1L;这里的具体内容可以我们自己决定

    一个对象被序列化的内容,只有对象的属性被序列化了

    既然Serializable接口中没有任何方法的话,是不是所有的类都可以实现该接口?

    不可以,这样在以后的JDK版本的升级中,会出现问题,以后升级的时候,有可能会在该接口中增加方法的.

    transient关键字

    如果一个对象中的某个属性不希望被序列化,则可以使用该关键字进行声明.

    package cn.io;
    
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.io.ObjectOutputStream;
    
    import cn.bean.Person;
    
    public class ObjectOutputStreamDemo {
        public static void main(String[] args) throws IOException, IOException {
            File f = new File("person.txt");
            ObjectOutputStream ops = new ObjectOutputStream(new FileOutputStream(f));
            ops.writeObject(new Person("wjd",20));
            ops.close();
        }
    
    }
    package cn.io;
    
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.IOException;
    import java.io.ObjectInputStream;
    
    import cn.bean.Person;
    
    public class ObjectInputStreamDemo {
        public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {
            File f = new File("person.txt");
            ObjectInputStream ops = new ObjectInputStream(new FileInputStream(f));
            Person p = (Person)ops.readObject();
            System.out.println(p.getName()+".."+p.getAge());
        }
    
    }
    package cn.bean;
    
    import java.io.Serializable;
    
    public class Person implements Serializable {
        private String name;
        private int age;
        public Person(String name, int age) {
            super();
            this.name = name;
            this.age = age;
        }
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
        public int getAge() {
            return age;
        }
        public void setAge(int age) {
            this.age = age;
        }
        
    }

    7,打印流(PrintStream和PrintWriter)

    PrintStream:字节打印流,提供的打印方法可以对多种的数据类型的值进行打印,并保持数据的表示形式,不会抛出IO异常,

                     构造函数传三种类型的数据,字符串路径,File对象,字节输出流(outputstream)

    PrintWriter:字符打印流,构造函数传入的参数有,字节输出流,字符输出流(writer),File对象,字符串路径

    TIPS:当在上面两个类中传入的是字符或者是字节输出流的时候,都可以在构造函数中加入一个自动刷新的功能true.

    package cn.io;
    
    import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.PrintWriter;
    
    public class PrintWriterDemo {
        public static void main(String[] args) throws IOException {
            BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));
            PrintWriter pw = new PrintWriter(System.out,true);
            String str = null;
            while((str=bufr.readLine())!=null){
                if("over".equals(str))
                    break;
                pw.println(str);
                //pw.flush();这里一定要刷新下,才能在控制台上打印出内容来
            }
            bufr.close();
            pw.close();
        }
    
    }
    package cn.io;
    
    import java.io.FileNotFoundException;
    import java.io.FileReader;
    import java.io.IOException;
    import java.io.PrintStream;
    
    public class PrintStreamDemo {
        public static void main(String[] args) throws IOException {
            //FileReader f = new FileReader("w.txt");
            PrintStream ps = new PrintStream("w.txt");
            ps.println("奥巴马");
            ps.println(11);
        }
    }                                                                                                                                                 

    8,将文件切割和合并的2个需求

    需求切割一首MP3歌曲,并要求有切割后的配置信息文件,记录歌曲的名字和切割后的切割文件的个数

    1,创建一个fileinputstream对象,将源和需要切割的文件相互关联起来

    2,创建一个fileoutputstream对象,由于被切割的文件的个数的不确定性,目的的初始值为null

    3,定义变量计数器count用来记录切割后文件名的前缀,变量len,用于进行读操作,记录读到数据的长度

    4,定义配置文件信息的对象,properties,创建一个File对象的目录文件夹mik,用来保存切割好的文件和配置文件

    5,进行标准的读写操作,在while循环中,确定切割文件的数量,利用计数器count来命名切割文件的文件名

    6,设置配置信息的键值对于关系,比如原始文件+原始文件名,part+切割后的数量

    7,fos对象中,设定需要放配置文件的位置和配置文件的名称

    8,将配置文件的信息放进去,store

    9,关闭流操作

    package cn.io;
    
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.util.Properties;
    
    public class SplitMP3 {
        public static void main(String[] args) throws IOException {
            File f = new File("d:"+File.separator+"pushu.mp3");
            SplitMP3(f);
        }
    
        public static void SplitMP3(File f) throws IOException {
            //创建源,将需要切割的文件与源进行关联
            FileInputStream fis = new FileInputStream(f);
            //创建目的,由于切割文件的数量不确定性,初始值为null
            FileOutputStream fos = null;
            //创建一个1m的缓冲区
            byte[] buf = new byte[1048576];
            //创建两个变量,计数器count和记录读到数据长度的len
            int count = 1;
            int len = 0;
            //创建配置信息对象,并创建相应的放切割文件和配置信息文件的File对象
            Properties pro = new Properties();
            File file = new File("d:"+File.separator+"parts");
            if(!file.exists())
                file.mkdirs();
            //开始读写操作,进行文件的切割并存储到对于的文件夹中
            while((len=fis.read(buf))!=-1){
                fos = new FileOutputStream(new File(file,(count++)+".part"));
                fos.write(buf, 0, len);
            }
            //将关键的信息加入到配置文件中
            pro.setProperty("filename", f.getName());
            pro.setProperty("part", count+"");
            fos = new FileOutputStream(new File(file,count+".info"));
            pro.store(fos, "splitMp3 info");
            //各种关闭流
            fis.close();
            fos.close();
        }
    
    }

    需求 将上面分割的文件,在合并成一首MP3歌曲,必须要读取配置文件中的信息来完成

    1,对方法中传入的File对象的目录参数进行listFiles方法,返回一个File数组,在listFiles的方法中传入一个文件过滤器,作用是在目录中去寻找后缀名是.info的配置文件.由于配置文件肯定只有一个存在,所以取出File[]数组中的第一个元素,这个就是配置文件的File对象

    2,创建配置文件properties对象,和fileinputstream对象,将配置文件的File对象穿入fileinputstream对象的构造函数,然后在加载配置文件

    Load,通过配置文件对象的方法,去得到mp3的文件名,被分割的文件的个数

    3,再次使用文件过滤器,找到目录中后缀名是.part的文件(被分割的文件),

    这样就得到一个File[]的数组

    4,创建arraylist集合对象,使用for循环,在File[]通过fileinputstream,加入到arraylist集合中,再使用collections这个集合工具类,来得到该集合的枚举对象

    5,创建源sequenceinputstream类,将枚举对象给它,再创建目的Fileoutputstream,传入file对象,这个使用到了上面在配置文件中得到的MP3文件名的那个字符串变量.

    6,进行标准的读写操作

    7,关闭各个流

    在上述2个分割和合并的需求中,我们在字节输入和输出流,传入File对象的时候,一定要留意到File中有个构造方法可以传两个对象的,一个是爹,一个是儿子.

    package cn.io;
    
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.io.SequenceInputStream;
    import java.util.ArrayList;
    import java.util.Collection;
    import java.util.Collections;
    import java.util.Enumeration;
    import java.util.Properties;
    
    public class MergeMP3 {
        public static void main(String[] args) throws IOException {
            File dir = new File("d:"+File.separator+"parts");
            //MergeMP3(dir);
            MergeMP3_2(dir);
        }
    
        public static void MergeMP3_2(File dir) throws IOException {
            File[] fileInfo = dir.listFiles(new FindNameFilter(".info"));
            File info = fileInfo[0];
            FileInputStream fis = new FileInputStream(info);
            Properties pro = new Properties();
            pro.load(fis);
            int part = Integer.parseInt(pro.getProperty("part"));
            String filename = pro.getProperty("filename");
            File[] filemp3 = dir.listFiles(new FindNameFilter(".part"));
            ArrayList<FileInputStream> al = new ArrayList<FileInputStream>();
            for(int i=0;i<filemp3.length;i++){
                al.add(new FileInputStream(filemp3[i]));
            }
            Enumeration<FileInputStream> en = Collections.enumeration(al);
            SequenceInputStream sis = new SequenceInputStream(en);
            FileOutputStream fos = new FileOutputStream(new File(dir,filename));
            byte[] buf = new byte[10];
            int len = 0;
            while((len=sis.read(buf))!=-1){
                fos.write(buf, 0, len);
            }
            fis.close();
            fos.close();
            sis.close();
        }
    
        public static void MergeMP3(File dir) throws IOException {
            ArrayList<FileInputStream> al = new ArrayList<FileInputStream>();
            for(int i=1;i<8;i++){
                al.add(new FileInputStream(new File(dir,i+".part")));
            }
            Enumeration<FileInputStream> en = Collections.enumeration(al);
            SequenceInputStream sis = new SequenceInputStream(en);
            int len=0;
            FileOutputStream fis = new FileOutputStream(new File(dir,"3.mp3"));
            while((len=sis.read())!=-1){
                fis.write(len);
            }
            sis.close();
            fis.close();
        }
    
    }
  • 相关阅读:
    《《《Postman 的官网下转
    《《《Springboot 配置 application.yml 连接MySQL数据库
    《《《MyBatis-Plus完成可用项目(通过测试可以连接数据库并返回数据(可以当作开发模板使用))
    《《《spring boot配置文件application.yml出现的异常信息:(java.lang.IllegalStateException: Failed to load property source from location 'classpath:/application.yml')
    《《《转载 Intellij IDEA 导入Maven项目
    start.spring.io访问不了,导致springboot项目创建不起来(Initialization failed for 'https://start.spring.io' Please check URL, network and proxy settings. Error mes...)
    《《《MyBatis-Plus入门 视频学习笔记
    在pom.xml中查看引入关系图
    《《《如何建立局域网共享文件夹(怎样通过IP地址共享文件夹(同一个局域网下))
    JS跨域问题及其解决办法
  • 原文地址:https://www.cnblogs.com/driverwjd/p/3861317.html
Copyright © 2011-2022 走看看