Skip to content

Instantly share code, notes, and snippets.

@jewelsea
Last active March 5, 2024 22:24
Show Gist options
  • Save jewelsea/4957967 to your computer and use it in GitHub Desktop.
Save jewelsea/4957967 to your computer and use it in GitHub Desktop.
Sample for accessing a local database from JavaFX using concurrent tasks for database operations so that the UI remains responsive.
package h2app;
import javafx.application.Application;
import javafx.collections.*;
import javafx.concurrent.Task;
import javafx.scene.Scene;
import javafx.scene.control.*;
import javafx.scene.layout.*;
import javafx.stage.Stage;
import java.sql.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.logging.Level;
import java.util.logging.Logger;
public class H2Tasks extends Application {
private static final Logger logger = Logger.getLogger(H2Tasks.class.getName());
private static final String[] SAMPLE_NAME_DATA = { "John", "Jill", "Jack", "Jerry" };
public static void main(String[] args) { launch(args); }
// executes database operations concurrent to JavaFX operations.
private ExecutorService databaseExecutor;
// the future's data will be available once the database setup has been complete.
private Future databaseSetupFuture;
// initialize the program.
// setting the database executor thread pool size to 1 ensures
// only one database command is executed at any one time.
@Override public void init() throws Exception {
databaseExecutor = Executors.newFixedThreadPool(
1,
new DatabaseThreadFactory()
);
// run the database setup in parallel to the JavaFX application setup.
DBSetupTask setup = new DBSetupTask();
databaseSetupFuture = databaseExecutor.submit(setup);
}
// shutdown the program.
@Override public void stop() throws Exception {
databaseExecutor.shutdown();
if (!databaseExecutor.awaitTermination(3, TimeUnit.SECONDS)) {
logger.info("Database execution thread timed out after 3 seconds rather than shutting down cleanly.");
}
}
// start showing the UI.
@Override public void start(Stage stage) throws InterruptedException, ExecutionException {
// wait for the database setup to complete cleanly before showing any UI.
// a real app might use a preloader or show a splash screen if this
// was to take a long time rather than just pausing the JavaFX application thread.
databaseSetupFuture.get();
final ListView<String> nameView = new ListView<>();
final ProgressIndicator databaseActivityIndicator = new ProgressIndicator();
databaseActivityIndicator.setVisible(false);
final Button fetchNames = new Button("Fetch names from the database");
fetchNames.setOnAction(event ->
fetchNamesFromDatabaseToListView(
fetchNames,
databaseActivityIndicator,
nameView
)
);
final Button clearNameList = new Button("Clear the name list");
clearNameList.setOnAction(event -> nameView.getItems().clear());
VBox layout = new VBox(10);
layout.setStyle("-fx-background-color: cornsilk; -fx-padding: 15;");
layout.getChildren().setAll(
new HBox(10,
fetchNames,
clearNameList,
databaseActivityIndicator
),
nameView
);
layout.setPrefHeight(200);
stage.setScene(new Scene(layout));
stage.show();
}
private void fetchNamesFromDatabaseToListView(
final Button triggerButton,
final ProgressIndicator databaseActivityIndicator,
final ListView<String> listView) {
final FetchNamesTask fetchNamesTask = new FetchNamesTask();
triggerButton.disableProperty().bind(
fetchNamesTask.runningProperty()
);
databaseActivityIndicator.visibleProperty().bind(
fetchNamesTask.runningProperty()
);
databaseActivityIndicator.progressProperty().bind(
fetchNamesTask.progressProperty()
);
fetchNamesTask.setOnSucceeded(t ->
listView.setItems(fetchNamesTask.getValue())
);
databaseExecutor.submit(fetchNamesTask);
}
abstract class DBTask<T> extends Task<T> {
DBTask() {
setOnFailed(t -> logger.log(Level.SEVERE, null, getException()));
}
}
class FetchNamesTask extends DBTask<ObservableList<String>> {
@Override protected ObservableList<String> call() throws Exception {
// artificially pause for a while to simulate a long running database connection.
Thread.sleep(1000);
try (Connection con = getConnection()) {
return fetchNames(con);
}
}
private ObservableList<String> fetchNames(Connection con) throws SQLException {
logger.info("Fetching names from database");
ObservableList<String> names = FXCollections.observableArrayList();
Statement st = con.createStatement();
ResultSet rs = st.executeQuery("select name from employee");
while (rs.next()) {
names.add(rs.getString("name"));
}
logger.info("Found " + names.size() + " names");
return names;
}
}
class DBSetupTask extends DBTask {
@Override protected Void call() throws Exception {
try (Connection con = getConnection()) {
if (!schemaExists(con)) {
createSchema(con);
populateDatabase(con);
}
}
return null;
}
private boolean schemaExists(Connection con) {
logger.info("Checking for Schema existence");
try {
Statement st = con.createStatement();
st.executeQuery("select count(*) from employee");
logger.info("Schema exists");
} catch (SQLException ex) {
logger.info("Existing DB not found will create a new one");
return false;
}
return true;
}
private void createSchema(Connection con) throws SQLException {
logger.info("Creating schema");
Statement st = con.createStatement();
String table = "create table employee(id integer, name varchar(64))";
st.executeUpdate(table);
logger.info("Created schema");
}
private void populateDatabase(Connection con) throws SQLException {
logger.info("Populating database");
Statement st = con.createStatement();
for (String name: SAMPLE_NAME_DATA) {
st.executeUpdate("insert into employee values(1,'" + name + "')");
}
logger.info("Populated database");
}
}
private Connection getConnection() throws ClassNotFoundException, SQLException {
logger.info("Getting a database connection");
Class.forName("org.h2.Driver");
return DriverManager.getConnection("jdbc:h2:~/test", "sa", "");
}
static class DatabaseThreadFactory implements ThreadFactory {
static final AtomicInteger poolNumber = new AtomicInteger(1);
@Override public Thread newThread(Runnable runnable) {
Thread thread = new Thread(runnable, "Database-Connection-" + poolNumber.getAndIncrement() + "-thread");
thread.setDaemon(true);
return thread;
}
}
}
@jewelsea
Copy link
Author

Answer to the StackOverflow question: JavaFX - Background Thread for SQL Query.

Task based implementation of Sample for accessing a local database from JavaFX. to demonstrate running the database operations on their own thread rather than on the JavaFX application thread.

@backtozero
Copy link

Thank you. Its work for me too.
Isn't better putting "triggerButton.setDisable(true);" and "databaseActivityIndicator.setVisible(true);" in to onScheduled of task? I am very newby, and very sorry if asking so some stupid thing.

# Lambda version
fetchNamesTask.setOnScheduled((WorkerStateEvent event) -> {
      triggerButton.setDisable(true);
      databaseActivityIndicator.setVisible(true);
});

# Regular one
fetchNamesTask.setOnScheduled(new EventHandler<WorkerStateEvent>() {

            @Override
            public void handle(WorkerStateEvent event) {
                      triggerButton.setDisable(true);
                      databaseActivityIndicator.setVisible(true);
            }
});

@jewelsea
Copy link
Author

You could use additional event handlers backtozero, I just find the code easier to understand and read without them.

I updated the gist for Java 8 and also replaced the event handler logic which sets visibility and disabled states with bindings.

private void fetchNamesFromDatabaseToListView(
        final Button triggerButton, 
        final ProgressIndicator databaseActivityIndicator, 
        final ListView<String> listView) {
  final FetchNamesTask fetchNamesTask = new FetchNamesTask();

  triggerButton.disableProperty().bind(
          fetchNamesTask.runningProperty()
  );
  databaseActivityIndicator.visibleProperty().bind(
          fetchNamesTask.runningProperty()
  );
  databaseActivityIndicator.progressProperty().bind(
          fetchNamesTask.progressProperty()
  );

  fetchNamesTask.setOnSucceeded(t ->
          listView.setItems(fetchNamesTask.getValue())
  );

  databaseExecutor.submit(fetchNamesTask);
}

@kingsley-einstein
Copy link

This is quite cool and genius I might add. I hope to delve into database manipulation with Java in the near future and your piece will serve as a template.

Copy link

ghost commented Jun 4, 2018

how to use it if you seperate main and controller

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment