|
[size=13.3333px]一 三种类型的套接字:
# ~6 ?5 H3 |+ ]( u, t, O7 ]& h1.流式套接字(SOCKET_STREAM)& P3 S. G) r! l9 e( O6 y: \
提供面向连接的可靠的数据传输服务。数据被看作是字节流,无长度限制。例如FTP协议就采用这种。+ K* e( w+ i; `. R1 h
2.数据报式套接字(SOCKET_DGRAM)) ^5 l+ q( v- B# k8 k r
提供无连接的数据传输服务,不保证可靠性。) B J: e9 O6 H: K
3.原始式套接字(SOCKET_RAW)
: A7 S' ?/ b0 K: r; L4 C9 F+ w7 d 该接口允许对较低层次协议,如IP,ICMP直接访问。
5 [& L& P% V. t" o- n
" J' O5 ~/ `4 {0 x) r" q& S二 基本套接字系统调有有如下一些:9 U, H. L& |' a X
创建套接字: socket()
8 V4 D- U; q w3 |4 g- Z 绑定本机端口: bind()
- q- Z5 Z% ]1 t+ I- k 建立连接: connect(),accept()
) |0 r% C9 y1 A" Y 侦听端口: listen(), l3 B6 L6 X9 e
数据传输: send(), recv()- X' |+ {/ X g: d
输入/输出多路复用: select()# A. L. E/ k! {- B
关闭套接只: closesocket()
1 l/ e! |- D. E# [) P
8 ~( n t) t- N% n5 M三 数据类型% F* Y3 d6 ]' G' h1 ~7 \& G# u8 d
struct sockaddr% i8 e/ t0 K2 V" J
{
! v' l) V6 D- P0 i* v unsigned short sa_family; //地址族, 一般为AF_INET: o2 u- f! H2 M
char sa_data[14]; //14字节的协议地址
4 ~0 U, V' k% y/ m/ k2 y8 d }
- n6 e' u+ p' |) g$ ~, s0 ^2 J o3 K) K& X9 `# [
struct sockaddr_in
3 Y3 U9 F1 R: } U# D, T {
. v2 a% p5 q$ b- S2 d2 s- r5 @4 Y short int sin_family; //地址族
# F4 S7 a# ]+ i/ U5 w7 R/ P unsigned short int sin_port; //端口号
3 I# m9 S L, s; r& R struct in_addr in_addr; //ip地址
0 p3 h* x- l1 {# v' Y unsigned char sin_zero[8]; //填充+ K4 @" D# X9 N( k6 u; @
}% h* \3 w3 t& V$ e ?/ Y
* _/ t7 y, G" M1 D" [四 常用函数3 G; [0 `: D7 C9 Z5 r# I
1 socket()
j: i2 o/ O: y( Z$ C8 f 头文件:
6 I/ \0 z/ \7 X! W #include <sys/types.h>5 K3 E- |; @. G% j8 Q! x
#include <sys/socket.h>. q1 B' P9 B7 g+ m
函数原型:
% W8 J# f) Z% H$ ^- E6 A3 w int socket(int domain, int type, int protocol)
, S7 C7 d1 b+ m/ G3 t: R R2 E. ^ domain: 协议类型,一般为AF_INET5 K! [; j" c$ C1 }7 D
type: socket类型( p9 [9 H& e( A) Z% t! u9 v
protocol:用来指定socket所使用的传输协议编号,通常设为0即可" |$ T ?! r$ f3 m+ r3 x+ v( U
$ i# }2 k3 K0 ` 2 bind(): C+ G9 F. R/ U3 t
头文件:- T1 U# [7 M7 o; W. Y) S; N& d
#include <sys/types.h>, j4 Q; d. h% ?* p' v* ~( V
#include <sys/socket.h>" _* l& F" P' h* V Q% @8 W8 R
函数原型:. `7 G! C8 F ]) }! C
int bind(int sockfd, struct sockaddr *my_addr, int addrlen)
3 E6 U0 m+ K! g) T0 Y4 H3 @, D sockfd: socket描述符
$ Z, p9 j( X9 `4 R& C& R8 { my_addr:是一个指向包含有本机ip地址和端口号等信息的sockaddr类型的指针4 a7 D4 z- |3 }5 \$ |9 s. h0 f
addrlen:常被设为sizeof(struct sockaddr)
/ b$ n4 ~* Z0 l `5 g* F2 r, P
7 t) B2 h4 x, f1 \9 K8 }7 @ 3 connect()! o# D* E2 c2 E: \
头文件:* V+ D% D/ g3 w
#include <sys/types.h>
" F* ^& I* Y: P e+ I# s #include <sys/socket.h>
5 O6 S6 `9 m1 D 函数原型:& J- ?6 B* W! @3 }) p
int connect(int sockfd, struct sockaddr *serv_addr, int addrlen)7 L" ~2 Z5 j" O+ Z
sockfd: 目的服务器的socket描述符" m( K K- w: S. O2 e1 a
serv_addr:包含目的机器ip地址和端口号的指针9 r p$ g& d% R# X/ L
addrlen:sizeof(struct sockaddr)! P, @/ e: P0 O1 n
9 s1 s5 V7 Z8 x7 A' W: K9 j% R
4 listen()
: d" _1 t7 ?% Q3 a4 E6 _; l- ^ 头文件:
' t2 a$ C8 n3 T #include <sys/socket.h>6 {/ N' s' L6 l1 K1 h" N
函数原型:
* I+ n, l# P9 n6 E# [. k int listen(int sockfd, int backlog);1 R) O# |$ J3 Z0 Q- y+ `" e% b
sockfd:socket()系统调用返回的socket描述符. O( P9 V$ `3 {6 i/ A4 p2 @$ T
backlog:指定在请求队列中的最大请求数,进入的连接请求将在队列中等待accept()它们。
5 R% Y3 p. S) {
6 R, F8 O, X6 h8 Y2 } 5 accept()
6 @: q" J0 ]. V4 q/ U* S 头文件: 5 x7 V' {: ?0 j
#include <sys/types.h>6 {3 ?. d' P/ C: A+ G; u# K
#inlcude <sys/socket.h>2 v4 k+ t D" d9 x; ]$ Q
函数原型:
1 Q* @* s6 U) W1 {; a* q% q int accept(int sockfd, void *addr, int addrlen)
) Z1 r, j; r$ V, s) h sockfd:是被监听的socket描述符
& P0 {6 H$ c" C' F addr:通常是一个指向sockaddr_in变量的指针,该变量用来存放提出连接请求服务的主机的信息4 `% |6 n, t* K6 d! V* h9 P/ R
addrlen:sizeof(struct sockaddr_in)
) }* i# \$ w' E
0 d4 ~+ M# M2 {/ Y5 C3 I! O 6 send()
* X$ V6 z" n( p D7 m, T 头文件:) j2 }8 e3 A/ m! J [# ^8 @
#include <sys/socket.h>
) j$ R! d- J: M, S7 O/ e 函数原型:
|5 F. Q& Q$ p3 e int send(int sockfd, const void *msg, int len, int flags);
' e# |- E3 l! j+ q( P; _0 Y% r sockfd:用来传输数据的socket描述符$ X. a$ R2 [" X7 b- d9 I6 o; @* \
msg:要发送数据的指针
! O$ N) @3 e2 n( O! _ flags: 0, m1 b( t8 Z8 h) o8 @; F
( d( N1 l- o7 o. d: p8 [- n) b 7 recv()
! e' g3 a9 I A# E' ]6 j 头文件:
. L6 j! t2 A( A #include <sys/types.h>( e1 d1 S+ o5 H6 K# R
#include <sys/socket.h>5 g% f$ I: T) B4 D9 R# ^$ m
函数原型:
- G F* s+ Y- O- K" t) h3 |# F int recv(int sockfd, void *buf, int len, unsigned int flags)5 s. {- w1 @+ x; r) i& h
sockfd:接收数据的socket描述符
7 z2 j+ }6 a% f+ S% ~" {/ W buf:存放数据的缓冲区
2 A% F. P6 z7 w, Q) |% ~' w# w9 X len:缓冲的长度& _; e7 b S/ i2 Z8 d$ K7 C# i
flags:0
7 }, U2 h4 Y, D! ^+ e6 B! B6 [- u1 E% d, b! r
8 sendto()! P- Z0 x. @- p3 T) ]! e m; X3 e/ b
头文件:9 f0 H1 x0 s8 X0 ]. J! g+ C8 o: w2 H
#include <sys/types.h>
2 B+ U$ y; e2 I9 b2 E #include <sys/socket.h>
& \, F' K; R4 J5 C: {5 _ 函数原型:
* l7 D' B2 D$ Q# V int sendto(int sockfd, const void *msg, int len, unsigned int flags, const struct sockaddr *to, int tolen);' k* z& X$ l& q* M5 G8 E) q* Y
3 a B0 V d5 b
8 I) E# H/ X5 L. P1 @ 9 recvfrom(). v8 S2 k* y! I
头文件:9 H2 @6 j4 n: f( ]6 ~
#include <sys/types.h>- z6 o2 `: U' G; G, B* H9 d$ p0 n r
#include <sys/socket.h>
) h9 {" a2 B5 _0 p! L$ y. p6 u 函数原型:
6 ?/ E( W# w. N( w$ `5 k7 V int recvfrom(int sockfd, void *buf, int len, unsigned int flags, struct sockaddr *from, int fromlen)% x; u+ w' U: S7 E
% o3 C" j/ I& W9 X$ s
, I8 k& ?1 c4 J8 {1 K; ]: b3 ` 10 read() write()
3 X1 N4 A0 Z8 ~; U/ X4 v int read(int fd, char *buf, int len)
( B/ S0 E* I! h1 U* K j& b int write(int fd, char *buf, int len)+ f: u- b2 x" `) Q8 q8 T
8 A' y0 C0 C* |, l
11 shutdown()
$ N9 I! y. @: r& [* V+ K close(sockfd)
. a, u8 p8 ~- z/ I4 s( H+ N int shutdown(int sockfd, int how)
^; s9 Q4 J5 ]4 m J& u: r----------------------------------- [size=13.3333px]netinet/if_ether.h ether_arp的数据结构 [size=13.3333px]netinet/ether.h 以太祯的网络字节和ascii字节的转换,包括ether_ntoa(),ether_aton这样的函数定义 [size=13.3333px]netinet/ip.h 这个头文件和linux/ip.h似乎很相似,也有iphdr的数据结构,同时还包括了timestamp结构,我的理解是,linux文件夹下的 ip.h是linux黑客编写的ip头文件,而这个则是gnu一开始就定义的头文件,同时还包括了bsd中的ipheader结构定义。同理的还有该目录 下的tcp.h等文件 [size=13.3333px]linux/ip.h iphdr的数据结构,以及一些ip层的数据定义,同理的还有tcp.h,udp.h等等 [size=13.3333px]linux/if.h 主要的socket头文件,似乎修改自unix的if.h,定义了网卡的接口信息的宏,例如IFF_UP.另外有数个重要的interface的数据结构定义,包括ifreq,ifconf,ifmap [size=13.3333px]linux/if_packet.h 原始数据包的数据结构定义,包括sockaddr_pkt,sockaddr_ll,想接收原始数据包的不能错过这个文件。同理的还有if_ppp.h,if_tun.h等等 [size=13.3333px]netinet/in.h 这个文件作的事情就多了。端口宏定义,著名ip(比如loopback),结构sockaddr_in,网络字节转换(ntoh,hton。。。。)。。。反正太多了,没事的话就把这个文件加到头文件包含里吧 [size=13.3333px]netdb.h 文件如其名,包括结构hostent(主机环境),获得主机的信息的几个函数(gethostbyname)。似乎这个就是定义主机的各项环境,例如hostname等等 [size=13.3333px]net/bpf.h berkeley的数据包过滤头文件,想用bpf进行包过滤的要重视一下这个文件 [size=13.3333px]net/ethernet.h 包括几个以太网的数据结构,ether_addr(mac帧结构),ether_header(以太帧的头部) [size=13.3333px]------------------------------- [size=13.3333px]<sys/types.h> //primitive system data types(包含很多类型重定义,如pid_t、int8_t等)
) K& O; V: Z$ p8 Q<sys/socket.h> //与套接字相关的函数声明和结构体定义,如socket()、bind()、connect()及struct sockaddr的定义等5 h, Y9 `7 r, Q, Z# i! o% k w
<sys/ioctl.h> //I/O控制操作相关的函数声明,如ioctl()1 W( J. s! y. K8 h& l( G; V
<stdlib.h> //某些结构体定义和宏定义,如EXIT_FAILURE、EXIT_SUCCESS等
0 g! D& Q+ t! `4 K% {. [+ M<netdb.h> //某些结构体定义、宏定义和函数声明,如struct hostent、struct servent、gethostbyname()、gethostbyaddr()、herror()等- m9 O* m+ @5 j7 N9 T8 V. |
<arpa/inet.h> //某些函数声明,如inet_ntop()、inet_ntoa()等
! P- s) J; |! ]2 y5 N3 j& {<netinet/in.h> //某些结构体声明、宏定义,如struct sockaddr_in、PROTO_ICMP、INADDR_ANY等 [size=13.3333px]------------------------------ [size=13.3333px]linux下socket编写常用头文件# ]) M( h& W0 E8 u" j7 F% t
$ G6 K8 v: [* z2 |* t$ F: h* T
[size=13.3333px]#include <sys/socket.h> //connect,send,recv,setsockopt等! `* B `% E* O
#include <sys/types.h> + d. d& N* m- B8 r9 B
5 F9 x$ O. j, D, y. }* O#include <netinet/in.h> // sockaddr_in, "man 7 ip" ,htons
6 Z% n* k% g9 q, g4 l5 b#include <poll.h> //poll,pollfd/ i( x/ O7 w+ ~
#include <arpa/inet.h> //inet_addr,inet_aton
% L8 ~& |1 T* r0 p0 z( F#include <unistd.h> //read,write5 z& e: {0 `, l" `/ ~" h! J
#include <netdb.h> //gethostbyname' t! U- V2 A6 n, o) Z1 {/ g
; {$ _9 ~4 ?" l( E9 j#include <error.h> //perror5 s, q0 ]: _5 b0 m
#include <stdio.h>
" ~. _+ ~5 b- o4 h#include <errno.h> //errno
- @, u {3 Q- J3 z" t; ?4 l) `1 C( [+ C( _
#include <string.h> // memset
% ~. E/ A2 d6 w; o#include <string>4 k. N: a P$ S, |0 U" E% o& B
#include <iostream>
* v9 _ {/ q4 A$ C, j
3 w; j* D, i" N& B, h+ H |