IO流(三)

来源:互联网 发布:电脑桌面提示语软件 编辑:程序博客网 时间:2024/05/17 02:15

File对象

File类用于将文件或者文件夹封装成对象,方便对文件和文件夹的属性信息进行操作。File类对象可以作为参数传递给流的构造函数。

构造函数

四种构造函数的代码示例:

public class FileDemo {    public static void main(String[] args) {        constructorDemo();    }    public static void constructorDemo() {        ///可以将一个已存在的或者不存在的文件或者目录封装成file对象。        File f1 =new File("c:\\a.txt");        File f2 =new File("c:\\","a.txt");        File f=new File("c:\\");        File f3=new File(f,"a.txt");//      File f4=new File("c:"+System.getProperty("file.separator")+"a.txt");        File f4=new File("c:"+File.separator+"a.txt");    }}

常见方法

File对象的常见方法有
1.获取、创建、删除、判断以及重命名。

public class FileMethodDemo {    public static void main(String[] args) throws IOException {/*              File对象的常见方法            1.获取                                     getDemo()                获取文件名称                获取文件路径                获取文件大小                获取文件修改时间            2.创建与删除   boolean           createAndDeleteDemo()            3.判断                        isDemo()             4.重命名                       renameToDemo()*///      getDemo();//      createAndDeleteDemo();//      isDemo();        renameToDemo();    }    public static void renameToDemo() {        File f1=new File("c:\\0.mp3");        File f2=new File("d:\\9.mp3");        boolean b=f1.renameTo(f2);        System.out.println("rename:"+b);    }    public static void isDemo() throws IOException {        File f=new File("a.txt");        boolean b=f.exists();/*              //带.txt的不一定是文件,不带后缀名的也不一定是目录        File f1=new File("aaa.txt");        f1.mkdir();        File f2=new File("aaa");        f2.createNewFile();        */        System.out.println("is exists:"+b);        //最好先判断是否存在,在判断是否目录。文件等        System.out.println(f.isFile());        System.out.println(f.isDirectory());        System.out.println(f.isHidden());        System.out.println(f.isAbsolute());    }    public static void createAndDeleteDemo() throws IOException {        File dir=new File("abc");        boolean d=dir.mkdir();//make directory        System.out.println("create dir:"+d);        System.out.println("delete dir:"+dir.delete());//如果目录里面有内容,则删除不了,因为Windows只能从里往外删除        File dirs=new File("abc\\a\\b\\c\\d\\e");//创建多级目录        boolean d1=dir.mkdirs();//make directorys        System.out.println("create dirs:"+d1);        System.out.println("delete dirs:"+dirs.delete());//删除的的目录e,前面都是父目录//      文件的创建与删除        File file=new File("file.txt");        /*               和输出流不一样,如果文件不存在,则创建,如果存在则不创建        */          boolean b=file.createNewFile();        System.out.println("create:"+b);        boolean b1=file.delete();        System.out.println("delete:"+b1);        }    public static void getDemo(){//      File file=new File("D:\MyEclipse Professional 2014\day22e\a.txt");        File file=new File("a.txt");        String name=file.getName();        String path=file.getAbsolutePath();        String path1=file.getPath();        long len=file.length();        long time=file.lastModified();        Date date=new Date(time);        DateFormat dateFormat=DateFormat.getDateTimeInstance(DateFormat.LONG,DateFormat.LONG);        String str_time=dateFormat.format(date);        System.out.println("name:"+name);        System.out.println("absPath:"+path);        System.out.println("path:"+path1);        System.out.println("len:"+len);        System.out.println("time:"+time);        System.out.println("str_time:"+str_time);        System.out.println("parent path:"+file.getParent());//当实例化时参数为相对目录时返回空,为绝对目录时返回父目录。    }}

2.获取系统根目录和容量

public class FileMethodDemo {    public static void main(String[] args) throws IOException {        listRootsDemo();    }    public static void listRootsDemo() {        File[] files=File.listRoots();        for(File file:files){            System.out.println(file);        }        File f=new File("d:\\");        System.out.println("FreeSpace:"+f.getFreeSpace());        System.out.println("TotalSpace:"+f.getTotalSpace());        System.out.println("UsableSpace:"+f.getUsableSpace());    }}

输出:
C:\
D:\
E:\
F:\
G:\
FreeSpace:241087184896
TotalSpace:371055390720
UsableSpace:241087184896

3.获取目录内容
调用list方法的File对象中封装的必须是目录,否则会发生NullPointerException。
如果访问的是系统级目录,也会发生空指针异常。
如果目录存在但是没有内容,会返回一个数组,但是长度为0。

public class FileMethodListDemo {    public static void main(String[] args) {        listDemo();    }    public static void listDemo() {        File file=new File("c:\\");        /*        函数功能:当前目录下的文件以及文件夹的名称,包含隐藏文件        */        String[] names=file.list();        for(String name:names){            System.out.println(name);        }    }}

4.过滤器
通过过滤器可以只获取那些带有固定特征的文件或者目录,例如获取后缀为”.java”的文件,或者隐藏文件。

后缀名为”.java”的过滤器:

public class FilterByJava implements FilenameFilter {    @Override    public boolean accept(File dir, String name) {        return name.endsWith(".java");    }}

将后缀名作为参数的过滤器:

public class SuffixFilter implements FilenameFilter {    private String suffix;    public SuffixFilter(String suffix) {        super();        this.suffix = suffix;    }    @Override    public boolean accept(File dir, String name) {        return name.endsWith(suffix);    }}

隐藏文件的过滤器:

public class FilterByHidden implements FileFilter {    @Override    public boolean accept(File pathname) {        return !pathname.isHidden();    }}

使用以上过滤器的实例:

public class FileMethodListDemo {    public static void main(String[] args) {        listDemo_2();        listDemo_3();    }    public static void listDemo_3() {        File dir=new File("c:\\");        File[] files=dir.listFiles(new FilterByHidden());        for(File file:files){            System.out.println(file);        }    }    public static void listDemo_2() {        File dir=new File("c:\\");//      String[] names=dir.list(new FilterByJava());        String[] names=dir.list(new SuffixFilter(".java"));        for(String name:names){            System.out.println(name);        }    }}

深度遍历

需求一:对指定目录进行所有内容的列出(包含子目录的内容)
即进行深度遍历。

public class FileTest {    public static void main(String[] args) {        File dir=new File("E:\\javawork");        listAll(dir,0);    }    public static void listAll(File dir,int level) {//      System.out.println("dir:"+dir.getAbsolutePath());        System.out.println(getSpace(level)+dir.getName());        level++;        //获取指定目录下当前的所有文件夹或者文件对象        File[] files=dir.listFiles();        //        for(int x=0;x<files.length;x++){            if(files[x].isDirectory()){                listAll(files[x],level);            }            else                System.out.println(getSpace(level)+files[x].getName());        }    }    private static String getSpace(int level) {        StringBuilder sb=new StringBuilder();        sb.append("|--");        for(int x=0;x<level;x++){            sb.insert(0,"|  ");        }        return sb.toString();    }}

递归
函数自身直接或间接地调用到自身就是递归。
一个功能在被重复使用,并每次使用时,参与运算的结果和上一次调用有关,这时可以用递归来解决问题。
注意:
1.递归一定要明确条件,否则容易栈溢出。
2.注意递归的次数。

需求二:删除一个带内容的目录。
原理:必须要从最里面往外删,需要进行深度遍历。

public class RemoveDirTest {    public static void main(String[] args) {        File dir=new File("e:\\demodir");        removeDir(dir);    }    public static void removeDir(File dir) {        File[] files=dir.listFiles();        for(File file:files){            if(file.isDirectory()){                removeDir(file);            }            else{                System.out.println(file+":"+file.delete());file.delete();            }        }        System.out.println(dir+":"+dir.delete());    }}

Properties集合

Properties集合是Hashtable集合下的一个子类:
Map–>Hashtable->Properties

Properties表示一个持久的属性集,该集合用于操作以键值对形式存在的配置文件。

Properties集合的特点:
1. 可以保存在流中或者从流中加载。
2. 属性列表中每个键及其对应值都是一个字符串。

Properties集合的常见使用:

public class PropertiesDemo {    public static void main(String[] args) throws IOException {//      propertiesDemo();//      methodDemo_2();//      methodDemo_3();//      methodDemo_4();//      myLoad();        test();    }    //对已有的配置文件中的信息进行修改    /*    读取这个文件    并将这个文件中的键值数据存储到集合中    再通过集合对数据进行修改    再通过流将修改后的数据存储到文件中    */    public static void test() throws IOException{        //读取这个文件        File file=new File("info.txt");        if(!file.exists()){            file.createNewFile();        }        FileReader fr=new FileReader(file);//流与数据相关联//      FileWriter fw=new FileWriter(file);//不能写在这里,因为这样是新创建了一个一个文件,原文件的数据被覆盖了,里面没数据        //创建集合存储配置信息        Properties prop =new Properties();        //将流中的信息存储到集合中        prop.load(fr);//      prop.list(System.out);//      通过集合对数据进行修改        prop.setProperty("marry", "21");//      prop.list(System.out);//      通过流将修改后的数据存储到文件中        FileWriter fw=new FileWriter(file);        prop.store(fw,"info");        fw.close();        fr.close();    }//  模拟load方法    public static void myLoad() throws IOException{        Properties prop=new Properties();        BufferedReader bufr=new BufferedReader(new FileReader("info.txt"));        String line=null;        while((line=bufr.readLine())!=null){            if(line.startsWith("#"))                continue;            String[] arr=line.split("=");//          System.out.println(arr[0]+"::"+arr[1]);            prop.setProperty(arr[0], arr[1]);        }        prop.list(System.out);        bufr.close();    }    public static void methodDemo_4() throws IOException {        Properties prop=new Properties();        /*        集合中的数据来自一个文件        注意:必须保证该文件的数据是键值对        需要使用到读取流        */        FileInputStream fis=new FileInputStream("info.txt");        //使用load方法        prop.load(fis);        prop.list(System.out);    }    public static void methodDemo_3() throws IOException {        Properties prop=new Properties();        //存储元素        prop.setProperty("marry", "28");        prop.setProperty("tom", "14");        prop.setProperty("jerry", "12");        prop.setProperty("lisa", "24");        /*         * 想要将这些集合中的字符串键值信息持久化存储到文件中         * 需要关联输出流         *          * */        FileOutputStream fos=new FileOutputStream("info.txt");        //将集合中的数据存储到文件中,使用store方法        prop.store(fos, "name+age");        fos.close();    }    /*    演示Properties集合和流对象相结合的功能    */    public static void methodDemo_2(){        Properties prop=new Properties();        //存储元素        prop.setProperty("marry", "28");        prop.setProperty("tom", "14");        prop.setProperty("jerry", "12");        prop.setProperty("lisa", "24");        //public void list(PrintStream out):将属性列表输出到指定的输出流        prop.list(System.out);//集合中的数据输出到输出流控制台上。调试用,开发用得少        Properties prop1=System.getProperties();        prop1.list(System.out);    }    /*    Properties集合的存和取    */    public static void propertiesDemo(){        //创建一个Properties集合        Properties prop=new Properties();        //存储元素        prop.setProperty("marry", "28");        prop.setProperty("tom", "14");        prop.setProperty("jerry", "12");        prop.setProperty("lisa", "24");        //修改元素        prop.setProperty("marry", "18");        //取出所有元素        Set<String> names=prop.stringPropertyNames();        for(String name:names)        {            String value=prop.getProperty(name);            System.out.println(name+":"+value);        }    }}

Properties集合练习
需求:定义功能,获取一个应用程序运行的次数,如果超过5次,给出使用次数已到请注册的提示,并且不要再运行程序。
思路:
1.应该有计数器;
每次程序启动都需要计数一次,并且是在原有的次数上进行计数。
2.计数器就是一个变量,程序启动时进行计数,计数器必须存在于内存中并进行运算。
但是程序一旦结束,计数器消失了,再次启动该程序的时候,计数器又重新被初始化了。

然而我们需要多次启动同一个应用程序,使用的是同一个计数器。
这就需要计数器的生命周期变长,从内存存储到硬盘文件中。

3.如何使用这个计数器:
首先,程序启动时,应该先读取这个用于记录计数器信息的配置文件。
获取上一次计数器次数,并进行试用次数的判断。
其次,对该次数进行自增,并将自增后的次数重新存储到配置文件中。

4.文件中的信息该如何进行存储并体现:
直接存储此数值可以,但是不明确该数据的含义,所以应该起一个名字。
(简单数据可以这么体现,用键值对。但是复杂数据不可以,要想进行更贴切的描述,应该用xml文件。)
这就有了名字和值的对应,所以可以使用键值对。
可以应用有映射关系的map集合搞定,又需要读取硬盘上的数据,所以map+io=Properties。

public class PropertiesTest {    public static void main(String[] args) throws IOException {        getAppCount();    }    public static void getAppCount() throws IOException{        //将配置文件封装成File对象//      .ini在Windows或者常见应用软件里比较常用,对于java制作的软件的键值对的配置信息用.properties作为扩展名        File confile=new File("count.ini");        if(!confile.exists()){            confile.createNewFile();        }        FileInputStream fis=new FileInputStream(confile);        Properties prop=new Properties();        prop.load(fis);        //从集合中通过键获取次数        String value=prop.getProperty("time");        //定义计数器,记录获取到的次数        int count=0;        if(value!=null){            count =Integer.parseInt(value);            if(count>=5){//              System.out.println("使用次数已到,请注册");//              return;                throw new RuntimeException("使用次数已到,请注册");            }        }        count++;        //将改变后的次数重新存储到集合中        prop.setProperty("time", count+"");        FileOutputStream fos=new FileOutputStream(confile);        prop.store(fos,"");        fos.close();        fis.close();    }}

综合练习

需求:获取指定目录下,指定扩展名的文件(包含子目录中的)这些文件的绝对路径写入到一个文本文件中。简单说,就是建立一个指定扩展名的文件的列表。

思路:
1.必须进行深度遍历;
2.要在遍历的过程中进行过滤,将符合条件的内容都存储到容器中;
3.对容器中的内容进行遍历并将绝对路径写入到文件中。

public class Test {    public static void main(String[] args) throws IOException {        File dir=new File("e:\\javawork");        FilenameFilter filter=new FilenameFilter(){            @Override            public boolean accept(File dir, String name) {                return name.endsWith(".java");            }        };        List<File> list=new ArrayList<File>();        getFiles(dir,filter,list);        File destFile=new File(dir,"javalist.txt");        write2File(list,destFile);    }/** * 对指定目录中的内容进行深度遍历,并按照指定的过滤器进行过滤,将过滤后的内容存储到指定容器list中 * @param dir * @param filter * @param list */    public static void getFiles(File dir,FilenameFilter filter,List<File> list){        File[] files=dir.listFiles();        for(File file:files){            if(file.isDirectory()){                //递归                getFiles(file,filter,list);            }else{                //对遍历到的文件文件进行过滤器的过滤,将符合条件的File对象,存储到List集合中。                if(filter.accept(dir, file.getName())){                    list.add(file);                }            }        }    }    public static void write2File(List<File> list,File destFile) throws IOException{        BufferedWriter bufw=null;        try {            bufw=new BufferedWriter(new FileWriter(destFile));            for(File file:list){                bufw.write(file.getAbsolutePath());                bufw.newLine();                bufw.flush();            }        } /*catch (IOException e) {//catch或者抛出            throw new RuntimeException("写入失败");        }*/finally{            if(bufw!=null)                try{                    bufw.close();                }catch(IOException e){                    throw new RuntimeException("关闭失败");                }        }    }}
原创粉丝点击