本文内容版权归本人所有,任何人或团体、机构全部转载或者部分转载、摘录,请保留本博客链接或标注来源!
题目要求: 设计一个模拟分析IP数据报头的程序 内容要求: 随意输入IP数据报头(二进制形式,20字节),分析该报头各字段的值,并以点分十进制表示输出源地址和目的地址。
时间要求: 6月16日 | | 6月17日~6月19日 | 功能分析,软件总体设计,拟定详细设计,编写程序,调试程序、运行系统; | 6月19日~6月19日 | |
那么,开始写程序之前自然要准备一下相关数据了,我们来看看IP数据包头的结构如何:
4bits: ip版本号 4bits: ip报头长度,4个字节为一个单位 8bits: ip数据报服务类型,PPPDTRUU PPP: 数据报优先级,很少使用 D: 0一般延迟,1低延迟 T: 0一般传输量,1高传输量 R: 0一般可靠度,1高可靠度 UU : 保留,尚未被使用 16bits: ip数据报总长度,包括头部和数据部分,就是总的ip数据报长度,包括被分割的ip数据报 16bits: ip数据报识别码,标记多个被分割的小的ip数据报来自同一个大的ip数据报 3bits: 特殊标志,分割or不分割数据报使用,0DM D: 0可以分段,1不可分段 M: 0此ip为最后分段,1此ip为非最后分段 13bits: 分段偏移,表示此ip分段,在整个ip数据报中所占的位置 8bits: 该数据报的生存时间,0~255,每通过1个路由器,减1,为0时,将被抛弃 8bits: 协议代码,ip数据报内数据的种类 1: ICMP 2: IGMP 3: GGP 4: IP 6: TCP 8: EGP 17: UDP 16bits: 报头效验码,检查报头错误 32bits: 来源的ip,即发送方的ip 32bits: 目标的ip,即接收方的ip 32bits: 其他参数+补齐项目。其他参数不足32bits,自动补齐
好了,结构到手,那程序就简单了! 如下为运行结果截图:
那么,上代码!
[mw_shl_code=java,true]package com.company;
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
//45 00 00 2e 13 23 00 00 80 11 5c 99 c0 a8 01 65 3b 37 cd be
System.out.println("请输入一串IP数据包头:");
System.out.println();
Scanner sc = new Scanner(System.in);
String packet = sc.nextLine().replace(" ", "");
sc.close();
//得到所有位数据
//版本号
final int version = Integer.parseInt(packet.substring(0, 1), 16);
//包头长度
final int headerLength = Integer.parseInt(packet.substring(1, 2), 16) * 4;
StringBuilder tmp = new StringBuilder(Integer.toBinaryString(Integer.parseInt(packet.substring(2, 4), 16)));
int len = tmp.length();
for (int i = 0; i < 8 - len; i++) tmp.insert(0, "0");
//优先权
final int priority = Integer.parseInt(tmp.substring(0, 3), 2);
//最小延时、高吞吐量、最高可靠性、低成本(全为0为普通包,否则只能有一个为1)
final String TOS;
if (tmp.charAt(3) == '1') TOS = "最小延迟";
else if (tmp.charAt(4) == '1') TOS = "最大吞吐量";
else if (tmp.charAt(5) == '1') TOS = "最高可靠性";
else if (tmp.charAt(6) == '1') TOS = "低成本";
else TOS = "一般服务";
//未使用位
//IP数据报文总长(包含头部和数据)
int packetLength = Integer.parseInt(packet.substring(4, 8), 16);
//标志位(让主机判断这个包属于哪个分组)
int identification = Integer.parseInt(packet.substring(8, 12), 16);
tmp = new StringBuilder(Integer.toBinaryString(Integer.parseInt(packet.substring(12, 16), 16)));
len = tmp.length();
for (int i = 0; i < 16 - len; i++) tmp.insert(0, "0");
//未使用, 必须为0
//是否分片 0:可以 1:不可以
String section = tmp.charAt(1) == '0' ? "可以分片" : "不能分片";
//如果分片, 包属于 0:最后一个分片的包 1:分片中段的包
String property = tmp.charAt(2) == '0' ? "最后一个分片的包" : "分片中段的包";
//分段偏移
String fragmentOffset = tmp.substring(3, 16);
//TTL值
int ttl = Integer.parseInt(packet.substring(16, 18), 16);
//协议号
int protocolCode = Integer.parseInt(packet.substring(18, 20), 16);
String protocol = "未知";
for (var val : protocolType.values()) {
if (val.getValue() == protocolCode)
protocol = val.name();
}
tmp = new StringBuilder(Integer.toBinaryString(Integer.parseInt(packet.substring(20, 24), 16)));
len = tmp.length();
for (int i = 0; i < 16 - len; i++) tmp.insert(0, "0");
//头部校验和
String checkSum = tmp.toString();
//原始地址
int[] sourceAddress = {
Integer.parseInt(packet.substring(24, 26), 16),
Integer.parseInt(packet.substring(26, 28), 16),
Integer.parseInt(packet.substring(28, 30), 16),
Integer.parseInt(packet.substring(30, 32), 16)
};
//目标地址
int[] targetAddress = {
Integer.parseInt(packet.substring(32, 34), 16),
Integer.parseInt(packet.substring(34, 36), 16),
Integer.parseInt(packet.substring(36, 38), 16),
Integer.parseInt(packet.substring(38, 40), 16)
};
//输出包信息
println("版本号:");
println(String.format("%d", version));
System.out.println();
println("包头长度:");
println(String.format("%d", headerLength));
System.out.println();
println("优先权:");
println(String.format("%d", priority));
System.out.println();
println("TOS:");
println(String.format("%s", TOS));
System.out.println();
println("包总长:");
println(String.format("%d", packetLength));
System.out.println();
println("标志位:");
println(String.format("%d", identification));
System.out.println();
println("分片:");
println(String.format("%s", section));
System.out.println();
println("包属性:");
println(String.format("%s", property));
System.out.println();
println("分段偏移:");
println(String.format("%s", fragmentOffset));
System.out.println();
println("TTL:");
println(String.format("%d", ttl));
System.out.println();
println("协议号:");
println(String.format("%d(%s)", protocolCode, protocol));
System.out.println();
println("头部校验和:");
println(String.format("%s", checkSum));
System.out.println();
println("原始地址:");
println(String.format("%d.%d.%d.%d", sourceAddress[0], sourceAddress[1], sourceAddress[2], sourceAddress[3]));
System.out.println();
println("目标地址:");
println(String.format("%d.%d.%d.%d", targetAddress[0], targetAddress[1], targetAddress[2], targetAddress[3]));
System.out.println();
}
static void println(String msg) {
System.out.printf("%-15s", msg);
}
enum protocolType {
ICMP(1),
IGMP(2),
TCP(6),
UDP(17);
private final int index;
protocolType(int index) {
this.index = index;
}
public int getValue() {
return index;
}
}
}
[/mw_shl_code] Felix 2020年6月17日 1:35 |