JAVA KeyEvent序列化错误
JAVA KeyEvent serialization error
我试图通过套接字对象输出流发送一个包含 KeyEvent
字段的对象,javadoc 说 KeyEvent
正在实现 Serializable
接口,但是每当我尝试发送它抛出 java.io.NotSerializableException
.
我要发送的对象是KeyboardCommand
import java.awt.Robot;
public interface ICommandAction {
void doAction(Robot operator);
public enum ActionType {GenericAction, MouseAction, KeyboardAction, CheckBooleanAction};
ActionType getAction();
}
import java.awt.Robot;
import java.io.Serializable;
public class CommandBase implements Serializable, ICommandAction{
private static final long serialVersionUID = 2L;
private ActionType action;
public CommandBase(ActionType action){
this.action = action;
}
public ActionType getAction(){
return this.action;
}
public void doAction(Robot operator){
try {
operator.wait(20);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
import java.awt.Robot;
import java.awt.event.KeyEvent;
public class KeyboardCommand extends CommandBase {
private static final long serialVersionUID = 8L;
private KeyEvent event;
public KeyboardCommand(KeyEvent event) {
super(ActionType.KeyboardAction);
this.event = event;
}
/**
* Presses and releases the event's key
*/
public void key(Robot operator, int delay){
operator.keyPress(event.getExtendedKeyCode());
operator.delay(delay);
operator.keyRelease(event.getExtendedKeyCode());
}
/**
* Performs the keyboard action
*/
@Override
public void doAction(Robot operator){
boolean isUpperCase = Character.isUpperCase(event.getKeyChar());
if(isUpperCase) operator.keyPress(KeyEvent.SHIFT_DOWN_MASK);
key(operator, 20);
if(isUpperCase) operator.keyRelease(KeyEvent.SHIFT_DOWN_MASK);
System.out.println("Keyboard command executed");
}
@Override
public String toString(){
return "KeyCode: " + this.event.getKeyCode() + ", keyChar: " + this.event.getKeyChar();
}
}
public synchronized void sendCommand(CommandBase command){
try {
commandOutputStream.writeObject(command); //error is here
commandOutputStream.flush();
System.out.println("Sent " + command.getAction().toString());
} catch (IOException e) {
System.out.println("Failed to send " + command.getAction().toString());
try {
commandOutputStream.close();
running = false;
this.dispose();
System.out.println("Closing command output stream");
} catch (IOException e1) {
e1.printStackTrace();
}
e.printStackTrace();
}
}
Error code
错误在 commandOutputStream.writeObject(command)
行,这意味着它无法在发件人端序列化。
据我所知,一切都应该是 Serializable
,如果有人知道我在哪里搞砸了,我们将不胜感激:)
微型例子:(与KeyboardCommand
相同)
public class Constants {
final static int senderPort = 20000;
final static int receiverPort = 20001;
}
import java.awt.BorderLayout;
import java.awt.Toolkit;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.io.IOException;
import java.io.ObjectOutputStream;
import java.net.Socket;
import java.net.SocketAddress;
import javax.swing.JFrame;
import commands.CommandBase;
import commands.KeyboardCommand;
public class ControlFrame extends JFrame implements KeyListener, Runnable{
private static final long serialVersionUID = 12L;
private Socket sendAction;
private ObjectOutputStream commandOutputStream;
public boolean running;
public ControlFrame(Socket socket){
super("Controller");
this.running = true;
this.sendAction = socket;
setSize(Toolkit.getDefaultToolkit().getScreenSize().width / 2, Toolkit.getDefaultToolkit().getScreenSize().height / 2);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setLayout(new BorderLayout());
setVisible(true);
try {
this.commandOutputStream = new ObjectOutputStream(sendAction.getOutputStream());
System.out.println("Command output stream initialized");
} catch (IOException e) {
System.out.println("Command output stream failed to initialize");
e.printStackTrace();
}
Thread t = new Thread(this);
t.start();
}
public synchronized void sendCommand(CommandBase command){
try {
commandOutputStream.writeObject(command);
commandOutputStream.flush();
System.out.println("Sent " + command.getAction().toString());
} catch (IOException e) {
System.out.println("Failed to send " + command.getAction().toString());
try {
commandOutputStream.close();
running = false;
this.dispose();
System.out.println("Closing command output stream");
} catch (IOException e1) {
e1.printStackTrace();
}
e.printStackTrace();
}
}
@Override
public void keyTyped(KeyEvent e) {
System.out.println("Key typed");
}
@Override
public void keyPressed(KeyEvent e) {
System.out.println("Key pressed");
System.out.println("Key command: " + e.toString());
sendCommand(new KeyboardCommand(e));
}
@Override
public void keyReleased(KeyEvent e) {
System.out.println("Key released");
}
@Override
public void run() {
addKeyListener(this);
while(running){
if(sendAction.isOutputShutdown())
running = false;
}
}
}
import java.awt.AWTException;
import java.awt.Robot;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.net.ServerSocket;
import java.net.Socket;
import commands.CommandBase;
import commands.ICommandAction.ActionType;
import commands.KeyboardCommand;
public class Receiver {
public static void main(String[] args) throws IOException, ClassNotFoundException, AWTException {
ServerSocket acceptConnection = new ServerSocket(Constants.receiverPort);
Socket receiver = acceptConnection.accept();
ObjectInputStream receiverInput = new ObjectInputStream(receiver.getInputStream());
Robot operator = new Robot();
while(true){
Object command = receiverInput.readObject();
if(((CommandBase)command).getAction().equals(ActionType.KeyboardAction)){
((KeyboardCommand)command).doAction(operator);
}
}
}
}
import java.io.IOException;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.Socket;
import FrameResources.ControlFrame;
public class Sender {
public static void main(String[] args) throws IOException {
final InetAddress thisIP = InetAddress.getLocalHost();
Socket sender = new Socket();
sender.bind(new InetSocketAddress(thisIP, Constants.senderPort));
sender.connect(new InetSocketAddress(thisIP, Constants.receiverPort));
ControlFrame frame = new ControlFrame(sender);
}
}
在这个例子中,同样的错误发生了。
如果有任何不清楚的地方,请评论问题。
感谢您的宝贵时间:D
你是对的,KeyEvent 是原因,可能是因为它包含对源对象的引用,你的 JFrame,一个可能包含可序列化和不可序列化字段的复杂对象......但你不确实不需要这些信息,因此想到了一个明显的解决方案——不要序列化 KeyEvent,而是序列化和反序列化 KeyEvent 包含的密钥信息,即 keyCode、keyChar 和 extendedKeyCode 字段。例如,注释更改:
import java.awt.Robot;
import java.awt.event.KeyEvent;
import java.io.Serializable;
public class KeyboardCommand extends CommandBase {
private static final long serialVersionUID = 8L;
// private KeyEvent event;
private char keyChar;
private int keyCode;
private int extendedKeyCode;
public KeyboardCommand(KeyEvent event) {
super(ActionType.KeyboardAction);
// this.event = event;
keyChar = event.getKeyChar();
keyCode = event.getKeyCode();
extendedKeyCode = event.getExtendedKeyCode();
}
/**
* Presses and releases the event's key
*/
public void key(Robot operator, int delay) {
// operator.keyPress(event.getExtendedKeyCode());
operator.keyPress(extendedKeyCode);
operator.delay(delay);
// operator.keyRelease(event.getExtendedKeyCode());
operator.keyRelease(extendedKeyCode);
}
/**
* Performs the keyboard action
*/
@Override
public void doAction(Robot operator) {
// boolean isUpperCase = Character.isUpperCase(event.getKeyChar());
boolean isUpperCase = Character.isUpperCase(keyChar);
if (isUpperCase)
operator.keyPress(KeyEvent.SHIFT_DOWN_MASK);
key(operator, 20);
if (isUpperCase)
operator.keyRelease(KeyEvent.SHIFT_DOWN_MASK);
System.out.println("Keyboard command executed");
}
@Override
public String toString() {
// return "KeyCode: " + this.event.getKeyCode() + ", keyChar: " + this.event.getKeyChar();
return "KeyCode: " + keyCode + ", keyChar: " + keyChar;
}
}
我试图通过套接字对象输出流发送一个包含 KeyEvent
字段的对象,javadoc 说 KeyEvent
正在实现 Serializable
接口,但是每当我尝试发送它抛出 java.io.NotSerializableException
.
我要发送的对象是KeyboardCommand
import java.awt.Robot;
public interface ICommandAction {
void doAction(Robot operator);
public enum ActionType {GenericAction, MouseAction, KeyboardAction, CheckBooleanAction};
ActionType getAction();
}
import java.awt.Robot;
import java.io.Serializable;
public class CommandBase implements Serializable, ICommandAction{
private static final long serialVersionUID = 2L;
private ActionType action;
public CommandBase(ActionType action){
this.action = action;
}
public ActionType getAction(){
return this.action;
}
public void doAction(Robot operator){
try {
operator.wait(20);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
import java.awt.Robot;
import java.awt.event.KeyEvent;
public class KeyboardCommand extends CommandBase {
private static final long serialVersionUID = 8L;
private KeyEvent event;
public KeyboardCommand(KeyEvent event) {
super(ActionType.KeyboardAction);
this.event = event;
}
/**
* Presses and releases the event's key
*/
public void key(Robot operator, int delay){
operator.keyPress(event.getExtendedKeyCode());
operator.delay(delay);
operator.keyRelease(event.getExtendedKeyCode());
}
/**
* Performs the keyboard action
*/
@Override
public void doAction(Robot operator){
boolean isUpperCase = Character.isUpperCase(event.getKeyChar());
if(isUpperCase) operator.keyPress(KeyEvent.SHIFT_DOWN_MASK);
key(operator, 20);
if(isUpperCase) operator.keyRelease(KeyEvent.SHIFT_DOWN_MASK);
System.out.println("Keyboard command executed");
}
@Override
public String toString(){
return "KeyCode: " + this.event.getKeyCode() + ", keyChar: " + this.event.getKeyChar();
}
}
public synchronized void sendCommand(CommandBase command){
try {
commandOutputStream.writeObject(command); //error is here
commandOutputStream.flush();
System.out.println("Sent " + command.getAction().toString());
} catch (IOException e) {
System.out.println("Failed to send " + command.getAction().toString());
try {
commandOutputStream.close();
running = false;
this.dispose();
System.out.println("Closing command output stream");
} catch (IOException e1) {
e1.printStackTrace();
}
e.printStackTrace();
}
}
Error code
错误在 commandOutputStream.writeObject(command)
行,这意味着它无法在发件人端序列化。
据我所知,一切都应该是 Serializable
,如果有人知道我在哪里搞砸了,我们将不胜感激:)
微型例子:(与KeyboardCommand
相同)
public class Constants {
final static int senderPort = 20000;
final static int receiverPort = 20001;
}
import java.awt.BorderLayout;
import java.awt.Toolkit;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.io.IOException;
import java.io.ObjectOutputStream;
import java.net.Socket;
import java.net.SocketAddress;
import javax.swing.JFrame;
import commands.CommandBase;
import commands.KeyboardCommand;
public class ControlFrame extends JFrame implements KeyListener, Runnable{
private static final long serialVersionUID = 12L;
private Socket sendAction;
private ObjectOutputStream commandOutputStream;
public boolean running;
public ControlFrame(Socket socket){
super("Controller");
this.running = true;
this.sendAction = socket;
setSize(Toolkit.getDefaultToolkit().getScreenSize().width / 2, Toolkit.getDefaultToolkit().getScreenSize().height / 2);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setLayout(new BorderLayout());
setVisible(true);
try {
this.commandOutputStream = new ObjectOutputStream(sendAction.getOutputStream());
System.out.println("Command output stream initialized");
} catch (IOException e) {
System.out.println("Command output stream failed to initialize");
e.printStackTrace();
}
Thread t = new Thread(this);
t.start();
}
public synchronized void sendCommand(CommandBase command){
try {
commandOutputStream.writeObject(command);
commandOutputStream.flush();
System.out.println("Sent " + command.getAction().toString());
} catch (IOException e) {
System.out.println("Failed to send " + command.getAction().toString());
try {
commandOutputStream.close();
running = false;
this.dispose();
System.out.println("Closing command output stream");
} catch (IOException e1) {
e1.printStackTrace();
}
e.printStackTrace();
}
}
@Override
public void keyTyped(KeyEvent e) {
System.out.println("Key typed");
}
@Override
public void keyPressed(KeyEvent e) {
System.out.println("Key pressed");
System.out.println("Key command: " + e.toString());
sendCommand(new KeyboardCommand(e));
}
@Override
public void keyReleased(KeyEvent e) {
System.out.println("Key released");
}
@Override
public void run() {
addKeyListener(this);
while(running){
if(sendAction.isOutputShutdown())
running = false;
}
}
}
import java.awt.AWTException;
import java.awt.Robot;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.net.ServerSocket;
import java.net.Socket;
import commands.CommandBase;
import commands.ICommandAction.ActionType;
import commands.KeyboardCommand;
public class Receiver {
public static void main(String[] args) throws IOException, ClassNotFoundException, AWTException {
ServerSocket acceptConnection = new ServerSocket(Constants.receiverPort);
Socket receiver = acceptConnection.accept();
ObjectInputStream receiverInput = new ObjectInputStream(receiver.getInputStream());
Robot operator = new Robot();
while(true){
Object command = receiverInput.readObject();
if(((CommandBase)command).getAction().equals(ActionType.KeyboardAction)){
((KeyboardCommand)command).doAction(operator);
}
}
}
}
import java.io.IOException;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.Socket;
import FrameResources.ControlFrame;
public class Sender {
public static void main(String[] args) throws IOException {
final InetAddress thisIP = InetAddress.getLocalHost();
Socket sender = new Socket();
sender.bind(new InetSocketAddress(thisIP, Constants.senderPort));
sender.connect(new InetSocketAddress(thisIP, Constants.receiverPort));
ControlFrame frame = new ControlFrame(sender);
}
}
在这个例子中,同样的错误发生了。
如果有任何不清楚的地方,请评论问题。
感谢您的宝贵时间:D
你是对的,KeyEvent 是原因,可能是因为它包含对源对象的引用,你的 JFrame,一个可能包含可序列化和不可序列化字段的复杂对象......但你不确实不需要这些信息,因此想到了一个明显的解决方案——不要序列化 KeyEvent,而是序列化和反序列化 KeyEvent 包含的密钥信息,即 keyCode、keyChar 和 extendedKeyCode 字段。例如,注释更改:
import java.awt.Robot;
import java.awt.event.KeyEvent;
import java.io.Serializable;
public class KeyboardCommand extends CommandBase {
private static final long serialVersionUID = 8L;
// private KeyEvent event;
private char keyChar;
private int keyCode;
private int extendedKeyCode;
public KeyboardCommand(KeyEvent event) {
super(ActionType.KeyboardAction);
// this.event = event;
keyChar = event.getKeyChar();
keyCode = event.getKeyCode();
extendedKeyCode = event.getExtendedKeyCode();
}
/**
* Presses and releases the event's key
*/
public void key(Robot operator, int delay) {
// operator.keyPress(event.getExtendedKeyCode());
operator.keyPress(extendedKeyCode);
operator.delay(delay);
// operator.keyRelease(event.getExtendedKeyCode());
operator.keyRelease(extendedKeyCode);
}
/**
* Performs the keyboard action
*/
@Override
public void doAction(Robot operator) {
// boolean isUpperCase = Character.isUpperCase(event.getKeyChar());
boolean isUpperCase = Character.isUpperCase(keyChar);
if (isUpperCase)
operator.keyPress(KeyEvent.SHIFT_DOWN_MASK);
key(operator, 20);
if (isUpperCase)
operator.keyRelease(KeyEvent.SHIFT_DOWN_MASK);
System.out.println("Keyboard command executed");
}
@Override
public String toString() {
// return "KeyCode: " + this.event.getKeyCode() + ", keyChar: " + this.event.getKeyChar();
return "KeyCode: " + keyCode + ", keyChar: " + keyChar;
}
}