подобная Python библиотека Java IO? - PullRequest
18 голосов
/ 10 мая 2010

Ява не мой основной язык программирования, поэтому я могу спросить очевидное.

Но есть ли в Java простая библиотека для обработки файлов, как в python ?

Например, я просто хочу сказать:

File f = Open('file.txt', 'w')
for(String line:f){
      //do something with the line from file
}

Спасибо!

ОБНОВЛЕНИЕ: Хорошо, стековый поток автоматически принял странный ответ. Это связано с вознаграждением, которое я разместил - поэтому, если вы хотите увидеть другие ответы, просто прокрутите вниз!

Ответы [ 10 ]

19 голосов
/ 10 мая 2010

Я думал о чем-то более похожем на:

File f = File.open("C:/Users/File.txt");

for(String s : f){
   System.out.println(s);
}

Вот мой исходный код:

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.Writer;
import java.util.Iterator;

public abstract class File implements Iterable<String>{
    public final static String READ = "r";
    public final static String WRITE = "w";

    public static File open(String filepath) throws IOException{
        return open(filepath, READ);
    }   

    public static File open(String filepath, String mode) throws IOException{
    if(mode == READ){
        return new ReadableFile(filepath);
    }else if(mode == WRITE){
        return new WritableFile(filepath);
    }
    throw new IllegalArgumentException("Invalid File Write mode '" + mode + "'");
    }

    //common methods
    public abstract void close() throws IOException;

    // writer specific
    public abstract void write(String s) throws IOException;

}

class WritableFile extends File{
    String filepath;
    Writer writer;

    public WritableFile(String filepath){
        this.filepath = filepath;
    }

    private Writer writer() throws IOException{
        if(this.writer == null){
            writer = new BufferedWriter(new FileWriter(this.filepath));
        }
        return writer;
    }

    public void write(String chars) throws IOException{
        writer().write(chars);
    }

    public void close() throws IOException{
        writer().close();
    }

    @Override
    public Iterator<String> iterator() {        
        return null;
    }
}

class ReadableFile extends File implements Iterator<String>{
    private BufferedReader reader;
    private String line;    
    private String read_ahead;

    public ReadableFile(String filepath) throws IOException{        
        this.reader = new BufferedReader(new FileReader(filepath)); 
        this.read_ahead = this.reader.readLine();
    }

    private Reader reader() throws IOException{
         if(reader == null){
               reader = new BufferedReader(new FileReader(filepath));   
         }
         return reader;
    }

    @Override
    public Iterator<String> iterator() {
        return this;
    }

    @Override
    public void close() throws IOException {
        reader().close();
    }

    @Override
    public void write(String s) throws IOException {
        throw new IOException("Cannot write to a read-only file.");
    }

    @Override
    public boolean hasNext() {      
        return this.read_ahead != null;
    }

    @Override
    public String next() {
        if(read_ahead == null)
            line = null;
        else
            line = new String(this.read_ahead);

        try {
            read_ahead = this.reader.readLine();
        } catch (IOException e) {
            read_ahead = null;
            reader.close()
        }
        return line;
    }

    @Override
    public void remove() {
        // do nothing       
    }
}

и вот для него юнит-тест:

import java.io.IOException;
import org.junit.Test;

public class FileTest {
    @Test
    public void testFile(){
        File f;
        try {
            f = File.open("File.java");
            for(String s : f){
                System.out.println(s);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testReadAndWriteFile(){
        File from;
        File to;
        try {
            from = File.open("File.java");
            to = File.open("Out.txt", "w");
            for(String s : from){           
                to.write(s + System.getProperty("line.separator"));
            }
            to.close();
        } catch (IOException e1) {
            e1.printStackTrace();
        }   
    }
}
11 голосов
/ 10 мая 2010

Чтение файла построчно в Java:

BufferedReader in = new BufferedReader(new FileReader("myfile.txt"));

String line;
while ((line = in.readLine()) != null) {
    // Do something with this line
    System.out.println(line);
}

in.close();

Большинство классов для ввода / вывода находятся в пакете java.io. Смотрите документацию API для этого пакета. Для получения более подробной информации ознакомьтесь с Руководством по вводу-выводу Sun .

добавление : В приведенном выше примере для чтения текстового файла будет использоваться кодировка символов по умолчанию вашей системы. Если вы хотите явно указать кодировку символов, например UTF-8, измените первую строку следующим образом:

BufferedReader in = new BufferedReader(
    new InputStreamReader(new FileInputStream("myfile.txt"), "UTF-8"));
4 голосов
/ 19 мая 2010

Если вы хотите перебирать файл по строкам, класс, который вам может пригодиться, это класс Scanner .

import java.io.*;
import java.util.Scanner;

    public class ScanXan {
        public static void main(String[] args) throws IOException {
            Scanner s = null;
            try {
                s = new Scanner(new BufferedReader(new FileReader("myFile.txt")));

                while (s.hasNextLine()) {
                    System.out.println(s.nextLine());
                }
            } finally {
                if (s != null) {
                    s.close();
                }
            }
        }
    }

API довольно полезен: http://java.sun.com/javase/7/docs/api/java/util/Scanner.html Вы также можете проанализировать файл с помощью регулярных выражений.

4 голосов
/ 10 мая 2010

Если у вас уже есть зависимости от Apache commons lang и commons io , это может быть альтернативой:

String[] lines = StringUtils.split(FileUtils.readFileToString(new File("myfile.txt")), '\n');
for(String line: lines){
      //do something with the line from file
}

(я бы предпочел ответ Джеспера)

3 голосов
/ 19 мая 2010

Мне никогда не надоест сутенерство библиотек гуавы от Google , которое избавляет от многих проблем ... ну, большинство вещей в Java.

Как насчет:

for (String line : Files.readLines(new File("file.txt"), Charsets.UTF_8)) {
   // Do something
}

В случае, когда у вас есть большой файл, и вы хотите построчный обратный вызов (а не чтение всего этого в памяти), вы можете использовать LineProcessor, который добавляет немного шаблона (из-за отсутствия замыканий ... вздох), но все же не дает вам разобраться с самим чтением и всеми связанными Exceptions:

int matching = Files.readLines(new File("file.txt"), Charsets.UTF_8, new LineProcessor<Integer>(){
  int count;

  Integer getResult() {return count;}

  boolean processLine(String line) {
     if (line.equals("foo")
         count++;
     return true;
  }
});

Если вы на самом деле не хотите, чтобы результат возвращался из процессора, и вы никогда не прерывали работу раньше (причина логического возврата из processLine), вы могли бы , тогда сделать что-то вроде:

class SimpleLineCallback extends LineProcessor<Void> {
    Void getResult{ return null; }

    boolean processLine(String line) {
       doProcess(line);
       return true;
    }

    abstract void doProcess(String line);
}

и тогда ваш код может быть:

Files.readLines(new File("file.txt"), Charsets.UTF_8, new SimpleLineProcessor(){
  void doProcess(String line) {
     if (line.equals("foo");
         throw new FooException("File shouldn't contain 'foo'!");
  }
});

что соответственно чище.

2 голосов
/ 23 мая 2010
  public static void main(String[] args) throws FileNotFoundException {
    Scanner scanner = new Scanner(new File("scan.txt"));
    try {
      while (scanner.hasNextLine()) {
        System.out.println(scanner.nextLine());
      }
    } finally {
      scanner.close();
    }
  }

Некоторые предостережения:

  • Это использует системную кодировку по умолчанию, но вы должны указать кодировку файла
  • Сканер проглатывает исключения ввода-вывода, поэтому вы можете проверить ioException () в конце для правильной обработки ошибок
1 голос
/ 19 мая 2010

Простой пример использования Files.readLines() из guava-io с обратным вызовом LineProcessor:

import java.io.File;
import java.io.IOException;

import com.google.common.base.Charsets;
import com.google.common.io.Files;
import com.google.common.io.LineProcessor;

public class GuavaIoDemo {

    public static void main(String[] args) throws Exception {
        int result = Files.readLines(new File("/home/pascal/.vimrc"), //
            Charsets.UTF_8, // 
            new LineProcessor<Integer>() {
                int counter;

                public Integer getResult() {
                    return counter;
                }

                public boolean processLine(String line) throws IOException {
                    counter++;
                    System.out.println(line);
                    return true;
                }
            });
    }
}
0 голосов
/ 25 мая 2010

Попробуйте взглянуть на заводной!

Это расширенный набор Java, который работает в JVM. Наиболее допустимый код Java также является допустимым Groovy, поэтому у вас есть доступ к любому из миллиона API Java напрямую.

Кроме того, он имеет много конструкций более высокого уровня, знакомых Pythonists, плюс ряд расширений, которые помогут избавиться от карт, списков, sql, xml, и вы уже догадались - файл IO.

0 голосов
/ 19 мая 2010

Хороший пример здесь: Строковая итерация

0 голосов
/ 19 мая 2010

Вы можете использовать jython , который позволяет запускать синтаксис Python в Java.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...