线程写入本地txt文件时出现问题

时间:2019-08-26 05:43:18

标签: java multithreading bufferedwriter

我试图使用隔离的此类从许多源中编写文本,而不每次都打开一个实例或文件连接,所以我添加到ArrayList并在隔离的线程中工作以查看该行是否,因此服务类可以尽早完成写入信息的工作,从而避免客户端类等待其时间来使用writer类。此类的问题是我正在构造函数中执行线程,但是该线程在执行方法write(gerenciadorDeArquivos()。escreveEmArquivo()的第一次调用中停止,它在文件中写入了一次,仅此而已,当我将程序调试时,如果我不这样做,他甚至不会执行一次,也许您可​​以提供一些见解。

主表单类

package EscreveArquivo;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;


public class Tela extends javax.swing.JFrame {

public Tela() {
    initComponents();
}
 private void initComponents() {....} /// Form Components

 EscreveArquivo escritorDeArquivos = new EscreveArquivo();

   private void recordTextButtonActionPerformed(java.awt.event.ActionEvent evt) {                                                 
    escritorDeArquivos.AdicionaArquivoEmListaDeInstrucoes(textField.getText());
}               

第二堂课

package Escrevearquivo;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;*

public class EscreveArquivo {

private static List<String> listaDeInstrucoes = new ArrayList<String>();
private boolean gerenciadorEstaEmExecucao = false;

EscreveArquivo() {
    Thread t = new Thread() {   
        public void run() //Anonymous class overriding run() method of Thread class
        {
            while (true) {
                if (!listaDeInstrucoes.isEmpty()) {
                    gerenciadorDeArquivos();
                }
            }
        }
    };
    t.start();
}

public static void AdicionaArquivoEmListaDeInstrucoes(String arquivoASerEscrito) {
    listaDeInstrucoes.add(arquivoASerEscrito);
}

private void gerenciadorDeArquivos() {

    if (gerenciadorEstaEmExecucao == false) {
        gerenciadorEstaEmExecucao = true;

        ArrayList<String> listaDeInstrucoesCopia = new ArrayList<String>();
        for (String textoAEscrever : listaDeInstrucoes) {
            listaDeInstrucoesCopia.add(textoAEscrever);
        }
        for (String textoAApagar : listaDeInstrucoesCopia) {
            EscreveEmArquivo(textoAApagar);
            listaDeInstrucoesCopia.remove(textoAApagar);
        }
    }
    gerenciadorEstaEmExecucao = false;
}

private static void EscreveEmArquivo(String texto) {
    File arquivo = new File("C://Users//Josué//Desktop//arquivoTexto.txt");
    try {
        if (!arquivo.exists()) {
            arquivo.createNewFile();
        }
        FileWriter writer = new FileWriter(arquivo, true);
        BufferedWriter saida = new BufferedWriter(writer);
        saida.write(texto);
        saida.newLine();
        saida.close();

    } catch (IOException ex) {
        Logger.getLogger(Tela.class
                .getName()).log(Level.SEVERE, null, ex);
    }
}

}

1 个答案:

答案 0 :(得分:1)

本节

for(String textoAApagar : listaDeInstrucoesCopia) {
   EscreveEmArquivo(textoAApagar);
   listaDeInstrucoesCopia.remove(textoAApagar); //error will be thrown here!
}

实际上会为ConcurrentModificationException生成错误,因为您正在遍历列表listaDeInstrucoesCopia时对其进行了修改。因此,在这种情况下,该呼叫将不会写入第二个字段。只有第一个调用会被执行。

删除行listaDeInstrucoesCopia.remove(textoAApagar);,它应该可以正常工作。但同样,您不会清空ArrayList,因此旧值将被重写。

您应该考虑使用Queue实现,并弹出end元素并连续写入文件。

相关问题