- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我需要使用 UI 组件更新模型类的数据,同时使用数据对象的更改更新 UI 组件。详细地说,大量数据依赖于其他数据。 e.a.:A和B的SUM。SUM需要显示在UI上并存储在模型类中。
在实际情况中,我有大约 58 个可编辑字段,其中包含文本和数字。计算字段数量减半。
经过思考,我想出了很多解决方案。我的问题是,我没有经验来决定或判断什么是最好的方法(如果有的话)。两个主要候选人是:
处理这种情况的最佳或好的方法是什么?
我用来测试的代码:
public class Comunication {
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
public void run() {
//Main Window
JFrame frame = new JFrame();
frame.setTitle("SumTest");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setMinimumSize(new Dimension(500,200));
frame.setVisible(true);
//Main Panel
JPanel pane = new JPanel();
frame.setContentPane(pane);
//Component
JTextField valueA = new JTextField("VALUE A");
JTextField valueB = new JTextField("VALUE B");
JTextField valueSum = new JTextField("VALUE SUM");
pane.add(valueA);
pane.add(valueB);
pane.add(valueSum);
}
});
}
}
class Data {
private int a;
private int b;
private int sum;
public Data() {
a = 1;
b = 2;
Calculate();
}
public void Calculate() {
sum = a + b;
}
public int getA() { return a; }
public int getB() { return b; }
public int getSUM() { return sum; }
public void setA(int i) { a = i; }
public void setB(int i) { b = i; }
}
<小时/>
第 2 部分:
通过用户提供的信息进行实验,我可以自由地尝试其他东西。一种解决方案是创建一个链接 View 和模型的监听器类。每次将监听器添加到字段( View )时,都应该稍微更改一下,这是在不使用反射的情况下将字段与模型中的方法链接起来的唯一方法。
因此,更新的算法是:当更改时, View 更新模型。之后:全局 Controller 使用模型上的新信息更新所有 View 。
我发现的问题,可能是由于缺乏经验:
1) 由于所有 View 都有文档更改监听器:当全局 Controller 更新所有 View /字段时,它们会再次调用监听器。我发现的一个解决方法是向监听器添加一个“silent”标志。
2) 当 View 更新模型并调用全局 Controller 更新所有其他 View 时,它无法更新自身。我还没找到原因,我认为可能是造成循环的原因。解决方法是告诉全局 Controller 谁调用了它,并更新除调用者之外的每个 View 。
包含解决方案和解决方法的完整工作代码如下。非常欢迎提出意见,我想做得正确,或者更好。
import java.awt.*;
import java.util.ArrayList;
import javax.swing.*;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
public class Comunication {
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
//Main Window
JFrame frame = new JFrame();
frame.setTitle("SumTest");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setMinimumSize(new Dimension(500,200));
frame.setVisible(true);
//Main Panel
JPanel pane = new JPanel();
frame.setContentPane(pane);
//Data Model
DataModel model = new DataModel();
GlobalUpdateController viewUpdateController = new GlobalUpdateController();
//Component
JTextField valueA = new JTextField("");
JTextField valueB = new JTextField("");
JTextField valueSum = new JTextField("");
valueA.setPreferredSize(new Dimension(30, 20));
valueB.setPreferredSize(new Dimension(30, 20));
valueSum.setPreferredSize(new Dimension(30, 20));
pane.add(valueA);
pane.add(valueB);
pane.add(valueSum);
//Listeners
valueA.getDocument().addDocumentListener(new StealthListener(valueA , viewUpdateController) {
@Override
public void updateView() {
this.view.setText( Integer.toString( model.getA() ) );
}
@Override
public void updateModel() {
model.setA( Integer.parseInt( this.view.getText() ) );
}
});
valueB.getDocument().addDocumentListener(new StealthListener(valueB , viewUpdateController) {
@Override
public void updateView() {
this.view.setText( Integer.toString( model.getB() ) );
}
@Override
public void updateModel() {
model.setB( Integer.parseInt( this.view.getText() ) );
}
});
valueSum.getDocument().addDocumentListener(new StealthListener(valueSum , viewUpdateController) {
@Override
public void updateView() {
this.view.setText( Integer.toString( model.getSUM() ) );
}
@Override
public void updateModel() {
//Do nothing
}
});
//Initial Update
viewUpdateController.updateAllViews(null);
}
});
}
}
class DataModel {
private int a;
private int b;
private int sum;
public DataModel() {
a = 3;
b = 5;
Calculate();
}
public void Calculate() {
sum = a + b;
}
public int getA() { return a; }
public int getB() { return b; }
public int getSUM() { return sum; }
public void setA(int i) { a = i; Calculate(); }
public void setB(int i) { b = i; Calculate(); }
}
class StealthListener implements DocumentListener {
JTextField view;
GlobalUpdateController viewList;
private boolean silent;
public StealthListener(JTextField view, GlobalUpdateController viewList) {
this.view = view;
this.viewList = viewList;
this.silent = false;
this.viewList.add(this);
}
public void setSilent(boolean val) {
this.silent = val;
}
public void updateView() {
// Unique to each view, to be Overriden
}
public void updateModel() {
// Unique to each view, to be Overriden
}
public void update() {
//The silent flag is meant to avoid ListenerLoop when changing the document.
//When the silent is true is meant to listen to internal changes.
if(this.silent == false) {
updateModel();
this.viewList.updateAllViews(this);
}
}
@Override
public void insertUpdate(DocumentEvent e) {
update();
}
@Override
public void removeUpdate(DocumentEvent e) {
update();
}
@Override
public void changedUpdate(DocumentEvent e) {
update();
}
}
class GlobalUpdateController {
private ArrayList<StealthListener> viewList;
public GlobalUpdateController() {
this.viewList = new ArrayList<StealthListener>();
}
public void add(StealthListener control) {
this.viewList.add(control);
}
public void updateAllViews(StealthListener caller) {
for( StealthListener view : viewList) {
if( caller==null || view != caller ) {
view.setSilent(true);
view.updateView();
view.setSilent(false);
}
}
}
}
最佳答案
在此相关example每个任意数量的可编辑文本字段都会添加一个 PropertyChangeListener
的实例和一个 FocusListener
。每个监听器都会调用一个公共(public) update()
重新计算派生的方法 sum
。在下面的变体中,字段共享 UpdateListener
的单个实例。即两者都是FocusListener
和 PropertyChangeListener
。让UpdateListener
还实现DocumentListener
仅当您需要 update()
每次击键。
实际上,模型是 List<JFormattedTextField>
中的数据, Controller 是常见的 update()
强制输入字段之间关系的方法。这种方法的可扩展性取决于 update()
所产生的复杂性。 。该示例使用 JFormattedTextField
方便相关字段的格式化。
每当我考虑使用反射时,我也会考虑 strategy pattern与 enum
结合使用作为备选。例子见here和 here .
import java.awt.EventQueue;
import java.awt.GridLayout;
import java.awt.event.FocusAdapter;
import java.awt.event.FocusEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.text.NumberFormat;
import java.util.ArrayList;
import java.util.List;
import javax.swing.JFormattedTextField;
import javax.swing.JFrame;
import javax.swing.JPanel;
/**
* @see https://stackoverflow.com/a/31764798/230513
* @see https://stackoverflow.com/q/8703464/230513
* @see https://stackoverflow.com/questions/6803976
*/
public class Adder extends JPanel {
private static final int MAX = 3;
private final List<JFormattedTextField> fields = new ArrayList<>();
private final NumberFormat format = NumberFormat.getNumberInstance();
private final JFormattedTextField sum = new JFormattedTextField(format);
private final UpdateListener listener = new UpdateListener();
private class UpdateListener extends FocusAdapter implements PropertyChangeListener {
@Override
public void propertyChange(PropertyChangeEvent e) {
update();
}
@Override
public void focusLost(FocusEvent e) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
update();
}
});
}
}
public Adder() {
this.setLayout(new GridLayout(0, 1));
for (int i = 0; i < MAX; i++) {
JFormattedTextField tf = init();
fields.add(tf);
this.add(tf);
}
sum.setHorizontalAlignment(JFormattedTextField.RIGHT);
sum.setEditable(false);
sum.setFocusable(false);
this.add(sum);
}
private JFormattedTextField init() {
JFormattedTextField jtf = new JFormattedTextField(format);
jtf.setValue(0);
jtf.setHorizontalAlignment(JFormattedTextField.RIGHT);
jtf.addFocusListener(listener);
jtf.addPropertyChangeListener("value", listener);
return jtf;
}
private void update() {
long total = 0;
for (JFormattedTextField tf : fields) {
Number v = (Number) tf.getValue();
total += v.longValue();
}
sum.setValue(total);
}
private void display() {
JFrame f = new JFrame("Adder");
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.add(this);
f.pack();
f.setLocationRelativeTo(null);
f.setVisible(true);
}
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
new Adder().display();
}
});
}
}
关于java - 在 JComponent 和模型对象之间传输数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31763734/
这个问题已经有答案了: Mirroring an Object in JFrame with paintComponent (1 个回答) 已关闭 6 年前。 我已经知道 JComponent 无法添
我在绘制 jcomponent 时遇到问题 //应在其中绘制矩形的类 public class Board extends JComponent { private Case[][] case
在 Swing 中,鼠标事件会自动定位到具有鼠标监听器的最深组件。当有一个组件具有鼠标监听器并添加到其他组件之上时,如何将它们再次定位到更深的组件? 应该在不将组件设置为不可见或删除其鼠标监听器的情况
最近我在使用 Swing 时遇到了一些问题。我正在尝试快速制作下图所示的内容,以说明算法的数据结构。 (来源:ius.edu) 我在接下来的类(class)中尝试做的就是画出一些带有数字的矩形。并翻译
我已经阅读了 oracle 提供的 Java trail,他们说对象属性设置为与操作属性相匹配。我想知道这是否是一组有限的共享属性,或者是否将 Action 强制转换为与您将操作设置为的对象相同的类。
我有一个 JCombobox,我在运行时添加一些项目。其中一些非常长,我的 Jcombobox 也长得很长。它位于带有 BoxLayout (PAGE_AXIS) 的 JPanel 容器中。我不知道如
我正在处理一个项目,该项目要求我能够在同一容器内的组件顶部绘制半透明的 JPanel。使用 setComponentZOrder() 方法对于重叠(即下方的组件未完全覆盖)效果很好,但当下方的组件完全
目前,我必须将 JComponent 包含在具有垂直框布局的 JPanel 中。这样,我可以将第一个组件居中,如下所示,并将底部组件(相当长)放在下面。但是,由于底部组件非常长,我想仅为该特定组件添加
我有一个 JComponent 子类,我用它在屏幕上绘制形状。在构造函数中,我尝试将 ballX 和 ballY 设置为 X 和 Y 大小值的一半JComponent,我想我做错了。我现在已经查了很多
我想完成一些与带有选择器线的矩形图像非常相似的事情。 基本上,我有一个矩形,我想在它周围有一条选择器线。为此,我想创建一个额外的 JComponent。 目前我只能绘制矩形。如何在 Rectangle
到目前为止,我一直在使用 VB 来开发应用程序。现在我必须使用Java来开发前端。我对组件很困惑。需要帮助。书籍引用或网站引用也可以完成这项工作。 基本上,我将使用菜单栏、工具栏、带有 JEditor
我有一个自定义 JComponent,它在调用 paint 时绘制一些内容。但是,在 Border 布局中,它的 minimumSize 并未得到尊重。我已经包含了这个 @Override publi
我是一名初学者程序员,我正在尝试让这个程序运行。尽管 JComponent 的新 Java 屏幕上没有出现任何图像,但一切都可以正确编译。该程序要做的几乎就是获取一个输入值并将其分配给条形图的大小值。
我有以下代码: package gui; import java.awt.Color; import java.awt.Dimension; import java.awt.Graphics; imp
如何获取JComponent名称(JComboBox,JCheckBox,JTextComponent...等) 我如何获取 fieldName 并将此名称放入 HashMap,例如:
我在面板上有一个 JTable 和一个 JLabel。我将其设置为背景图像,但无法摆脱表格和标签中的灰色背景。 我该怎么做? 最佳答案 尝试在 JTable 和 JLabel 上执行 .setOpaq
我正在尝试将组件添加到我的 JPanel,但是它们的大小不正确且位置不正确。这是我的组件代码。 button = new JButton(); button.setSize(100, 100); bu
好的。我正在编写我的 Ball 类,但球没有显示。我尝试将其他组件添加到我的容器中并且它们被显示,所以我认为可以安全地假设问题出在我的球上。类代码: import java.awt.Color; im
作为项目的一部分,我有一个自定义 JComponent,它实现了一部分用户界面。这是一个相当奇怪的控制,涉及围绕一组 x-y 点移动,但这并不重要,因为它已经实现并且工作正常。 该组件有一个正方形的可
嗨,我是java初学者,这里我有我的程序,但是,当我尝试调整面板大小时,我的文本就消失了?此外,如何在文本下绘制一条粗绿线,即使在调整大小时它也会保留在文本下方,我很无能? import javax.
我是一名优秀的程序员,十分优秀!