我正在使用WinPcap学习网络编程。这是片段:
int ip_hlen = (ih->ver_ihl & 0xf) * 4; /* get ip header length */
tcp_header *th = (tcp_header *) ((u_char*)ih + ip_len);
int tcp_hlen = (ntohs(th->th_len_resv_code) & 0xf000) >> 12)*4; /* get tcp header length */
问题是ntohs
仅在获取tcp_hlen
而不是ip_hlen
时使用的原因。
实际上,ntohs
仅在u_short
中获取,因为参数解释了一点。 http://msdn.microsoft.com/en-us/library/windows/desktop/ms740075%28v=vs.85%29.aspx
当我没有使用ntoh时,我仍然感到困惑。
这是IP和TCP数据包定义的结构:
/* ipv4 header */
typedef struct ip_header {
u_char ver_ihl; /* version and ip header length */
u_char tos; /* type of service */
u_short tlen; /* total length */
u_short identification; /* identification */
u_short flags_fo; // flags and fragment offset
u_char ttl; /* time to live */
u_char proto; /* protocol */
u_short crc; /* header checksum */
ip_address saddr; /* source address */
ip_address daddr; /* destination address */
u_int op_pad; /* option and padding */
}ip_header;
/* tcp header */
typedef struct tcp_header {
u_short th_sport; /* source port */
u_short th_dport; /* destination port */
u_int th_seq; /* sequence number */
u_int th_ack; /* acknowledgement number */
u_short th_len_resv_code; /* datagram length and reserved code */
u_short th_window; /* window */
u_short th_sum; /* checksum */
u_short th_urp; /* urgent pointer */
}tcp_header;
答案 0 :(得分:3)
因为IP header length field非常小(只有4位),所以假设它适合一个字节,因此它永远不会有任何字节序问题。只有一个字节,因此不需要使用ntohs()
函数交换字节。
答案 1 :(得分:2)
如果您的值为8位长,则无需担心字节序。就这些。您不能在一个字节中重新排序字节。
答案 2 :(得分:0)
ntohs
是网络托管 - 简短。短路是16位。使用ntohs的一个例子是16位端口' value,如果要从sockaddr结构中提取端口值。
uint16_t portNo = ntohs(sock.sin_port);
当oyu首先将端口号放入sockaddr结构时,你应该做收敛,htons。
您只需在期望进行字幕转换的地方使用它。特别是标题和协议变量的部分。对于数据包的用户部分(您的数据),由您自行决定。
答案 3 :(得分:0)
这很简单。每当遇到通过网络接收的16位值时,您将需要ntohs
(读取 Net TO Host转换为短值)。原因是将这些数字编码为多字节时不同系统的字节顺序。