私は経験豊富なプログラマーですが、最近まで Java を使用したことがありませんでした。インターネットからコードをコピーすることで機能するクラス(huzzah)があり、main()関数を使用するときに以下のコードを独自のスレッドとして実行すると、Jframeは独自のスレッドで実行されます。
しかし、私には問題があります。この Jframe を別のスレッドのインタラクティブな部分として (implements Runnable クラスのプライベート メンバーとして) 使用したいと考えています。そして、インタラクティブにしたい。Jframe メソッド actionPerformed は親スレッド オブジェクトの関数を呼び出す必要があり (コマンドを入力すると、何かを実行します)、親は .append() メソッドを呼び出して結果を Jframe に報告する必要があります。Jframe 内に親への参照を含めることができるので、そのメソッドを呼び出すことができます。
したがって、私の問題は、スレッド同期のためにこれを適切に設計することです。私のスレッドはスレッドセーフではない Jframe を適切に使用できるため、スレッド同期とイベントハンドラーがどのように相互作用するのかわかりません。
Jframeを適切に初期化する場所と方法は? (具体的には、何らかの方法で EventQueue を使用することになっているのでしょうか?) それとも、目立ったものを見逃していて、これを完全に間違った方法で行っているのでしょうか?
import javax.swing.*;
import javax.swing.border.EmptyBorder;
import javax.swing.plaf.ActionMapUIResource;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.KeyEvent;
public class CmdPrompt3 extends JFrame {
private static final String progname = "CmdPrompt <Monitor Game Client>";
private JTextField input;
private JTextArea history;
private JPanel mainpanel;
private JMenuBar greenMenuBar;
private String lastCmd;
private boolean unusedCmd;
//public boolean defaultConstruct = true;
public CmdPrompt3(int x, int y, String name) { //offset coordinates
super("CmdPrompt3 <" + name + ">");
lastCmd = "";
unusedCmd = false;
//defaultConstruct = false;
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
//Create the menu bar. Make it have a green background.
greenMenuBar = new JMenuBar();
greenMenuBar.setOpaque(true);
greenMenuBar.setBackground(new Color(154, 165, 127));
greenMenuBar.setPreferredSize(new Dimension(200, 20));
this.setJMenuBar(greenMenuBar);
mainpanel = new JPanel(new BorderLayout());
mainpanel.setBorder(new EmptyBorder(2, 3, 2, 3));
// adjust numbers for a bigger default area
JTextArea history = new JTextArea(25, 40); //25 lines, 40 columns
Font font = new Font(
Font.MONOSPACED,
Font.PLAIN,
16); //font size 16
//history.getFont().getSize()+4);
mainpanel.setFont(font);
history.setFont(font);
mainpanel.add(new JScrollPane(history,
JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS));
this.add(mainpanel);
input = new JTextField(80);
input.setFont(font);
this.add(input, BorderLayout.SOUTH);
ActionMap actionMap = new ActionMapUIResource();
actionMap.put("enter", new AbstractAction() {
@Override
public void actionPerformed(ActionEvent e) {
String cmd = input.getText();
//String sofar = history.getText ();
history.append("\tinput box>>\t" + cmd + "\n");
//DO IMPORTANT STUFF HERE
//Set flags for retrieving commands
lastCmd = cmd;
unusedCmd = true;
//clear input line
input.setText("");
}
});
InputMap keyMap = new ComponentInputMap(input);
keyMap.put(KeyStroke.getKeyStroke(KeyEvent.VK_ENTER, 0), "enter");
SwingUtilities.replaceUIActionMap(input, actionMap);
SwingUtilities.replaceUIInputMap(input, JComponent.WHEN_IN_FOCUSED_WINDOW, keyMap);
setLocation(x, y);
this.pack();
setVisible(true);
}
//owner thread writes history to window
//TODO this doesn't work when Client calls the function
//Apparently,
public void append(String S) {
history.append(S);
return;
}
//owner thread gets last issued command, if any
public String getUnusedCmd() {
if (unusedCmd) {
unusedCmd = false;
} else {
lastCmd = "";
}
return lastCmd;
}
public static void main(final String args[]) {
Runnable runner = new Runnable() {
public void run() {
new CmdPrompt3(100, 100, "Basic Class Test");
}
};
EventQueue.invokeLater(runner);
}
}