- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个按钮,它从字段中获取我的用户名和密码,并向后端发送请求以进行身份验证。我正在使用线程,因此我的按钮动画和场景不会像这样卡住:
Service<Void> service = new Service<Void>() {
@Override
protected Task<Void> createTask() {
return new Task<Void>() {
@Override
protected Void call() throws Exception {
//do authentication
if(responseStatus != 200){
authenticated = false
}else{
authenticate = true
}
Platform.runLater(() -> {
try{
if(authenticated) {
changeScene();
}
}finally{
latch.countDown();
}
});
latch.await();
return null;
}
};
}
};
service.start();
private void changeScene(){
try {
Stage window = (Stage)loginPane.getScene().getWindow();
LoggedFirstStyle.displayLoggedScene();
window.close();
} catch (IOException e) {
e.printStackTrace();
}
}
但问题是,如果我多次单击按钮,平台运行稍后会执行多次,changeScene 也是如此,并且会打开多个场景。我这样做的方式好吗?如果可以的话,我怎样才能防止在同一个方法中打开多个线程?
最佳答案
一个Service
提供“重用”Task
的能力。我将“重用”放在引号中,因为真正发生的是 Service
每次启动时都会创建一个新的 Task
。有关 Service
和 Task
之间的差异和用法的更多信息,您可以:
Service
的文档和 Task
-也许Worker
也是如此。由于服务
旨在重复使用,因此您应该只创建一个实例。它还维护状态,因此只能“一次”执行一次。换句话说,同一个 Service
不能并行执行多次。当Service
完成时,它将处于SUCCEEDED
、CANCELLED
或FAILED
状态;要再次启动Service
,您必须调用restart()
(将取消正在运行的Service
)或调用reset()
再次调用 start()
之前。
当 Service
运行时,您需要禁用某些 UI 组件,以便用户无法尝试多次启动它。您可以通过监听器和/或绑定(bind)来完成此操作。如果需要,您还可以进行检查,以便您的代码不会尝试启动 Service
(如果它已在运行)。是否需要这些检查取决于哪些代码可以启动Service
以及如何执行它。
这是一个小例子。它使用 FXML 创建界面,但重要的部分是 LoginController
和 LoginService
类。根据您的应用程序,您可能还想添加一种取消登录的方法。
Main.java
package com.example;
import java.io.IOException;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
public class Main extends Application {
@Override
public void start(Stage primaryStage) throws IOException {
// Login.fxml is in the same package as this class
Parent root = FXMLLoader.load(getClass().getResource("Login.fxml"));
primaryStage.setScene(new Scene(root));
primaryStage.setTitle("Service Example");
primaryStage.show();
}
}
LoginService.java
package com.example;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
import javafx.concurrent.Service;
import javafx.concurrent.Task;
public class LoginService extends Service<Boolean> {
private final StringProperty username = new SimpleStringProperty(this, "username");
public final void setUsername(String username) { this.username.set(username); }
public final String getUsername() { return username.get(); }
public final StringProperty usernameProperty() { return username; }
private final StringProperty password = new SimpleStringProperty(this, "password");
public final void setPassword(String password) { this.password.set(password); }
public final String getPassword() { return password.get(); }
public final StringProperty passwordProperty() { return password; }
@Override
protected Task<Boolean> createTask() {
return new LoginTask(getUsername(), getPassword());
}
private static class LoginTask extends Task<Boolean> {
private final String username;
private final String password;
public LoginTask(String username, String password) {
this.username = username;
this.password = password;
}
@Override
protected Boolean call() throws Exception {
Thread.sleep(3_000L); // simulate long running work...
return !isCancelled() && "root".equals(username) && "root".equals(password);
}
}
}
LoginController.java
package com.example;
import javafx.beans.binding.Bindings;
import javafx.beans.binding.BooleanBinding;
import javafx.concurrent.Worker;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.scene.Cursor;
import javafx.scene.control.Alert;
import javafx.scene.control.Button;
import javafx.scene.control.PasswordField;
import javafx.scene.control.TextField;
import javafx.scene.layout.GridPane;
public class LoginController {
@FXML private GridPane root;
@FXML private TextField userField;
@FXML private PasswordField passField;
@FXML private Button loginBtn;
private LoginService service;
@FXML
private void initialize() {
service = new LoginService();
service.usernameProperty().bind(userField.textProperty());
service.passwordProperty().bind(passField.textProperty());
// Don't let user interact with UI while trying to login
BooleanBinding notReadyBinding = service.stateProperty().isNotEqualTo(Worker.State.READY);
userField.disableProperty().bind(notReadyBinding);
passField.disableProperty().bind(notReadyBinding);
loginBtn.disableProperty().bind(notReadyBinding);
root.cursorProperty().bind(
Bindings.when(service.runningProperty())
.then(Cursor.WAIT)
.otherwise(Cursor.DEFAULT)
);
service.setOnSucceeded(event -> serviceSucceeded());
service.setOnFailed(event -> serviceFailed());
}
private void serviceSucceeded() {
if (service.getValue()) {
/*
* Normally you'd change the UI here to show whatever the user needed to
* sign in to see. However, to allow experimentation with this example
* project we simply show an Alert and call reset() on the LoginService.
*/
showAlert(Alert.AlertType.INFORMATION, "Login Successful", "You've successfully logged in.");
service.reset();
} else {
showAlert(Alert.AlertType.ERROR, "Login Failed", "Your username or password is incorrect.");
service.reset();
}
}
private void serviceFailed() {
showAlert(Alert.AlertType.ERROR, "Login Failed", "Something when wrong while trying to log in.");
service.getException().printStackTrace();
service.reset();
}
private void showAlert(Alert.AlertType type, String header, String content) {
Alert alert = new Alert(type);
alert.initOwner(root.getScene().getWindow());
alert.setHeaderText(header);
alert.setContentText(content);
alert.showAndWait();
}
@FXML
private void handleLogin(ActionEvent event) {
event.consume();
// isBlank() is a String method added in Java 11
boolean blankUsername = userField.textProperty().getValueSafe().isBlank();
boolean blankPassword = passField.textProperty().getValueSafe().isBlank();
if (blankUsername || blankPassword) {
showAlert(Alert.AlertType.ERROR, null, "Both username and password must be specified.");
} else {
service.start();
}
}
}
登录.fxml
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.PasswordField?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.ColumnConstraints?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.layout.RowConstraints?>
<?import javafx.scene.text.Font?>
<GridPane fx:id="root" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" vgap="20.0"
xmlns:fx="http://javafx.com/fxml/1" xmlns="http://javafx.com/javafx/10.0.1"
fx:controller="com.example.LoginController">
<columnConstraints>
<ColumnConstraints hgrow="SOMETIMES" minWidth="-Infinity" percentWidth="50.0"/>
<ColumnConstraints halignment="RIGHT" hgrow="SOMETIMES" minWidth="-Infinity" percentWidth="50.0"/>
</columnConstraints>
<rowConstraints>
<RowConstraints minHeight="-Infinity" percentHeight="25.0" vgrow="SOMETIMES"/>
<RowConstraints minHeight="-Infinity" percentHeight="25.0" vgrow="SOMETIMES"/>
<RowConstraints minHeight="-Infinity" percentHeight="25.0" vgrow="SOMETIMES"/>
<RowConstraints minHeight="-Infinity" percentHeight="25.0" vgrow="SOMETIMES"/>
</rowConstraints>
<children>
<Button fx:id="loginBtn" defaultButton="true" mnemonicParsing="false" onAction="#handleLogin" text="Login"
GridPane.columnIndex="1" GridPane.rowIndex="3"/>
<Label minWidth="-Infinity" text="Welcome">
<font>
<Font name="Segoe UI" size="32.0"/>
</font>
</Label>
<TextField fx:id="userField" prefColumnCount="20" promptText="Username" GridPane.columnSpan="2"
GridPane.rowIndex="1"/>
<PasswordField fx:id="passField" prefColumnCount="20" promptText="Password" GridPane.columnSpan="2"
GridPane.rowIndex="2"/>
</children>
<padding>
<Insets bottom="50.0" left="50.0" right="50.0" top="50.0"/>
</padding>
</GridPane>
关于即使单击按钮多次,Javafx 也仅启动一个线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54832485/
我在选项卡中有我的网络应用程序的选项。 aaa bbb ccc ddd eee 当用户单击任何选项卡(在同一窗口中)时,我会使用此代码获得淡出效果,然后自动
如何获得一个简单的调用单击来实现如下功能:http://jsfiddle.net/ftwPS/6/我显然错过了一些基本的东西,因为当您单击“CLICK”时这不起作用。 $('tr').click(fu
我有一个列表,在左侧显示一个插入符号图标,单击它时应该更改图标。每当我单击一个图标时,所有图标都会更改。 detailedInfo = []; subordinateInfo; openSu
我有一个 div,我想对其进行设置,以便当我单击其他内容时,它会隐藏该 div。 我也是这样 $('body').click(function(){ if(loginOpened) {
我有很多装有不同内容的盒子。 如果单击第一个框,您可以看到一个包含更多详细信息的弹出框。如果单击第二个、第三个等框,您必须查看这些框的详细信息。 我的问题:如果我点击框号。 2 我看到了盒子号的详细信
当我点击 .add-btn 时,我将 .add-btn 的样式更改为 background:#cccccc 并将 val() 更改为“-”。 现在当我用 tr td .list 删除添加的文本时,它是
感谢 SO 的出色贡献者!当您开始了解 jQuery 时,它会变得更酷。 :) 所以我有一个 LI,单击时会显示/隐藏子 UL。我想做的是能够单击 LI 内的链接,打开一个空白窗口,但也不会关闭子 U
我有这份文档,它使我能够获得一份带有点击进度的多项选择选择调查问卷。 如何用图像代替此处的文字? (并且仍然有这样的机制,一旦单击[图像],就会提出下一个问题) 我已经尝试使用 UL/H1 设置背景图
我想在 JQGrid 的刷新按钮单击上编写代码。有什么事件吗? 最佳答案 如果您需要在开始刷新之前执行一些操作,您应该使用 beforeRefresh打回来: $("#grid_id").jqGrid
问题是将对象或多个参数从模板传递到组件,并使用它们将数据添加到 API。 任务.service.ts addTasks(task: Task): Observable{ let headers =
我有一个像这样的primefaces选项卡 View : This tab has static content. this t
我在 jquery 中有一个有效的 a.click() 函数...但是如果我单击一个 anchor ,我会打开一个新窗口...但是我怎样才能阻止浏览器本身打开一个新窗口? 示例: $('a')
有没有简单的方法来创建代码:如果 URL 更改或单击链接显示 div(例如加载 gif 3 秒),则显示页面?有点像空白的白色页面,加载 gif 旋转 3 秒然后显示页面? 谢谢! 最佳答案 给定 G
我需要知道此时按钮的状态是否被点击? 谢谢 最佳答案 if (myButton.state & UIControlStateHighlighted) { // Do your stuff the
我正在 NSImageView 上绘制一条 NSBezierPath 线。我正在创建 NSBezierPath 对象,设置 moveToPoint,设置 lineToPoint,设置 setLineW
我的 Selenium 代码存在问题,无法正确执行按键 + 单击操作。 测试应打开 jqueryui.com 链接并选择页面上的前 2 个 li 元素。 我正在使用 Selenium 2.23 和 F
单击时我将更改字符串一部分的样式。例如“TEXT”,然后单击“T”,之后它会将样式从黑色更改为红色,仅 T在我的代码中,当我单击文本时,我拆分文本并保留在“split”数组中,它将调用handleCl
我在网站上有一个 anchor 。当有人点击它时,我在 jquery 中执行一些操作并更改名称,但是当再次单击它时,事件被触发,尽管我已经更改了它的名称。代码在这里: $(".like_cont a[
我有一个下载链接Download我希望每次当有人点击“下载”时,我都可以将其插入数据库total_downloads+1。为了插入数据库,我通常使用 然后 if (isset($_POST['d
我是一名优秀的程序员,十分优秀!