对不起,如果这个问题听起来很愚蠢,我已经编程了太长时间了,所以我有这种新手问题...
我有一个JFrame(来自netBeans的JFrame表单)和一些变量(arrayList,int ...)。经过一个过程,这个变量就会改变。这个过程是通过JCDB驱动程序的mySQL查询,一些带数据的arrayList更新,我用它来填充Jtable ......依此类推......
起初(可怜的我)我做了一个SwingWorker。通过SwingWorker构造函数,我传递这些变量(大约6个变量或多或少),并使用它们进行处理并填充表格。
我以为我能够在覆盖的Done()方法中更新这些变量的值(再次让我感觉不好),而不仅仅是GUI组件。
我在这次失败中学到了很多东西: 1)即使我通过构造函数传递了变量,但这并不意味着它们会在它们来自的地方得到更新。 2)SwingWorker只能返回1个变量,并修改GUI组件。
所以,这是我的主要观点,我该怎样做我想做的事情?我知道它不能用SwingWorker类完成,但是,怎么办呢?
我不想将代码放在鼠标点击事件中,因为它会阻止我的EDT,而不会通知用户发生了什么。
我虽然做了类似的事情:将代码放在鼠标点击事件中,并显示一个Dialog,以便用户知道此刻正在进行一个过程。
private void jButton_calcular_rutaMousePressed(java.awt.event.MouseEvent evt) {
// BOTON CALCULAR RUTA
int index = jL_empGeo.getSelectedIndex();
JOptionPane optionPane = new JOptionPane("Descargando ruta, espere por favor.", JOptionPane.INFORMATION_MESSAGE, JOptionPane.DEFAULT_OPTION, null, new Object[]{}, null);
JDialog dialog = new JDialog();
dialog.setTitle("Descarga");
dialog.setModalityType(Dialog.DEFAULT_MODALITY_TYPE);
dialog.setDefaultCloseOperation(JDialog.DO_NOTHING_ON_CLOSE);
Dimension dim = Toolkit.getDefaultToolkit().getScreenSize();
dialog.setLocation(dim.width / 2 - dialog.getSize().width / 2, dim.height / 2 - dialog.getSize().height / 2);
dialog.setContentPane(optionPane);
dialog.pack();
dialog.setLocationRelativeTo(this);
dialog.setVisible(true);
//
// THE PROCESSING TAKES PLACE HERE
//
//
// mySQL query
//
// update arrayLists and variables
//
// update GUI components
//
//
dialog.setVisible(false);
}
这有什么意义吗?有没有办法在不同的线程中进行处理,然后将arrayLists和变量返回到主线程?
提前感谢您的帮助。
修改
SwingWorker类
package descargas;
import clases.Empleados;
import clases.InfoPuntoRuta;
import com.mysql.jdbc.Connection;
import com.mysql.jdbc.Statement;
import java.awt.Dimension;
import java.awt.Image;
import java.io.UnsupportedEncodingException;
import java.net.MalformedURLException;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.ImageIcon;
import javax.swing.JDialog;
import javax.swing.JLabel;
import javax.swing.SwingWorker;
import javax.swing.table.DefaultTableModel;
import maps.java.Geocoding;
import maps.java.Route;
import maps.java.StaticMaps;
public class ActualizarRuta2 extends SwingWorker<Void, Void> {
int index;
ArrayList<Empleados> arrayEmpleados;
ArrayList<InfoPuntoRuta> listaPuntosRuta;
DefaultTableModel modeloTablaRutas;
JDialog dialog;
ArrayList<String> listaIntermedios;
JLabel jLab_RutaMAP;
ImageIcon icono;
int origenInt;
int destinoInt;
int destinoMax;
public ActualizarRuta2(int index, int origenInt, int destinoInt, int destinoMax, ArrayList<Empleados> arrayEmpleados, ArrayList<InfoPuntoRuta> listaPuntosRuta, DefaultTableModel modeloTablaRutas, JDialog dialog, JLabel jLab_RutaMAP) {
this.index = index;
this.arrayEmpleados = arrayEmpleados;
this.listaPuntosRuta = listaPuntosRuta;
this.modeloTablaRutas = modeloTablaRutas;
this.dialog = dialog;
this.jLab_RutaMAP = jLab_RutaMAP;
this.origenInt = origenInt;
this.destinoInt = destinoInt;
this.destinoMax = destinoMax;
}
@Override
protected Void doInBackground() throws Exception {
listaIntermedios = new ArrayList<String>();
// recogemos la fecha actual
DateFormat formatoFecha = new SimpleDateFormat("yyyy-MM-dd");
Calendar cal = Calendar.getInstance();
String dateMIN = formatoFecha.format(cal.getTime()) + " 00:00:01";
String dateMAX = formatoFecha.format(cal.getTime()) + " 23:59:59";
// GENERAR MAPA DE LA RUTA DEL DIA
Connection conexion;
conexion = conexiondb.ConexionDB.getConnection();
if (conexion != null) {
try {
Statement st;
ResultSet rs;
st = (Statement) conexion.createStatement();
rs = st.executeQuery("SELECT * \n"
+ "FROM position\n"
+ "WHERE nombre = '" + arrayEmpleados.get(index) + "'\n"
+ "AND position_date BETWEEN '" + dateMIN + "' AND '" + dateMAX + "'\n"
+ "ORDER BY position_date;");
// vaciamos la lista
//listaPuntosRuta.clear();
//System.out.println(rs.);
// rellenamos la lista
rs.beforeFirst();
while (rs.next()) {
InfoPuntoRuta punto = new InfoPuntoRuta(rs.getString("nombre"),
rs.getString("position_date"), rs.getDouble("latitud"),
rs.getDouble("longitud"));
Geocoding ObjGeocod = new Geocoding();
ArrayList<String> resultadoCI = ObjGeocod.getAddress(punto.getLatitud(), punto.getLongitud());
String direccion = resultadoCI.get(0);
punto.setDireccion(direccion);
listaIntermedios.add(direccion);
listaPuntosRuta.add(punto);
}
//
// DESCARGAR MAPA RUTA
//
int posicionUltimo = (listaIntermedios.size()) - 1;
String origen, destino;
ArrayList<String> waypoints = new ArrayList<String>();
if (listaIntermedios.size() < 10) {
origen = listaIntermedios.get(0);
destino = listaIntermedios.get(posicionUltimo);
for (int i = 1; i < posicionUltimo; i++) {
waypoints.add(listaIntermedios.get(i));
}
} else {
origen = listaIntermedios.get(0);
destino = listaIntermedios.get(9);
for (int i = 1; i < 9; i++) {
waypoints.add(listaIntermedios.get(i));
}
}
// ArrayList<String> prueba = new ArrayList<String>();
// prueba.add("coruña avenida finisterre 65");
// prueba.add("coruña ronda de outeiro 125");
// prueba.add("coruña avenida del ejercito 20");
Route ObjRout = new Route();
String[][] resultadoRuta = ObjRout.getRoute(origen, destino, waypoints, Boolean.TRUE, Route.mode.driving, Route.avoids.nothing);
//String[][] resultadoRuta = ObjRout.getRoute("Madrid", "Barcelona", prueba, Boolean.TRUE, Route.mode.driving, Route.avoids.nothing);
// String[][] resultadoRuta = ObjRout.getRoute("coruña virrey ossorio 25", "pla y cancela, 16, 15005 la coruña, españa", prueba, Boolean.TRUE, Route.mode.driving, Route.avoids.nothing);
String polylinea = ObjRout.getGeneralPolyline();
StaticMaps ObjStatMap = new StaticMaps();
Image resultadoMapa = ObjStatMap.getStaticMapRoute(new Dimension(585, 405), 1, StaticMaps.Format.png, StaticMaps.Maptype.hybrid, polylinea);
icono = new ImageIcon(resultadoMapa);
// RELLENAR TABLA DETALLES DE LA RUTA
modeloTablaRutas.setRowCount(0);
} catch (SQLException | UnsupportedEncodingException | MalformedURLException ex) {
Logger.getLogger(ActualizarRuta2.class.getName()).log(Level.SEVERE, null, ex);
} finally {
try {
conexion.close();
} catch (SQLException ex) {
Logger.getLogger(ActualizarRuta2.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
return null;
}
@Override
protected void done() {
dialog.dispose();
// RELLENAR TABLA DETALLES DE LA RUTA
for (InfoPuntoRuta p : listaPuntosRuta) {
modeloTablaRutas.addRow(new Object[]{
p.getNombre(), p.getFecha(), p.getDireccion()});
}
// ponemos la imagen de la ruta
jLab_RutaMAP.setIcon(icono);
//
listaIntermedios.clear();
// listaPuntosRuta.clear();
origenInt = 0;
destinoMax = listaPuntosRuta.size() - 1;
if (destinoMax < 9) {
destinoInt = destinoMax;
} else {
destinoInt = 9;
}
}
}
我调用SwingWorker的鼠标事件
private void jButton_calcular_rutaMousePressed(java.awt.event.MouseEvent evt) {
// BOTON CALCULAR RUTA
int index = jL_empGeo.getSelectedIndex();
JOptionPane optionPane = new JOptionPane("Descargando ruta, espere por favor.", JOptionPane.INFORMATION_MESSAGE, JOptionPane.DEFAULT_OPTION, null, new Object[]{}, null);
JDialog dialog = new JDialog();
dialog.setTitle("Descarga");
dialog.setModalityType(Dialog.DEFAULT_MODALITY_TYPE);
dialog.setDefaultCloseOperation(JDialog.DO_NOTHING_ON_CLOSE);
Dimension dim = Toolkit.getDefaultToolkit().getScreenSize();
dialog.setLocation(dim.width / 2 - dialog.getSize().width / 2, dim.height / 2 - dialog.getSize().height / 2);
dialog.setContentPane(optionPane);
dialog.pack();
dialog.setLocationRelativeTo(this);
listaPuntosRuta.clear();
modeloTablaRutas.setRowCount(0);
ActualizarRuta2 task = new ActualizarRuta2(index, origenInt, destinoInt, destinoMax, arrayEmpleados, listaPuntosRuta, modeloTablaRutas, dialog, jLab_RutaMAP);
task.execute();
dialog.setVisible(true);
}
答案 0 :(得分:2)
您需要通过SwingWorker&#39;与GUI进行通信。 Java参数按值传递 - 因此,您需要某种形式的值持有者。
制作数据bean&#39;从Observable下降,可能是最简单的方法。
public class QueryResults extends Observable {
protected List<SomeItem> itemList;
// getters and setters.
}
从SwingWorker中,您需要存储结果&amp;通知观察员。必须在事件派发线程上完成通知,并且javax.swing.SwingWorker提供了在EDT上调用SwingWorker.done()的内置功能。</ p>
我使用过&#39; Void&#39;类型为&#34;结果&#34;,因为数据持有者已经存在&amp;我们只是将数据加载到其中。 UI应该已经绑定(并观察)它。像这样:
public class QueryWorker<Void,Void> extends SwingWorker {
protected QueryResults holder;
public QueryWorker (QueryResults holder) {
this.holder = holder; // keep the holder & put data into it.
}
@Override
protected Void doInBackground() {
// run the query..
}
@Override
protected void done() {
holder.notifyObservers();
}
}
这是一个简化的例子。例如,它忽略了想要在加载过程中读取结果hodler(用于显示刷新)的UI的可能性。另一种方法是将数据bean 与 observable reference 分开,保存到bean中。
答案 1 :(得分:1)
有没有办法在不同的线程中进行处理,然后将arrayLists和变量重新放回主线程?
是的,有。您可以(并且可能应该)在dialog.setVisible(true)
之后使用
new Thread() {
@Override
public void run() {
getNewValues();
}
}.start();
完成这项工作后,您可以从“工人”线程调用
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
//overwrite existing variables, update GUI elements, trigger repaints, etc.
dialog.setVisible(false);
}
};
但是,您应该始终尝试将数据与视图分开。这意味着:您可能不应该从工作线程调用这些操作,而是实现observer pattern以通知GUI数据已更改并自行刷新。