|
[size=13.3333px]一 三种类型的套接字:) N N. S( i9 w
1.流式套接字(SOCKET_STREAM)
* q! m: b- W; @! ]1 L* l 提供面向连接的可靠的数据传输服务。数据被看作是字节流,无长度限制。例如FTP协议就采用这种。. Y6 w% n% R5 Z+ j, V. ^
2.数据报式套接字(SOCKET_DGRAM) K E" T* W8 B' b! g2 n5 S3 b$ t
提供无连接的数据传输服务,不保证可靠性。
# ?% V4 l* v" y0 i! u3.原始式套接字(SOCKET_RAW); B8 r1 f3 S, r% y* Q0 |2 e T
该接口允许对较低层次协议,如IP,ICMP直接访问。
2 w$ `/ w8 f6 X6 [8 G/ } G* x
8 c. h1 a5 \! O& A" Q! E二 基本套接字系统调有有如下一些:, w8 F4 W- W, y- n2 L
创建套接字: socket()* q0 R. u+ P( p2 l z! F) b6 A
绑定本机端口: bind(): F' x4 ?2 p) D& M. R9 r
建立连接: connect(),accept()
# l( J# T" ~. s% W: e1 ~# j 侦听端口: listen()
8 J3 x5 ]0 n$ P% `6 U) m 数据传输: send(), recv() Y; C* W# d+ E0 D w0 ]
输入/输出多路复用: select()$ E" K: q% m+ Y- o! k i
关闭套接只: closesocket()
: @6 n0 w; X4 ]9 d# ]$ A7 }" g
4 A9 D+ A- _6 i x L+ C+ y三 数据类型% Q0 y* t' ~( w, T, g* l# H, Z
struct sockaddr
3 _# K- z0 R# N6 K! L. x2 E {6 q) u( X# R. Y7 Y
unsigned short sa_family; //地址族, 一般为AF_INET: c' Q8 ]' } j* p
char sa_data[14]; //14字节的协议地址) C! M" L1 G' M$ J% @. b2 `6 C, Q/ j7 {) G
}& F" E" N9 l8 S+ ^$ s- q, c# }
5 W% a$ n! Z+ r5 K' A2 n. ` struct sockaddr_in
* d1 K6 i5 N& | {
- U/ a: J" H, c! d8 K short int sin_family; //地址族( I& F% R# Q. J; B
unsigned short int sin_port; //端口号$ }% f; K. {; p8 [
struct in_addr in_addr; //ip地址0 ?7 |0 S( ^% H P$ x" P. P
unsigned char sin_zero[8]; //填充
$ i% U# f, A( ?5 Y( v" y }: ^% I% _- B- w b [/ Z4 D
8 [% }& ? j" k" V# k5 _
四 常用函数
6 Z% w: y, I/ s$ X 1 socket()
6 W% V ~1 r+ f& v$ q" E 头文件:
$ F6 z8 j+ ?, [6 K( g$ } #include <sys/types.h>" S- y. i4 K' {' j
#include <sys/socket.h>/ A/ {+ L; H, t. |; I( o
函数原型:
( j& y6 j% ?8 K/ m int socket(int domain, int type, int protocol)9 F$ Y, D7 l# e6 @* F! X5 _
domain: 协议类型,一般为AF_INET
2 I- o* ^3 V5 Z7 y type: socket类型2 Z& L7 i% g; ^3 A+ B# Z
protocol:用来指定socket所使用的传输协议编号,通常设为0即可( Y# c+ y. l! G8 |: c# d1 C4 A
5 V* b3 A$ r/ {6 R
2 bind(), y9 N4 j& N% X4 d! [/ H/ H
头文件:7 S5 {8 r2 @. e* K( _ I- G' G
#include <sys/types.h>
) p5 d4 z* F; T- N* P3 P #include <sys/socket.h>
/ B6 I) o7 `! F: u* p1 } 函数原型:, @6 r0 s) V6 F
int bind(int sockfd, struct sockaddr *my_addr, int addrlen)
# ~, a& ^% Y% f. V sockfd: socket描述符, L4 v& y$ N- p! D
my_addr:是一个指向包含有本机ip地址和端口号等信息的sockaddr类型的指针
. n1 M9 L. c5 }. x, d' y addrlen:常被设为sizeof(struct sockaddr)/ @! H& \3 |. I8 U* G# `* d
; ]* r, h# U' U: L$ J/ w 3 connect()
. I/ R0 [5 f6 ]4 M* X 头文件:2 _0 `4 A4 J3 g; u: E+ Q5 ?2 b( M
#include <sys/types.h>
! P" f( B$ N) V$ k5 r R+ j1 Y #include <sys/socket.h>: @9 u% ?2 {, c8 T+ w! R5 |
函数原型:
, d; d, u5 Y$ Y4 ^/ \, H6 { int connect(int sockfd, struct sockaddr *serv_addr, int addrlen)6 l4 C' A' g4 O
sockfd: 目的服务器的socket描述符- y, a5 Q2 D5 L; r! G8 F/ x
serv_addr:包含目的机器ip地址和端口号的指针 J8 \: O: P1 c" R' E3 e
addrlen:sizeof(struct sockaddr)
' y4 U5 `1 Y( ]! C4 C8 w
* |! |. H2 J: i+ P* M 4 listen()
* M- J2 ?2 g, O o, { 头文件:' `- p: D3 l4 R2 Z
#include <sys/socket.h>4 F9 {1 ^) Q' H; _
函数原型:
5 ~8 M( O; V. l4 G5 m int listen(int sockfd, int backlog);
/ m+ ~' m h- V4 b' @2 n sockfd:socket()系统调用返回的socket描述符' B4 z ^; j6 [7 @; J2 m; r
backlog:指定在请求队列中的最大请求数,进入的连接请求将在队列中等待accept()它们。
% W3 q {; W7 o9 u, z: E6 ^/ A! N8 I/ T3 |; {3 C
5 accept()
& o; @5 q2 _! b( T9 P/ B; H+ }9 B 头文件: # v- u. E( s3 X0 f0 G- t
#include <sys/types.h>
! ]) `; g* a) p0 p9 ^ #inlcude <sys/socket.h>
! z! N2 y! i# Q/ P4 s 函数原型:2 F: L- j) e: p0 Z, B) e( R
int accept(int sockfd, void *addr, int addrlen); c+ I% g5 p r+ l
sockfd:是被监听的socket描述符5 ^4 }7 [( V7 _; v( V
addr:通常是一个指向sockaddr_in变量的指针,该变量用来存放提出连接请求服务的主机的信息* V3 T% q( G5 x' s, p
addrlen:sizeof(struct sockaddr_in)2 T$ c- Z+ m) x
. U% ]7 x3 r' P6 p6 s
6 send()
5 `6 W: ?6 i+ \+ ?# ~4 e* @- B 头文件:
* r- |7 A w6 b4 O0 O #include <sys/socket.h>
8 V' v+ z7 Q5 ~+ N 函数原型:
% w; @( i) n, v+ X int send(int sockfd, const void *msg, int len, int flags);. O4 O3 A1 C& c7 b$ s) t
sockfd:用来传输数据的socket描述符
8 x! I8 S& W8 |. w* x msg:要发送数据的指针 ( c5 {6 o3 f9 E8 ]% G* v9 q
flags: 0
# U! N/ c/ y' ~- H
7 ^- v9 Y% ^$ j: r+ x1 H" H 7 recv(); h; X$ Z5 l2 l: k
头文件:- U' m) R {% c \1 t. d1 x
#include <sys/types.h>4 D% Z: _( u" E, o" e
#include <sys/socket.h>
9 ]: @( p+ q3 I$ k% B" k4 Y 函数原型:
# p" v5 P$ S1 ^$ C1 s; e, d2 ^ int recv(int sockfd, void *buf, int len, unsigned int flags)+ E2 K0 r1 X" }, X: U) c, w
sockfd:接收数据的socket描述符
1 o& l; x8 g4 G$ `# i6 J! F9 U/ B buf:存放数据的缓冲区# @2 B$ n3 `) j2 x9 m9 {; o( Q
len:缓冲的长度6 n- r+ ]: T9 M! t d
flags:0
+ ~+ ~( i; u- n
2 x6 u- J4 T% F0 n% ? 8 sendto()
% V6 d P; Q9 i4 Y1 I3 V 头文件:
9 j9 U- q2 l" |$ D# G+ v+ d* | #include <sys/types.h>9 \, Z2 G8 N5 d, E6 T
#include <sys/socket.h>& r0 X& u7 X9 v5 k' R
函数原型:
0 `, \+ R7 z. c! v! S int sendto(int sockfd, const void *msg, int len, unsigned int flags, const struct sockaddr *to, int tolen);' M5 \4 ~$ X2 T; U0 I+ E+ F6 ~" ]: y! K
. Z8 G5 J! \* }& i% Y
* j$ t& @8 p6 o. ]' Z 9 recvfrom()
+ k. A0 \$ l: b, |' ^. T 头文件:% j2 k1 H2 @2 _9 n/ V
#include <sys/types.h>2 Z, x, a9 _) G5 ^; {
#include <sys/socket.h>% n2 u+ {3 O3 K- M' v! A$ x
函数原型:
2 d) d3 B: n! f0 Z1 x* G5 x int recvfrom(int sockfd, void *buf, int len, unsigned int flags, struct sockaddr *from, int fromlen)
) {9 M+ D! D1 f& z8 J
* j0 `+ `! |" H
' d' O/ R; X% v4 n1 ^; ~1 T 10 read() write()) q2 \; w0 S- j3 Y8 p- n$ ~* f
int read(int fd, char *buf, int len)1 T$ T6 T4 Z7 C
int write(int fd, char *buf, int len)9 u( V4 Q& }3 ^* u' j, C6 l
" S9 J% i; p8 {* R
11 shutdown()
t' c( T8 ]' Y h close(sockfd) i( J0 N, }5 A
int shutdown(int sockfd, int how)/ q. d% R& D, @
----------------------------------- [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等) 8 m$ z3 L! h3 ]; M1 c4 Z7 d
<sys/socket.h> //与套接字相关的函数声明和结构体定义,如socket()、bind()、connect()及struct sockaddr的定义等
: m+ y$ z- B- h<sys/ioctl.h> //I/O控制操作相关的函数声明,如ioctl()
$ l b! f) s( x6 t8 G8 ?, d. d, f<stdlib.h> //某些结构体定义和宏定义,如EXIT_FAILURE、EXIT_SUCCESS等
3 c1 F5 T' B4 Y# x: v* j<netdb.h> //某些结构体定义、宏定义和函数声明,如struct hostent、struct servent、gethostbyname()、gethostbyaddr()、herror()等
& l/ K5 i( E9 d' ]<arpa/inet.h> //某些函数声明,如inet_ntop()、inet_ntoa()等
% ~1 ^) F/ ]( v p) r<netinet/in.h> //某些结构体声明、宏定义,如struct sockaddr_in、PROTO_ICMP、INADDR_ANY等 [size=13.3333px]------------------------------ [size=13.3333px]linux下socket编写常用头文件
6 `9 r) F. }: W F$ J- F
3 }5 @) I( q6 X- k[size=13.3333px]#include <sys/socket.h> //connect,send,recv,setsockopt等
- L% m8 {4 w, h* N! K; q#include <sys/types.h>
) [3 q) E0 I% W) s1 C1 ]/ U4 K
5 t1 c; I* _; C. \#include <netinet/in.h> // sockaddr_in, "man 7 ip" ,htons. } L( J; o% k7 e6 v4 d( n/ \3 p4 S
#include <poll.h> //poll,pollfd4 |+ m3 [1 V. @1 X' L! F( O4 C/ B
#include <arpa/inet.h> //inet_addr,inet_aton) G" m8 }, r& u0 y' E, n
#include <unistd.h> //read,write/ R6 S/ V( k2 K5 q( p8 q" y/ D1 {' F
#include <netdb.h> //gethostbyname
3 \8 g ?' l7 O n$ e5 L
' [8 E; O4 L* K$ W$ u3 ^#include <error.h> //perror" {( B/ R. H% p& j
#include <stdio.h>$ H" r$ V# O! z" s
#include <errno.h> //errno
4 [2 Z6 a- f$ y5 b$ f3 m7 t% a! I5 q6 S- c
#include <string.h> // memset, Z! J9 N# N/ q
#include <string>. D, t9 `8 Z3 h4 I( Z3 W# \0 x
#include <iostream>" a% [ m2 M# S. g
W Y0 Q+ R! \& O8 Z
|