【问题标题】:JAVA KeyEvent serialization errorJAVA KeyEvent 序列化错误
【发布时间】:2018-06-17 13:46:28
【问题描述】:

我试图通过套接字对象输出流发送一个包含KeyEvent 字段的对象,javadoc 说KeyEvent 正在实现Serializable 接口,但是每当我尝试发送它时都会抛出一个@987654325 @。

我要发送的对象是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

【问题讨论】:

  • @HovercraftFullOfEels 好主意,添加了导入和错误消息
  • 谢谢,但请以代码格式的文本发布错误,并指出涉及哪些行(堆栈跟踪会告诉您)
  • 请贴出你用来序列化的代码。
  • 确实——错误来自 ControlFrame 类,我什至没有看到你发布这个类的代码。
  • @POrekhov 我没有明确实现Serializable,我使用默认方法,因为该类包含的所有对象都是可序列化的。

标签: java oop serialization client-server keyevent


【解决方案1】:

你是对的,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;
    }
}

【讨论】:

  • 感谢您的帮助 :) 我是一名学生,这是我的业余项目,我对这个恼人的错误感到非常沮丧,但现在我可以继续了,顺便说一句,该项目是关于远程控制计算机的
  • @OE.omergunr100:感谢您尝试改进问题
  • 顺便说一句,我也喜欢 Autoit
  • @OE.omergunr100:是的,它是一个非常方便且强大的实用程序,但我希望它是面向对象的
  • 也许吧,但到目前为止,它在自动执行重复性任务方面仍然非常有用,如果您使用颜色或图像匹配,它甚至可以根据屏幕上的变化做出小的决定
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2014-07-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2017-08-24
  • 2014-02-19
  • 1970-01-01
相关资源
最近更新 更多