IO-3

IO-3

1_IO流(字符流FileReader)

* 1.字符流是什么

         *字符流是可以直接读写字符的IO流

         *字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节再写出.   

* 2.FileReader

         *FileReader类的 int read()方法读取单个字符!!!读取的字符,如果已到达流的末尾,则返回 -1

*               FileReaderfr = new FileReader("xxx.txt"); //里面的内容为:大家好

                   intx = fr.read();

                   System.out.println(x);   //输出22823

                   charc = (char)x;  

                   System.out.println(c);   //强转后输出:大

---------------------------------------------------------------------------------------------

                   FileReaderfr = new FileReader("aaa.txt");     //创建输入流对象,关联aaa.txt,如果没有这个文件运行的时候会报FileNotFoundException 文件找不到异常

                   intch;

                   while((ch= fr.read()) != -1) {              //将读到的字符赋值给ch,ch是码表值,通过项目默认的码表一次读取一个字符(汉字第一个字节一定是负数,当读到负数时就一次读俩!!!)

                            System.out.println((char)ch);    //将读到的字符强转后打印!!!!!!!!!

                   }                

                   fr.close(); //关流

 

2_IO流(字符流FileWriter)

* FileWriter类的write()方法可以自动把字符转为字节写出,参数可以是int,Byte,也可以是string!!

         FileWriterfw = new FileWriter("aaa.txt");//没有这个文件就创建,有就先清空!!!

         fw.write("aaa大家好");

         fw.write(97);

         fw.close();

 

3_IO流(字符流的拷贝)

         FileReaderfr = new FileReader("a.txt");

         FileWriterfw = new FileWriter("b.txt");

        

         intch;

         while((ch= fr.read()) != -1) {       //读取一个字节

                   fw.write(ch);       //将c写出

         }

        

         fr.close();

         fw.close();//Writer类中有一个2k的小缓冲区,如果不关流,就会将内容写到缓冲区里(并没有写到文件中),关流会将缓冲区内容刷新出来,再关闭

 

4_IO流(什么情况下使用字符流)

* 字符流也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符,写出时还要把字符转回字节;而字节流只是充当字节的搬运工,不涉及转换

* 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流(只读或只写时用字符流)

* 读取的时候是按照字符的大小读取的,不会出现半个中文

* 写出的时候可以直接将字符串写出,不用转换为字节数组(转换应该是交由底层做了)

 

5_IO流(字符流是否可以拷贝非纯文本的文件)

* 不可以拷贝非纯文本的文件

* 因为在读的时候会将字节转换为字符,在转换过程中(两个字节转换成一个字符,拼接的时候),可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去

* 如果是?,直接写出,这样写出之后的文件就乱了,看不了了 

 

6_IO流(自定义字符数组的拷贝)      

                  

         FileReaderfr = new FileReader("aaa.txt");     //创建字符输入流,关联aaa.txt

         FileWriterfw = new FileWriter("bbb.txt");      //创建字符输出流,关联bbb.txt

                  

         intlen;

         char[]arr = new char[1024*8];                 //创建字符数组

*注:将数据读到字符数组中,并返回读取到的有效字符的个数,数组里面是一个个的字符如:大家 好 a bc...  (6个字符)!!!!!!

         while((len= fr.read(arr)) != -1) {       //将文件上的数据读取到字符数组中

                   fw.write(arr,0, len);                  //将字符数组中的数据写到文件上

         }

                  

         fr.close(); //关流释放资源

         fw.close();        

 

7_IO流(带缓冲的字符流)

* BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区, 然后逐个返回给程序, 降低读取文件的次数, 提高效率

* BufferedWriter的write()方法写出字符时会先写到缓冲区, 缓冲区写满时才会写到文件, 降低写文件的次数, 提高效率

*

                   BufferedReaderbr = new BufferedReader(new FileReader("aaa.txt"));        //创建字符输入流对象,关联aaa.txt

                   BufferedWriterbw = new BufferedWriter(new FileWriter("bbb.txt")); //创建字符输出流对象,关联bbb.txt

                  

                   intch;                                  

                   while((ch= br.read()) != -1) {             //read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量ch    (16K)

                            bw.write(ch);   //write一次,是将数据装到字符数组,装满后再一起写出去

                   }

                  

                   br.close();         //关流

                   bw.close(); 

 

8_IO流(readLine()和newLine()方法) 

* BufferedReader的readLine()方法可以读取一行字符(不包含换行符号),如果已到达流末尾,则返回null

* BufferedWriter的newLine()可以输出一个跨平台的换行符号"\r\n"

*

                   BufferedReaderbr = new BufferedReader(new FileReader("aaa.txt"));

                   BufferedWriterbw = new BufferedWriter(new FileWriter("bbb.txt"));

                   Stringline;

                   while((line= br.readLine()) != null) {

                            bw.write(line);       //将读取的整行写入bbb.txt

                            //System.out.println(line);    //直接打印出读取的每一行数据

                            //bw.write("\r\n");  //只支持windows系统

                            bw.newLine();  //跨平台的方法,  都是写出回车换行符!!!

                   }

                  

                   br.close();

                   bw.close();

 

9_IO流(将文本反转)

* 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换

          * 分析:

          * 1,创建输入输出流对象

          * 2,创建集合对象

          * 3,将读到的数据存储在集合中

          * 4,倒着遍历集合将数据写到文件上

          * 5,关流

 

          * 注意事项:

          * 流对象尽量晚开早关!!!!!!

         publicstatic void main(String[] args) throws IOException {

                   //改写后是尽量晚开早关

                   //1,创建输入输出流对象

                   BufferedReaderbr = new BufferedReader(new FileReader("zzz.txt"));

                  

                   //2,创建集合对象

                   ArrayList<String>list = new ArrayList<>();

                   //3,将读到的数据存储在集合中

                   Stringline;

                   while((line= br.readLine()) != null) {

                            list.add(line);

                   }

                   br.close();         //关流

                  

                   //4,倒着遍历集合将数据写到文件上

                   BufferedWriterbw = new BufferedWriter(new FileWriter("revzzz.txt"));

                   for(inti = list.size() - 1; i >= 0; i--) {

                            bw.write(list.get(i));

                            bw.newLine();

                   }

                   //5,关流

                  

                   bw.close();

         }

 

10_IO流(LineNumberReader)

* LineNumberReader是BufferedReader的子类, 具有相同的功能, 并且可以统计行号

         *调用getLineNumber()方法可以获取当前行号

         *调用setLineNumber()方法可以设置当前行号

*

                   LineNumberReaderlnr = new LineNumberReader(new FileReader("aaa.txt"));

                   Stringline;

                   lnr.setLineNumber(100);          //设置行号

                   while((line= lnr.readLine()) != null) {  //readLine()一次行号就++一次!!!

                        System.out.println(lnr.getLineNumber() +":" + line);//获取行号,从101开始

                   }

                  

                   lnr.close();

 

11_IO流(装饰设计模式)

          * 装饰设计模式的好处是:

          * 耦合性不强,被装饰的类的变化与装饰类的变化无关,其实我也可以让HeiMaStudent去继承Student,然后对父类里面的code方法重写,但这样耦合性太强,父类如果做改动的话子类也跟着做改动了,而在这种装饰模式中,装饰类不会跟着student的改变而改变,耦合性就小了;另外对于student来说它的选择性也比较多,他可以选择HeiMaStudent对他进行包装,也可以选择其他类对他进行包装

public class Demo6_Wrap {

         publicstatic void main(String[] args) {

                   HeiMaStudenthms = new HeiMaStudent(new Student());

                   hms.code();

         }

 

}

 

interface Coder {

         publicvoid code();

}

 

class Student implements Coder {

 

         @Override

         publicvoid code() {

                   System.out.println("javase");

                   System.out.println("javaweb");

         }

        

}

class HeiMaStudent implements Coder {

         //1,获取被装饰类的引用

         privateStudent s;    //获取学生引用

        

         //2,在构造方法中传入被装饰类的对象

         publicHeiMaStudent(Student s) {

                   this.s= s;   //this指的是当前类,也就是HeiMaStudent,Student s=new Student()!!!

         }

 

         //3,对原有的功能进行升级

         @Override

         publicvoid code() {

                   s.code();    //s调用的是Student中的code方法!!!!

                   System.out.println("ssh");

                   System.out.println("数据库");

                   System.out.println("大数据");

                   System.out.println("...");

         }

}

 

12_IO流(使用指定的码表读写字符)

* FileReader是使用默认码表读取文件, 如果需要使用指定码表读取,那么可以使用InputStreamReader(字节流,编码表),InputStreamReader是字节流通向字符流的桥梁

* FileWriter是使用默认码表写出文件, 如果需要使用指定码表写出,那么可以使用OutputStreamWriter(字节流,编码表),OutputStreamWriter是字符流通向字节流的桥梁

public class Demo7_TransIO {

         *先将utf-8.txt文件的编码表设为utf-8编码表(平台默认的是gbk)!!!!!!

   public static void main(String[] args) throws IOException {

         //demo1();

         //demo2();

         BufferedReaderbr =                                                      //更高效的读

              new BufferedReader(newInputStreamReader(new FileInputStream("utf-8.txt"),"utf-8"));

         BufferedWriterbw =                                                     //更高效的写

              new BufferedWriter(newOutputStreamWriter(new FileOutputStream("gbk.txt"),"gbk"));      eclipse默认的编码表为gbk,所以该处可以省略!!!

                   intc;

                   while((c= br.read()) != -1) {

                            bw.write(c);

                   }                

                   br.close();

                   bw.close();

         }

 

  public static void demo2() throws IOException {

         InputStreamReaderisr = new InputStreamReader(new FileInputStream("utf-8.txt"),"uTf-8");  //指定码表读字符

         OutputStreamWriterosw = new OutputStreamWriter(new FileOutputStream("gbk.txt"),"gbk");       //指定码表写字符

                  

         intc;

         while((c= isr.read()) != -1) {

                   osw.write(c);

         }

                  

         isr.close();

         osw.close();

}

 

   public static void demo1() throws IOException {

         //用默认编码表读写,出现乱码

         FileReaderfr = new FileReader("utf-8.txt");                              

         FileWriterfw = new FileWriter("gbk.txt");

                  

         intc;

         while((c= fr.read()) != -1) {

                   fw.write(c);

         }

                  

         fr.close();

         fw.close();

  }

}


用字节流读而不用字符流读,因为字符流相当于内置了一个编码表,如果你用字符流的话就相当于用平台默认的编码表gbk,去转换utg-8,就会乱码

13_IO流(获取文本上字符出现的次数)

* 获取一个文本上每个字符出现的次数,将结果写在times.txt上

 

public static void main(String[] args)throws IOException {

                   //1,创建带缓冲的输入流对象

                   BufferedReaderbr = new BufferedReader(new FileReader("zzz.txt"));

                   //2,创建双列集合对象TreeMap

                   TreeMap<Character,Integer> tm = new TreeMap<>();

                   //3,将读到的字符存储在双列集合中,存储的时候要做判断,如果不包含这个键,就将键和1存储,如果包含这个键,就将该键和值加1存储

                   intch;

                   while((ch= br.read()) != -1) {

                            charc = (char)ch;       //强制类型转换

                            /*if(!tm.containsKey(c)){

                                     tm.put(c,1);

                            }else{

                                     tm.put(c,tm.get(c) + 1);

                            }*/

                            tm.put(c,!tm.containsKey(c) ? 1 : tm.get(c) + 1);

                   }

                   //4,关闭输入流

                   br.close();

                   //5,创建输出流对象

                   BufferedWriterbw = new BufferedWriter(new FileWriter("times.txt"));

                   //6,遍历集合将集合中的内容写到times.txt中

                   for(Characterkey : tm.keySet()) {

                            switch(key) {

                            case'\t':

                                     bw.write("\\t"+ "=" + tm.get(key));      

                                     break;

                            case'\n':

                                     bw.write("\\n"+ "=" + tm.get(key));

                                     break;

                            case'\r':

                                     bw.write("\\r"+ "=" + tm.get(key));

                                     break;

                            default:

                                     bw.write(key+ "=" + tm.get(key));         //写出键和值

                                     break;

                            }

                            bw.newLine();

                   }

                   //7,关闭输出流

                   bw.close();

         }

 

14_IO流(试用版软件)

* 当我们下载一个试用版软件,没有购买正版的时候,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版

          * 分析:

          * 1,创建带缓冲的输入流对象,因为要使用readLine方法,可以保证数据的原样性

          * 2,将读到的字符串转换为int数

          * 3,对int数进行判断,如果大于0,就将其--写回去,如果不大于0,就提示请购买正版

          * 4,在if判断中要将--的结果打印,并将结果通过输出流写到文件上

         publicstatic void main(String[] args) throws IOException {

                   //1,创建带缓冲的输入流对象,因为要使用readLine方法,可以保证数据的原样性

                   BufferedReaderbr = new BufferedReader(new FileReader("config.txt"));

                   //2,将读到的字符串转换为int数

                   Stringline = br.readLine();

                   inttimes = Integer.parseInt(line);   //将数字字符串转换为数字

                   //3,对int数进行判断,如果大于0,就将其--写回去,如果不大于0,就提示请购买正版

                   if(times> 0) {

                            //4,在if判断中要将--的结果打印,并将结果通过输出流写到文件上

                            System.out.println("您还有" +times-- + "次机会");

                            FileWriterfw = new FileWriter("config.txt");

                            fw.write(times+ "");!!!!!!!

                            fw.close();  //忘关流的话,第二次运行会报错,因为数据还在缓冲区,没有写到文件去!!!

                   }else{

                            System.out.println("您的试用次数已到,请购买正版");

                   }

                   //关闭流

                   br.close();

         }

 

15_File类(递归)方法自己调用自己

          * 递归的弊端:不能调用次数过多,容易导致栈内存溢出

          * 递归的好处:不用知道循环次数

          *

          * 构造方法是否可以递归调用?

          * 构造方法不能使用递归调用

          *

          * 递归调用是否必须有返回值?

          * 不一定(可以有,也可以没有)

* 5的阶乘

public static void main(String[] args) {

                   /*intresult = 1;

                  

                   for(inti = 1; i <= 5; i++) {

                            result= result * i;

                   }

                  

                   System.out.println(result);*/

                   System.out.println(fun(6000));

         }

        

         publicstatic int fun(int num) {

                   if(num== 1) {

                            return1;

                  }else {

                            returnnum * fun(num - 1);

                   }

         }       

 

16_File类(练习)

* 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.java文件名

         publicstatic void main(String[] args) {

                   Filedir = getDir();

                   printJavaFile(dir);

         }

 

         /*

          * 获取键盘录入的文件夹路径

          * 1,返回值类型File

          * 2,不需要有参数

          */

         publicstatic File getDir() {

                   Scannersc = new Scanner(System.in);    //创建键盘录入对象

                   System.out.println("请输入一个文件夹路径");

                   while(true){

                            Stringline = sc.nextLine();        //将键盘录入的文件夹路径存储

                            Filedir = new File(line);    //封装成File对象

                            if(!dir.exists()){

                                     System.out.println("您录入的文件夹路径不存在,请重新录入");

                            }elseif(dir.isFile()) {

                                     System.out.println("您录入的是文件路径,请重新录入文件夹路径");

                            }else{

                                     returndir;

                            }

                   }

         }

         /*

          * 获取文件夹路径下的所.java文件

          * 1,返回值类型 void

          * 2,参数列表File dir

          */

         publicstatic void printJavaFile(File dir) {

                   //1,获取到该文件夹路径下的所有的文件和文件夹,存储在File数组中

                   File[]subFiles = dir.listFiles();

                   //2,遍历数组,对每一个文件或文件夹做判断

                   for(File subFile : subFiles) {

                            //3,如果是文件,并且后缀是.java的,就打印

                            if(subFile.isFile()&& subFile.getName().endsWith(".java")) {

                                     System.out.println(subFile);

                            //4,如果是文件夹,就递归调用

                            }elseif (subFile.isDirectory()){

                                     printJavaFile(subFile);

                            }

                   }

         }

}

 

17_IO流(总结)

public FileWriter(File file,booleanappend)   如果第二个参数为 true,则将字节写入文件末尾处,而不是写入文件开始处。

public FileWriter(String fileName, booleanappend)  append - 一个 boolean 值,如果为 true,则将数据写入文件末尾处,而不是写入文件开始处。

 

* 1.会用BufferedReader读取GBK码表和UTF-8码表的字符

* 2.会用BufferedWriter写出字符到GBK码表和UTF-8码表的文件中

* 3.会使用BufferedReader从键盘读取一行


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值