分享
 
 
 

用Java制作十六进制编辑器

王朝java/jsp·作者佚名  2008-05-31
窄屏简体版  字體: |||超大  

本人在用java编制一个网络应用测试工具的时候,迫切希望能以十六进制方式显示和编辑socket上传输的数据,于是自己动手写了一个编辑器类,实现基本的十六进制编辑功能。效果如图一、图二所示,可以看到,编

辑器可以支持常规方式和十六进制两种方式对数据进行编辑。

分析:对于同一段数据值,用两种方式来显示和编辑,则用MVC(模型-视图-控制器)模式来作为主结构是再合适不过的了。模型的作用是保存真实的数据值,同时提供若干提取和修改数据的方法;视图是数据在用户界面上的表示,控制器定义用户界面对用户输入的响应方式,即把用户的键盘动作和鼠标动作解释成模型中的数据操作方法。出于简化的考虑,本例中把视图和控制器合并到了一起。

模型的设计:编辑器必须能处理任意字节块,所以考虑模型内用字节数组来存储数据;要提供在指定偏移处增加、修改和删除字节块的操作;当模型内的数据被改变时要及时通知视图来刷新用户界面或其它感爱好的对象;

图一

图二

视图和控制器的设计:

对于常规编辑的视图,只需把模型中的字节数组转化成String,使用一个文本区域组件JTextArea来显示即可。JTextArea本身也是一个遵循MVC模式的Swing组件,它的控制器即可被用来作为我们自己的控制器,监听它的文本增加、修改和删除事件,从而控制我们自己的数据模型;

对于16进制编辑的视图,同样可以用JTextArea来显示,只是在显示之前,要对模型中的数据进行若干加工,如每行显示16个字节,每行都要加表示偏移量的头,行尾要加这一行数据的字符串表示形式。它的控制器则不能简单的利用JTextArea的控制器了,为了保证显示格式不被打乱,需要监视它的所有光标移动事件、键盘击键事件等。同时,为了保持与UltraEdit的十六进制编辑器的功能一致性,对它的数据的增加、删除功能提供两个按钮,询问用户操作的字节数,假如增加n个字节,则在输入光标处插入n个十六进制值为20的字符(字符空格)。

十六进制编辑器的主要结构请参见下图,由于篇幅关系,图中只列出了十六进制编辑部分,常规编辑部分请读者自行设计:

图三

下面分别就几个主要的方法的功能和主要流程加以说明。

HeXPane.displayValue方法,它主要完成数据的显示工作:

public void diplayValue() {

canvas.setText("");

byte[] data = model.getData();

int dataLen = data.length;

// 把字节数组按每16个字节为一块进行分块;

int lines = dataLen / 16 + 1;

int tails = dataLen % 16;

int offset = 0;

for(int i = 0; i

// 在canvas的新行上加上行标,如“000020:”,表示这是第3行;

canvas.append(lineHead(i));

canvas.append(" ");

// 把数据块的字节值用Integer.toHexString()转化成长度为48的字符串,数据块不足16个字节的,在字符串后用空格补足;把字符串加入canvas的当前行;

for(offset = 0; offset

canvas.append((i

canvas.append(" ");

}

canvas.append(" ");

// 把数据块构造成字符串,添在canvas的行尾;

canvas.append(bytesToStr(data, i * 16, (i == lines - 1) ? tails : 16));

if(i

};

}

这里的bytesToStr方法有两点非凡需要注重的地方,一是不可见字符,假如不屏蔽这些字符,则我们的编辑器的显示格式会被搞得乱七八糟,一般可以把ASCII值0到0x1F和0x7F的33个字符全部替换成0x2E,即字符小数点。二是中文字符,因为每个中文字符是2个字节,假如数据块的起始字节一个中文字符的一半(可以用ASCII值大于0x7F来判定)的时候,将会显示一串乱字符,处理方法是不显示该字节。

为叙述方便,我们把canvas中显示每一行的行标的区域称为标号区,它宽度固定为8个字符(6个字符显示标号,一个冒号和一个空格);把canvas中显示十六进制数据的区域称为数据区,宽度固定为48个字符(每字节用十六进制显示为2字符宽,两两之间有一个空格,则总宽为16×3);把canvas中每行以字符串形式显示数据的区域称为字串值区,宽度不定(最短为8个字符――全中文状态,最长为16个字符――全英文状态)。

我们的canvas是一个Swing的文本组件,我们不但用它显示数据,还显示标号和字串值,而只有数据才是答应被编辑的,所以我们给canvas增加了CaretListener和KeyListener,当输入光标落在不答应编辑的区域时,我们要把光标自动移到最近的答应编辑的地方去。

public void caretUpdate(CaretEvent e) { // 这个方法在输入光标移动时被触发

int pos = canvas.getCaretPosition(); // 输入光标相对canvas第0行第0个字符的偏移量

int line = 0;

int startPos = 0;

try {

line = canvas.getLineOfOffset(pos); // 输入光标位于第几行

startPos = canvas.getLineStartOffset(line); // 当前行的第0个字符相对canvas第0行第0个字符的偏移量

}catch(BadLocationException exception) { }

if(pos - startPos

canvas.setCaretPosition(startPos + 8); // 移动到数据区第0个字符

else if(pos - startPos 54) // 输入光标在字串值区

canvas.setCaretPosition(startPos + 54); // 移动到数据区最后一个字节

else if((pos - startPos - 8) % 3 == 2) { // 在数据区的间隙空格上

canvas.setCaretPosition(pos - 1); // 往前移一个字符

}

}

public void keyPRessed(KeyEvent e) { // 当键盘被按下时触发

int key = e.getKeyCode();

switch(key) { // 假如是方向键则移动输入光标

case KeyEvent.VK_LEFT:

setCaretPrev();

break;

case KeyEvent.VK_RIGHT:

setCaretNext();

break;

case KeyEvent.VK_UP:

setCaretPrevLine();

break;

case KeyEvent.VK_DOWN:

setCaretNextLine();

break;

default:

return;

}

}

public void keyTyped(KeyEvent e) { // 在键盘的可见字符被输入时触发

char ch = e.getKeyChar();

if((ch = '0' && ch = 'a' && ch

(ch = 'A' && ch

int pos = canvas.getCaretPosition(); // 先获取光标位置的信息

int line = 0, startPos = 0;

try {

line = canvas.getLineOfOffset(pos);

startPos = canvas.getLineStartOffset(line);

}catch(BadLocationException exception) { }

char c = (char)0;

if((pos - startPos - 8) % 3 == 0) { // 一个字节值的前4位

c = canvas.getText(pos + 1, 1).charAt(0);

model.updateBytes(line * 16 + (pos - startPos - 8) / 3, Byte.parseByte("" + (char)((ch

}else{ // 一个字节值的后4位

c = canvas.getText(pos - 1, 1).charAt(0);

model.updateBytes(line * 16 + (pos - startPos - 8) / 3, Byte.parseByte("" + (char)((c

setCaretNext();

}

}

}

到这里为止,十六进制编辑的显示和输入控制已经基本完成了,下面开始解决数据Model的问题。Model是用来保存数据的,并且提供增加、修改和删除数据的方法,还要维护一个监听者组,在数据被改变时向监听者发出通知。这里提供一个简单的实现版本。

import javax.swing.event.EventListenerList;

public class DefaultBytesModel implements BytesModel{

private EventListenerList listeners = new EventListenerList(); // 监听者组

private byte[] data = null;

public DefaultBytesModel (byte[] bytes) {

data = new byte[bytes.length];

}

public void addModelListener(BytesModelListener listener) {

listeners.add(BytesModelListener.class, listener);

}

public void removeModelListener(BytesModelListener listener) {

listeners.remove(BytesModelListener.class, listener);

}

/**

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有