十六、IO流(二)

一、字符流

1.1 引入

public class Demo1 {
    public static void main(String[] args) throws IOException {
        FileInputStream fi = new FileInputStream("a.txt");
        int res = -1;
        while ((res = fi.read()) != -1) {
            System.out.print((char)res);   // ä½ å¥½world  出现乱码
            // 中文UTF-8编码占用3个字节,read()每次读1个字节,所以有乱码
        }
        fi.close();
    }
}
  • 字符流的介绍

  • 由于字节流操作中文不是特别的方便,所以Java就提供字符流

    • 字符流 = 字节流 + 编码表
  • 中文的字节存储方式

    • 用字节流复制文本文件时,文本文件也会有中文,但是没有问题,原因是最终底层操作会自动进行字节拼接成中文,如何识别是中文的呢?
    • 汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数

1.2 编码表

  • 什么是字符集

    • 是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
    • 计算机要准确的存储和识别各种字符集符号,就需要进行字符编码,一套字符集必然至少有一套字符编码。常见字符集有ASCII字符集、GBXXX字符集、Unicode字符集等
  • 常见的字符集

    • ASCII字符集:

      • ASCII:是基于拉丁字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)
    • 基本的ASCII字符集,使用7位表示一个字符,共128字符。ASCII的扩展字符集使用8位表示一个字符,共256字符,方便支持欧洲常用字符。是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
      在这里插入图片描述

    • GBXXX字符集

      • GBK:最常用的中文码表。是在GB2312标准基础上的扩展规范,使用了双字节编码方案,共收录了21003个汉字,完全兼容GB2312标准,同时支持繁体汉字以及日韩汉字等
    • Unicode

      • Unicode,是计算机科学领域里的一项业界标准。为每种语言中的每个字符设定了统一并且唯一的整数值,把这个数值称为码点(Code Point)

      • 但是它并不规定计算机如何存储和传输这个数值(以多少个字节存储,是定长还是变长,Unicode没有字节长度的概念)。

      • Unicode 定义了一个码点空间包含1,114,112个码点,范围从00x10FFFF,也就是说,它仅仅是一个字符映射集。其中0x0000 ~ 0xFFFF的字符表示常用字符集,称BMP字符,0x10000 ~ 0x10FFFF字符叫增补字符。

      • Unicode目前规划的总空间是17个平面(平面0至16),0x0000 至 0x10FFFF,每个平面有 65536 个码点。
        例如,"AB中国"这个字符串,对应的Unicode编码为:

        A  -> \u0041
        B  -> \u0042
        中 -> \u0049
        国 -> \u56FD
        
      • UTF32编码

        • 4个字节表示一个字符,总共2^32=4294967296
        • 其高位均为0,空间浪费比较严重,因此应用很少
      • 常用的UTF16一般使用两个字节表示常用字符,对于不能表示的或不常用的字符才使用32位编码,这是Windows程序默认的Unicode编码方式

        • 使用 1 ~ 2 个16bit变长编码表示1,112,064Unicode码点
        • 它扩展于固定16bit长度的UCS-2
      • UTF8编码按照不同的国家文字的多少分别使用1个字节、2个字节、3个字节和4个字节表示,常用于网络传速。

        • 使用 1 ~ 4 个字节变长编码表示1,112,064Unicode码点
        • 兼容ASCII
        • 码点数值越小,使用的字节数越少,出现的频率越高

在这里插入图片描述

1.3 字符串中的编码解码问题

  • 相关方法

    方法名说明
    byte[] getBytes()使用平台的默认字符集将该String编码为一系列字节
    byte[] getBytes(String charsetName)使用指定的字符集将该String编码为一系列字节
    String(byte[] bytes)使用平台的默认字符集解码指定的字节数组来创建字符串
    String(byte[] bytes, String charsetName)通过指定的字符集解码指定的字节数组来创建字符串
  • 代码演示

    public class Demo2 {
        public static void main(String[] args) throws UnsupportedEncodingException {
            // byte[] getBytes() 使用平台的默认字符集将该String编码为一系列字节
            // byte[] getBytes(String charsetName) 使用指定的字符集将该String编码为一系列字节
            String str = "字符流编码";
            byte[] bytes1 = str.getBytes();
            System.out.println(Arrays.toString(bytes1));
            byte[] gbks = str.getBytes("GBK");
            System.out.println(Arrays.toString(gbks));
    
            // String(byte[] bytes) 使用平台的默认字符集解码指定的字节数组来创建字符串
            // String(byte[] bytes, String charsetName)通过指定的字符集解码指定的字节数组来创建字符串
            String s1 = new String(bytes1);
            String s2 = new String(gbks,"gbk");
            System.out.println(s1);
            System.out.println(s2);
    
        }
    }
    

1.4 字符流写数据

  • 介绍

    • Writer:用于写入字符流的抽象父类
    • FileWriter:用于写入字符流的常用子类

在这里插入图片描述

  • 字符流的底层依旧调用了字节流,其实质:字节流+编码表

  • 构造方法

    方法名说明
    FileWriter(File file)根据给定的File对象构造一个FileWriter对象
    FileWriter(File file, boolean append)根据给定的File对象构造一个FileWriter对象
    FileWriter(String fileName)根据给定的文件名构造一个FileWriter对象
    FileWriter(String fileName, boolean append)根据给定的文件名以及指示是否附加写入数据的boolean值来构造FileWriter对象
    • 注:

      • 若文件不存在会默认创建,但要先确保其父级路径存在
      • 若文件存在,则默认会清空文件内容!!,除非设定追加
  • 成员方法

    方法名说明
    void write(int c)写一个字符
    void write(char[] cbuf)写入一个字符数组
    void write(char[] cbuf, int off, int len)写入字符数组的一部分
    void write(String str)写一个字符串
    void write(String str, int off, int len)写一个字符串的一部分
    • 注:写入的默认都是数字在编码表对应的字符,若要写入数字,可以使用字符串的方式写入
  • 刷新和关闭的方法

    方法名说明
    flush()刷新流,之后还可以继续写数据
    close()关闭流,释放资源,但是在关闭之前会先刷新流。一旦关闭,就不能再写数据
    • 注:刷新流会立即将缓冲区的内容刷新到磁盘上
  • 代码演示

    public class Demo3 {
        public static void main(String[] args) throws IOException {
            FileWriter fw = new FileWriter("a.txt");
    
            //void write(int c):写一个字符
            fw.write(97);
            
            //void writ(char[] cbuf):写入一个字符数组
            char[] chars = {97, 98, 99, 100};
            fw.write(chars);
            
            //void write(char[] cbuf, int off, int len):写入字符数组的一部分
            fw.write(chars, 0, 3);
            
            //void write(String str):写一个字符串
            String str1 = "字符流写入字符串";
            fw.write(str1);
    
            //void write(String str, int off, int len):写一个字符串的一部分
            fw.write(str1, 1, 2);
            // a.txt内容:aabcdabc字符流写入字符串符流
            
            //释放资源
            fw.close();
        }
    }
    

1.5 字符流读数据

  • 介绍
    • Reader:用于读取字符流的抽象父类
    • FileReader:用于读取字符流的常用子类

在这里插入图片描述

  • 构造方法

    方法名说明
    FileReader(File file)在给定从中读取数据的File的情况下创建一个新FileReader
    FileReader(String fileName)在给定从中读取数据的文件名的情况下创建一个新FileReader
  • 成员方法

    方法名说明
    int read()一次读一个字符数据
    int read(char[] cbuf)一次读一个字符数组数据
  • 代码演示

    public class Demo4 {
        public static void main(String[] args) throws IOException {
            FileReader fr = new FileReader("a.txt");
            System.out.println(fr.read());  // 一次读一个字符数据
    
            // 一次读取一个字符数组
            char[] chars = new char[1024];
            int len;
            while ((len = fr.read(chars)) != -1) {
                System.out.println(new String(chars, 0, len));
            }
            fr.close();
    
        }
    }
    

1.6 字符流用户注册案例

  • 案例需求

    • 将键盘录入的用户名和密码保存到本地实现永久化存储
  • 实现步骤

    • 获取用户输入的用户名和密码
    • 将用户输入的用户名和密码写入到本地文件中
    • 关流,释放资源
  • 代码实现

    public class Demo5 {
        public static void main(String[] args) throws IOException {
            // 实现键盘录入,把用户名和密码录入进来
            Scanner sc = new Scanner(System.in);
            System.out.println("请录入用户名");
            String username = sc.next();
            System.out.println("请录入密码");
            String password = sc.next();
    
            // 分别把用户名和密码写到本地文件。
            FileWriter fw = new FileWriter("a.txt");
            // 将用户名和密码写到文件中
            fw.write(username);
            // 表示写出一个回车换行符 windows \r\n  MacOS \r  Linux \n
            fw.write("\r\n");
            fw.write(password);
            
            // 关流,释放资源
            fw.close();
        }
    }
    

1.7 字符缓冲流

  • 字符缓冲流介绍

    • BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入,可以指定缓冲区大小,或者可以接受默认大小。默认值足够大(8192),可用于大多数用途

    • BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取,可以指定缓冲区大小,或者可以使用默认大小。 默认值足够大(8192),可用于大多数用途

  • 构造方法

    方法名说明
    BufferedWriter(Writer out)创建字符缓冲输出流对象
    BufferedReader(Reader in)创建字符缓冲输入流对象
  • 代码演示

    public class Demo6 {
        public static void main(String[] args) throws IOException {
            // BufferedWriter(Writer out)
            BufferedWriter bw = new BufferedWriter(new FileWriter("a.txt"));  // 存在默认清空
            bw.write("line1");
            bw.write("\r\n");
            bw.write("line2");
    
            bw.close();
    
            // BufferedReader(Reader in)
            BufferedReader br = new BufferedReader(new FileReader("a.txt"));
            char[] chars = new char[1024];
            int len;
            while ((len = br.read(chars)) != -1) {
                System.out.println(new String(chars, 0, len));
            }
    
            br.close();
    
        }
    }
    

1.8 字符缓冲流特有功能

  • 方法介绍

    • BufferedWriter类中:
    方法名说明
    void newLine()写一行 行分隔符,行分隔符字符串由系统属性定义
    • BufferedReader类中:
    方法名说明
    String readLine()读一行文字。结果包含行的内容的字符串,不包括任何行终止字符如果流的结尾已经到达,则为null
  • 代码演示

    public class Demo7 {
        public static void main(String[] args) throws IOException {
            // BufferedWriter(Writer out)
            BufferedWriter bw = new BufferedWriter(new FileWriter("a.txt"));  // 存在默认清空
            bw.write("line1");
            bw.newLine();
            bw.write("line2");
    
            bw.close();
    
            // BufferedReader(Reader in)
            BufferedReader br = new BufferedReader(new FileReader("a.txt"));
            System.out.println(br.readLine());
            System.out.println(br.readLine());
    
            br.close();
    
        }
    }
    

1.9 字符缓冲流操作文件中数据排序案例

  • 案例需求

    使用字符缓冲流读取文件中的数据,排序后再次写到本地文件

  • 实现步骤

    • 将文件中的数据读取到程序中
    • 对读取到的数据进行处理
    • 将处理后的数据添加到集合中
    • 对集合中的数据进行排序
    • 将排序后的集合中的数据写入到文件中
  • 代码实现

    import java.io.*;
    import java.util.Arrays;
    
    public class Demo8 {
        public static void main(String[] args) throws IOException {
            BufferedWriter bw = new BufferedWriter(new FileWriter("a.txt"));
            bw.write("1 3 4 6 2 7 2 5 3 9");
            bw.close();
    
            // 读取文件中乱序的字符串
            BufferedReader br = new BufferedReader(new FileReader("a.txt"));
            String s = br.readLine();
            System.out.println("原内容:" + s);
            String[] strings = s.split(" ");
            int[] array = new int[strings.length];
            for (int i = 0; i < strings.length; i++) {
                array[i] = Integer.parseInt(strings[i]);
            }
            // 排序
            Arrays.sort(array);
            System.out.println("排序后:" + Arrays.toString(array));
            // 清空源文件写入排序后的数组
            BufferedWriter bw2 = new BufferedWriter(new FileWriter("a.txt"));
            for (int i = 0; i < array.length; i++) {
                bw2.write(array[i] + " ");
            }
            bw2.close();
    
        }
    }
    

1.10 IO流小结

  • IO流小结
    在这里插入图片描述

二、转换流

2.1 字符流中和编码解码问题相关的两个类

  • InputStreamReader:是从字节流到字符流的桥梁,父类是Reader

    • 它读取字节,并使用指定的编码将其解码为字符
    • 它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集
  • OutputStreamWriter:是从字符流到字节流的桥梁,父类是Writer

    • 是从字符流到字节流的桥梁,使用指定的编码将写入的字符编码为字节
    • 它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集

在这里插入图片描述

  • JDK 11以后可以直接使用FileReader(String fileName, Charset charset)指定字符集读取文件

在这里插入图片描述

2.2 转换流读写数据

  • 构造方法

    方法名说明
    InputStreamReader(InputStream in)使用默认字符编码创建InputStreamReader对象
    InputStreamReader(InputStream in,String chatset)使用指定的字符编码创建InputStreamReader对象
    OutputStreamWriter(OutputStream out)使用默认字符编码创建OutputStreamWriter对象
    OutputStreamWriter(OutputStream out,String charset)使用指定的字符编码创建OutputStreamWriter对象
  • 代码演示

    public class Demo9 {
        public static void main(String[] args) throws IOException {
            outputStreamWriter();  // 写一个GBK编码的文件
            inputStreamReader();  // 以GBK解码读取
    
            InputStreamReader isr = new InputStreamReader(new FileInputStream("C:\\Users\\16085\\Desktop\\a.txt"));
            char[] chars = new char[1024];
            int len;
            while ((len = isr.read(chars)) != -1) {
                // 读取GBK编码的文件出现乱码
                System.out.println(new String(chars, 0, len));   // GBK�����ַ�
            }
            isr.close();
            
            // JDK11 以后可以直接指定以何种方式解码
            FileReader fileReader = new FileReader("C:\\Users\\16085\\Desktop\\a.txt",Charset.forName("GBK"));
            BufferedReader br = new BufferedReader(fileReader);
            System.out.println(br.readLine());  // GBK编码字符
        }
    
        private static void outputStreamWriter() throws IOException {
            OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("C:\\Users\\16085\\Desktop\\a.txt"),"GBK");
            osw.write("GBK编码字符");
            osw.close();
        }
    
        private static void inputStreamReader() throws IOException {
            InputStreamReader isr = new InputStreamReader(new FileInputStream("C:\\Users\\16085\\Desktop\\a.txt"),"GBK");
            char[] chars = new char[1024];
            int len;
            while ((len = isr.read(chars)) != -1) {
                // 指定GBK解码以后正常打印
                System.out.println(new String(chars, 0, len));   // GBK编码字符
            }
            isr.close();
        }
    }
    

三、对象操作流

3.1 对象序列化流

  • 对象序列化介绍

    • 对象序列化:就是将对象保存到磁盘中,或者在网络中传输对象
    • 这种机制就是使用一个字节序列表示一个对象,该字节序列包含:对象的类型、对象的数据和对象中存储的属性等信息
    • 字节序列写到文件之后,相当于文件中持久保存了一个对象的信息
    • 反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化
  • 对象序列化流:ObjectOutputStream

    • Java对象的原始数据类型和图形写入OutputStream。 可以使用ObjectInputStream读取(重构)对象。 可以通过使用流的文件来实现对象的持久存储。 如果流是网络套接字流,则可以在另一个主机上或另一个进程中重构对象
  • 构造方法

    方法名说明
    ObjectOutputStream(OutputStream out)创建一个写入指定的OutputStreamObjectOutputStream
  • 序列化对象的方法

    方法名说明
    void writeObject(Object obj)将指定的对象写入ObjectOutputStream
  • 示例代码

    • 学生类
    public class Student implements Serializable {
        private String name;
        private int age;
    
        public Student() {
        }
    
        public Student(String name, int age) {
            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;
        }
    
        @Override
        public String toString() {
            return "Student{" +
                    "name='" + name + '\'' +
                    ", age=" + age +
                    '}';
        }
    }
    

    测试类

    public class Demo10 {
        public static void main(String[] args) throws IOException {
            Student s = new Student("zhangsan", 23);
            ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("a.txt"));
            oos.writeObject(s);  // 将对象s保存到文件中
            oos.close();
        }
    }
    
  • 注意事项

    • 一个对象要想被序列化,该对象所属的类必须必须实现Serializable接口
    • Serializable是一个标记接口,实现该接口,不需要重写任何方法

3.2 对象反序列化流

  • 对象反序列化流:ObjectInputStream

    • ObjectInputStream反序列化先前使用ObjectOutputStream编写的原始数据和对象
  • 构造方法

    方法名说明
    ObjectInputStream(InputStream in)创建从指定的InputStream读取的ObjectInputStream
  • 反序列化对象的方法

    方法名方法名
    Object readObject()ObjectInputStream读取一个对象
  • 几种读写结束符对比

方法名说明
read()读取到文件末尾返回值是 -1
readLine()读取到文件的末尾返回值 null
readObject()读取到文件的末尾 直接抛出异常
  • 如果要序列化的对象有多个,不建议直接将多个对象序列化到文件中,因为反序列化时容易出异常

    • 建议:将要序列化的多个对象存储到集合中,然后将集合序列化到文件中
  • 示例代码

    public class Demo11 {
        public static void main(String[] args) throws IOException, ClassNotFoundException {
            ObjectInputStream ois = new ObjectInputStream(new FileInputStream("a.txt"));
            //Object readObject():从ObjectInputStream读取一个对象
            System.out.println(ois.readObject());  // Student{name='zhangsan', age=23}
            ois.close();
        }
    }
    

3.3 serialVersionUID&transient

  • serialVersionUID

    • 用对象序列化流序列化了一个对象后,假如我们修改了对象所属的类文件,读取数据会不会出问题呢?
      • 会出问题,会抛出InvalidClassException异常
    • 如果出问题了,如何解决呢?
      • 重新序列化
      • 给对象所属的类加一个serialVersionUID
        • private static final long serialVersionUID = 42L;
  • transient

    • 如果一个对象中的某个成员变量的值不想被序列化,又该如何实现呢?
      • 给该成员变量加transient关键字修饰,该关键字标记的成员变量不参与序列化过程
  • 示例代码

    学生类

    public class Student implements Serializable {
        private static final long serialVersionUID = 42L;
        private String name;
    //    private int age;
        private transient int age;
    	......
    }
    

    测试类

    public class Demo12 {
        public static void main(String[] args) throws IOException, ClassNotFoundException {
            write();  // age被transient修饰,存储int默认值0
            read();  
    
        }
    
        //反序列化
        private static void read() throws IOException, ClassNotFoundException {
            ObjectInputStream ois = new ObjectInputStream(new FileInputStream("a.txt"));
            //Object readObject():从ObjectInputStream读取一个对象
            System.out.println(ois.readObject());  // Student{name='zhangsan', age=0}
            ois.close();
        }
    
        //序列化
        private static void write() throws IOException {
            Student s = new Student("zhangsan", 23);
            ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("a.txt"));
            oos.writeObject(s);  // 将对象s保存到文件中
            oos.close();
        }
    }
    

3.4 对象操作流练习

  • 案例需求

    • 创建多个学生类对象写到文件中,再次读取到内存中
  • 实现步骤

    • 创建序列化流对象
    • 创建多个学生对象
    • 将学生对象添加到集合中
    • 将集合对象序列化到文件中
    • 创建反序列化流对象
    • 将文件中的对象数据,读取到内存中
  • 代码实现

    学生类(同上)

    测试类

    public class Demo13 {
        public static void main(String[] args) throws Exception {
            // 序列化
            // 创建序列化流对象
            ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("a.txt"));
            ArrayList<Student> arrayList = new ArrayList<>();
            // 创建多个学生对象
            Student s1 = new Student("zhangsan", 23);
            Student s2 = new Student("lisi", 24);
            // 将学生对象添加到集合中
            arrayList.add(s1);
            arrayList.add(s2);
            // 将集合对象序列化到文件中
            oos.writeObject(arrayList);
            oos.close();
    
            // 反序列化
            // 创建反序列化流对象
            ObjectInputStream ois = new ObjectInputStream(new FileInputStream("a.txt"));
            // 将文件中的对象数据,读取到内存中
            Object obj = ois.readObject();
            ArrayList<Student> arrayList2 = (ArrayList<Student>) obj;
            ois.close();
            // 遍历
            for (Student s : arrayList2) {
                System.out.println(s);
            }
        }
    }
    

四、Properties集合

4.1 Properties作为Map集合的使用

  • Properties介绍

    • 是一个Map体系的集合类
    • Properties可以保存到流中或从流中加载
    • 属性列表中的每个键及其对应的值都是一个字符串
  • Properties基本使用

    public class Demo14 {
        public static void main(String[] args) {
            Properties properties = new Properties();
            // 增加键值
            properties.put("id1", "zhangsan");
            properties.put("id2", "lisi");
            properties.put("id3", "wangwu");
    
            // 删除键值
            properties.remove("id3");
    
            // 修改键值
            properties.put("id2", "zhaoliu");
    
            // 查询键值
            System.out.println(properties.get("id1"));
    
            // 遍历
            // 先获取键,再获取值
            Set<Object> keySet = properties.keySet();
            for (Object key : keySet) {
                Object value = properties.get(key);
                System.out.println(key + "=" + value);
            }
    
            System.out.println("--------------");
            // 先获取entry集合,再分别获取键值
            Set<Map.Entry<Object, Object>> entries = properties.entrySet();
            for (Map.Entry<Object, Object> entry : entries) {
                Object key = entry.getKey();
                Object value = entry.getValue();
                System.out.println(key + "=" + value);
            }
        }
    }
    

4.2 Properties作为Map集合的特有方法

  • 特有方法

    方法名说明
    Object setProperty(String key, String value)设置集合的键和值,都是String类型,底层调用 Hashtable方法put
    String getProperty(String key)使用此属性列表中指定的键搜索属性
    Set<String> stringPropertyNames()从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串
  • 示例代码

    public class Demo15 {
        public static void main(String[] args) {
            Properties prop = new Properties();
            // Object setProperty(String key, String value):设置集合的键和值,都是String类型
            prop.setProperty("id1", "zhangsan");
            prop.setProperty("id2", "lisi");
            prop.setProperty("id3", "wangwu");
    
            // String getProperty(String key):使用此属性列表中指定的键搜索属性
            System.out.println(prop.getProperty("id1"));
    
            // Set<String> stringPropertyNames():从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串
            Set<String> names = prop.stringPropertyNames();
            for (String key : names) {
                String value = prop.getProperty(key);
                System.out.println(key + "=" + value);
            }
        }
    }
    

4.3 Properties和IO流相结合的方法

  • IO流结合的方法

    方法名说明
    void load(Reader reader)从输入字符流读取属性列表(键和元素对)
    void store(Writer writer, String comments)将此属性列表(键和元素对)写入此Properties表中,以适合使用 load(Reader)方法的格式写入输出字符流
  • 示例代码

public class Demo16 {
    public static void main(String[] args) throws IOException {
        Properties prop = new Properties();
        // void load(Reader reader):从输入字符流读取属性列表(键和元素对)
        FileReader fr = new FileReader("prop.properties");
        prop.load(fr);
        fr.close();
        System.out.println(prop);

        // void store(Writer writer, String comments)
        // 将此属性列表(键和元素对)写入此Properties表中,comments为开头注释
        prop.setProperty("id3","wangwu");
        FileWriter fw = new FileWriter("prop.properties");
        prop.store(fw,"");
        fw.close();
    }

}
  • prop.properties内容

在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
1JAVA SE 1.1深入JAVA API 1.1.1Lang包 1.1.1.1String类和StringBuffer类 位于java.lang包中,这个包中的类使用时不用导入 String类一旦初始化就不可以改变,而stringbuffer则可以。它用于封装内容可变的字符串。它可以使用tostring()转换成string字符串。 String x=”a”+4+”c”编译时等效于String x=new StringBuffer().append(“a”).append(4).append(“c”).toString(); 字符串常量是一种特殊的匿名对象,String s1=”hello”;String s2=”hello”;则s1==s2;因为他们指向同一个匿名对象。 如果String s1=new String(“hello”);String s2=new String(“hello”);则s1!=s2; /*逐行读取键盘输入,直到输入为“bye”时,结束程序 注:对于回车换行,在windows下面,有'\r'和'\n'两个,而unix下面只有'\n',但是写程序的时候都要把他区分开*/ public class readline { public static void main(String args[]) { String strInfo=null; int pos=0; byte[] buf=new byte[1024];//定义一个数组,存放换行前的各个字符 int ch=0; //存放读入的字符 system.out.println(“Please input a string:”); while(true) { try { ch=System.in.read(); //该方法每次读入一个字节的内容到ch变量中。 } catch(Exception e) { } switch(ch) { case '\r': //回车时,不进行处理 break; case '\n': //换行时,将数组总的内容放进字符串中 strInfo=new String(buf,0,pos); //该方法将数组中从第0个开始,到第pos个结束存入字符串。 if(strInfo.equals("bye")) //如果该字符串内容为bye,则退出程序。 { return; } else //如果不为bye,则输出,并且竟pos置为0,准备下次存入。 { System.out.println(strInfo); pos=0; break; } default: buf[pos++]=(byte)ch; //如果不是回车,换行,则将读取的数据存入数组中。 } } } } String类的常用成员方法 1、构造方法: String(byte[] byte,int offset,int length);这个在上面已经用到。 2、equalsIgnoreCase:忽略大小写的比较,上例中如果您输入的是BYE,则不会退出,因为大小写不同,但是如果使用这个方法,则会退出。 3、indexOf(int ch);返回字符ch在字符串中首次出现的位置 4、substring(int benginIndex); 5、substring(int beginIndex,int endIndex); 返回字符串的子字符串,4返回从benginindex位置开始到结束的子字符串,5返回beginindex和endindex-1之间的子字符串。 基本数据类型包装类的作用是:将基本的数据类型包装成对象。因为有些方法不可以直接处理基本数据类型,只能处理对象,例如vector的add方法,参数就只能是对象。这时就需要使用他们的包装类将他们包装成对象。 例:在屏幕上打印出一个*组成的矩形,矩形的宽度和高度通过启动程序时传递给main()方法的参数指定。 public class testInteger { public static void main(String[] args) //main()的参数是string类型的数组,用来做为长,宽时,要转换成整型。 { int w=new Integer(args[0]).intValue(); int h=Integer.parseInt(args[1]); //int h=Integer.valueOf(args[1]).intValue(); //以上为三种将字符串转换成整形的方法。 for(int i=0;i<h;i++) { StringBuffer sb=new StringBuffer(); //使用stringbuffer,是因为它是可追加的。 for(int j=0;j<w;j++) { sb.append('*'); } System.out.println(sb.toString()); //在打印之前,要将stringbuffer转化为string类型。 } } } 比较下面两段代码的执行效率: (1)String sb=new String(); For(int j=0;j<w;j++) { Sb=sb+’*’; } (2) StringBuffer sb=new StringBuffer(); For(int j=0;j<w;j++) { Sb.append(‘*’); } (1)和(2)在运行结果上相同,但效率相差很多。 (1)在每一次循环中,都要先将string类型转换为stringbuffer类型,然后将‘*’追加进去,然后再调用tostring()方法,转换为string类型,效率很低。 (2)在没次循环中,都只是调用原来的那个stringbuffer对象,没有创建新的对象,所以效率比较高。 1.1.1.2System类与Runtime类 由于java不支持全局函数和全局变量,所以java设计者将一些与系统相关的重要函数和变量放在system类中。 我们不能直接创建runtime的实例,只能通过runtime.getruntime()静态方法来获得。 编程实例:在java程序中启动一个windows记事本程序的运行实例,并在该运行实例中打开该运行程序的源文件,启动的记事本程序5秒后关闭。 public class Property { public static void main(String[] args) { Process p=null; //java虚拟机启动的进程。 try { p=Runtime.getRuntime().exec("notepad.exe Property.java"); //启动记事本并且打开源文件。 Thread.sleep(5000); //持续5秒 p.destroy(); //关闭该进程 } catch(Exception ex) { ex.printStackTrace(); } } } 1.1.1.3Java语言中两种异常的差别 Java提供了两类主要的异常:runtime exception和checked exception。所有的checked exception是从java.lang.Exception类衍生出来的,而runtime exception则是从java.lang.RuntimeException或java.lang.Error类衍生出来的。    它们的不同之处表现在两方面:机制上和逻辑上。    一、机制上    它们在机制上的不同表现在两点:1.如何定义方法;2. 如何处理抛出的异常。请看下面CheckedException的定义:    public class CheckedException extends Exception    {    public CheckedException() {}    public CheckedException( String message )    {    super( message );    }    }    以及一个使用exception的例子:    public class ExceptionalClass    {    public void method1()    throws CheckedException    {     // ... throw new CheckedException( “...出错了“ );    }    public void method2( String arg )    {     if( arg == null )     {      throw new NullPointerException( “method2的参数arg是null!” );     }    }    public void method3() throws CheckedException    {     method1();    }    }    你可能已经注意到了,两个方法method1()和method2()都会抛出exception,可是只有method1()做了声明。另外,method3()本身并不会抛出exception,可是它却声明会抛出CheckedException。在向你解释之前,让我们先来看看这个类的main()方法:    public static void main( String[] args )    {    ExceptionalClass example = new ExceptionalClass();    try    {    example.method1();    example.method3();    }    catch( CheckedException ex ) { } example.method2( null );    }    在main()方法中,如果要调用method1(),你必须把这个调用放在try/catch程序块当中,因为它会抛出Checked exception。    相比之下,当你调用method2()时,则不需要把它放在try/catch程序块当中,因为它会抛出的exception不是checked exception,而是runtime exception。会抛出runtime exception的方法在定义时不必声明它会抛出exception。    现在,让我们再来看看method3()。它调用了method1()却没有把这个调用放在try/catch程序块当中。它是通过声明它会抛出method1()会抛出的exception来避免这样做的。它没有捕获这个exception,而是把它传递下去。实际上main()方法也可以这样做,通过声明它会抛出Checked exception来避免使用try/catch程序块(当然我们反对这种做法)。    小结一下:    * Runtime exceptions:    在定义方法时不需要声明会抛出runtime exception;    在调用这个方法时不需要捕获这个runtime exception;    runtime exception是从java.lang.RuntimeException或java.lang.Error类衍生出来的。    * Checked exceptions:    定义方法时必须声明所有可能会抛出的checked exception;    在调用这个方法时,必须捕获它的checked exception,不然就得把它的exception传递下去;    checked exception是从java.lang.Exception类衍生出来的。    二、逻辑上    从逻辑的角度来说,checked exceptions和runtime exception是有不同的使用目的的。checked exception用来指示一种调用方能够直接处理的异常情况。而runtime exception则用来指示一种调用方本身无法处理或恢复的程序错误。    checked exception迫使你捕获它并处理这种异常情况。以java.net.URL类的构建器(constructor)为例,它的每一个构建器都会抛出MalformedURLException。MalformedURLException就是一种checked exception。设想一下,你有一个简单的程序,用来提示用户输入一个URL,然后通过这个URL去下载一个网页。如果用户输入的URL有错误,构建器就会抛出一个exception。既然这个exception是checked exception,你的程序就可以捕获它并正确处理:比如说提示用户重新输入。 

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值