|
[size=13.3333px]一 三种类型的套接字:
1 [- B7 e. X* ]% I4 q2 [) }) v1.流式套接字(SOCKET_STREAM), W# F3 f$ H8 ~7 Z6 A5 o0 C
提供面向连接的可靠的数据传输服务。数据被看作是字节流,无长度限制。例如FTP协议就采用这种。
5 ^) g* B( ~( G& C" B2.数据报式套接字(SOCKET_DGRAM)! B- N# _ }: U% V# O- h$ o1 N
提供无连接的数据传输服务,不保证可靠性。( U; ]: p3 Y0 X4 _
3.原始式套接字(SOCKET_RAW)
$ ~% e7 B0 ~; O m 该接口允许对较低层次协议,如IP,ICMP直接访问。2 N$ K2 J% J% p/ S
: i+ j, o. `' t, H9 p
二 基本套接字系统调有有如下一些:
v j+ h' x5 y7 M5 n 创建套接字: socket()
: Q, o5 u P3 x9 C# _# |( f5 v6 T 绑定本机端口: bind()
3 P4 Y2 Q0 ]& c; n) c( o/ } 建立连接: connect(),accept()
" t. n+ ^( b6 _ 侦听端口: listen()# c& c" W4 {; C( d& B6 P8 B* t
数据传输: send(), recv()
) |5 g4 i. i+ e+ V 输入/输出多路复用: select()# J* ]5 K( g% Q+ Q, I
关闭套接只: closesocket()
- g @' c9 ~2 {$ V! u b+ L
# p- I" `. I1 D% I三 数据类型% m! ? n5 X% {1 R7 z8 O" g: b9 R
struct sockaddr+ d8 S, T8 k4 h
{
& i2 L! a+ D* f7 f, f unsigned short sa_family; //地址族, 一般为AF_INET/ W4 K! D/ y+ B1 l- c2 O, e) `' ]
char sa_data[14]; //14字节的协议地址
8 A/ }, j6 n6 P( I! Z }5 ^- K4 C" _6 G7 K: H2 c* i# v
8 E8 u: o7 Z' b' o1 U' g; G5 [ struct sockaddr_in
/ R K) J' ^9 ]0 D6 e {
* a! Y7 y8 V: @6 ^+ }/ ~ short int sin_family; //地址族
$ ?7 |/ Y0 C; @$ \7 d6 V unsigned short int sin_port; //端口号
; Y7 Y2 X, P* K+ I struct in_addr in_addr; //ip地址6 N& A5 `) O2 F! B2 y W6 c; X n5 k
unsigned char sin_zero[8]; //填充$ x7 N, `- E W" q4 i
}' p2 D o8 H1 D* y8 p! W2 @4 U: o
8 T8 d# E" n5 g& x; b# ^7 e% l
四 常用函数
7 b g0 P6 Y \ 1 socket()2 u2 I+ k. O @3 i9 v- Q7 x5 }
头文件:
0 k- I0 B9 {/ v4 y #include <sys/types.h>2 W# S# A; C2 r, d+ r
#include <sys/socket.h>
) R* R) M1 @/ p4 |! ^1 p5 U 函数原型:
9 J( M: l: {% r! j' B! H: ^ int socket(int domain, int type, int protocol)
% D+ n0 V* E9 ? domain: 协议类型,一般为AF_INET
7 ]& b. h9 J/ @* m, j k6 A U type: socket类型/ M N" h B+ k7 v# N
protocol:用来指定socket所使用的传输协议编号,通常设为0即可
' d8 S% E& K- C* b
4 L: |0 s3 ?4 e5 w" s1 Z 2 bind()
2 Z! q) b) d( h3 L+ ^ 头文件:3 f, `) V7 n( ?! K
#include <sys/types.h>' ~& O2 s D& I: d. m; h: D) @
#include <sys/socket.h>
2 b1 J6 E% I V) a6 n 函数原型:
3 _* F, l9 }, D& \* d$ [ int bind(int sockfd, struct sockaddr *my_addr, int addrlen)
" [5 Q4 F# h5 w sockfd: socket描述符. S+ |# v; N- J6 A# V
my_addr:是一个指向包含有本机ip地址和端口号等信息的sockaddr类型的指针) {; D& F9 C4 }; j( F% E" p
addrlen:常被设为sizeof(struct sockaddr)
6 i9 j1 O% a7 C0 c) C! u9 x0 U6 O) c! p" t. w" H
3 connect()
- _6 H6 G1 u! u' B. B4 [7 t C9 s 头文件:6 d: x8 ^3 C s
#include <sys/types.h> ; [7 ~, w5 H9 H% I, v8 V4 `
#include <sys/socket.h>9 E+ K- @- m1 j+ h0 g
函数原型:
3 s; W3 u1 I" z/ b2 ^0 w3 C int connect(int sockfd, struct sockaddr *serv_addr, int addrlen)
$ M2 ^, S9 [% O/ H$ S: f sockfd: 目的服务器的socket描述符" J9 Z/ \& h2 f
serv_addr:包含目的机器ip地址和端口号的指针
1 N; T) C, t# L$ W addrlen:sizeof(struct sockaddr)
& v+ y) r8 @: |- a2 X
( D w/ [4 I2 z 4 listen()
, y; o; h f8 O1 N+ Q6 t, G( N 头文件:
8 Y* J$ E4 p: l. G4 I: @ #include <sys/socket.h>
( F% n( B3 y! q5 K" [% E2 l 函数原型:9 ~2 j5 P9 x% d, I; F
int listen(int sockfd, int backlog);
% A# J+ W8 u* G1 t sockfd:socket()系统调用返回的socket描述符7 W8 M: [# g* f( a- W9 W
backlog:指定在请求队列中的最大请求数,进入的连接请求将在队列中等待accept()它们。
* x/ r F& @6 g! j+ z" B7 x4 C7 w8 s. x
5 accept()
+ O5 ~9 X4 K, l8 ~0 I- L 头文件: # u5 ~# t) `3 q$ R
#include <sys/types.h>
# l; `. ^3 j) W9 h4 ~' q #inlcude <sys/socket.h>
0 }3 Z# n+ [6 s. |- n 函数原型:
, s2 A9 N) b1 C( z; y4 o- ~* } int accept(int sockfd, void *addr, int addrlen)" @9 A7 b8 I! M6 D+ b9 h
sockfd:是被监听的socket描述符
3 H8 a# z* k7 c# q; j | addr:通常是一个指向sockaddr_in变量的指针,该变量用来存放提出连接请求服务的主机的信息
0 g( U8 j/ e2 ]# | }+ @7 F addrlen:sizeof(struct sockaddr_in)* b; ?, M h5 K' f
( i6 W5 f7 O/ A1 j/ p+ E: Q
6 send()
7 Z/ K/ D0 t5 G( l8 L+ z 头文件:3 H3 K" n+ q s2 i
#include <sys/socket.h>
2 F: R8 V) e, r" `) X 函数原型:! H8 }& Q4 l# G& Q a8 O' ?
int send(int sockfd, const void *msg, int len, int flags);/ v5 H& ?: C$ _4 Q2 F1 ~
sockfd:用来传输数据的socket描述符% h( h4 b! o1 ^% n: f: R1 t
msg:要发送数据的指针 * j1 V9 e& ?, K' T: q0 F
flags: 0' b# P* y8 K& g6 _" H
4 S8 O+ h! I! r6 V. G) ^
7 recv()
" C x; ]# B$ C2 Y o 头文件:
" q: ?: [ S7 T( ~& v #include <sys/types.h># v" w4 M$ @- X" \
#include <sys/socket.h>9 |7 N4 O7 j' T* O# [1 {( N
函数原型:
; h4 g* S/ {* K6 i7 s int recv(int sockfd, void *buf, int len, unsigned int flags)
7 z6 T8 ?2 [; P4 q sockfd:接收数据的socket描述符& |* u/ I! M) H2 o0 E$ R
buf:存放数据的缓冲区8 L0 i3 @& m$ T7 Y* Z
len:缓冲的长度
. v/ P4 G. z7 X( U Y8 k7 J) {% x flags:0
+ |/ a7 I9 G( n! }( S6 X/ i8 R; J. q; b
8 sendto()
3 b. B. C" @# A$ l5 Y. E. p 头文件:
0 H, \5 X% O6 U' L- k* K- }; j( R( T #include <sys/types.h>; k' `- P# b! f* g8 `- S1 j' V
#include <sys/socket.h>
' _2 c1 ]# J% Q( Y) m4 ?. H 函数原型:
2 _( k" J- F) r! c& I/ f% u int sendto(int sockfd, const void *msg, int len, unsigned int flags, const struct sockaddr *to, int tolen);$ r% y6 Y# S, P2 W) [% ? P
+ ?6 `1 {3 R* ?: t8 i+ [+ ^) Q / S# c6 X) \. g
9 recvfrom()
" U e3 {: d, W" o" \4 U; t' q3 g3 a: o 头文件:
/ m6 W* E2 @5 T8 f5 R #include <sys/types.h>; A3 N& T6 k8 D; r: f) C7 r
#include <sys/socket.h>9 |+ \8 W$ H. W0 i* E2 E
函数原型:
8 u+ J7 |( }" K+ J int recvfrom(int sockfd, void *buf, int len, unsigned int flags, struct sockaddr *from, int fromlen)
+ x% O8 \. i Z( @
) t" J2 Z' U3 P% P' P/ Z- n8 Z& Z( \
: r' {) }. ]3 h+ I# | 10 read() write()- [3 Y! e0 s! z7 q5 f/ X' k: W
int read(int fd, char *buf, int len). \! ]# @! z1 h; i8 t! Q
int write(int fd, char *buf, int len)" X1 {& [9 m- s' q; u& p+ i
; z- N. G& R# ~2 `7 |! L9 F; V. q- H 11 shutdown(): o7 r+ F/ m7 u7 z1 k
close(sockfd)
3 {6 O8 P; _; L int shutdown(int sockfd, int how)+ x3 l* y1 {8 Y2 H
----------------------------------- [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等)
) S1 q" ^5 _* ~8 r<sys/socket.h> //与套接字相关的函数声明和结构体定义,如socket()、bind()、connect()及struct sockaddr的定义等. x* Z" u6 _2 I& L$ B2 I, L: g
<sys/ioctl.h> //I/O控制操作相关的函数声明,如ioctl()# p/ m0 k8 Z/ i0 L7 z# V, U8 w' h, ?# q
<stdlib.h> //某些结构体定义和宏定义,如EXIT_FAILURE、EXIT_SUCCESS等3 u) B1 F' T! Z4 U
<netdb.h> //某些结构体定义、宏定义和函数声明,如struct hostent、struct servent、gethostbyname()、gethostbyaddr()、herror()等
5 e$ C: j9 O+ l<arpa/inet.h> //某些函数声明,如inet_ntop()、inet_ntoa()等 m- v; X$ } c1 P; A+ `
<netinet/in.h> //某些结构体声明、宏定义,如struct sockaddr_in、PROTO_ICMP、INADDR_ANY等 [size=13.3333px]------------------------------ [size=13.3333px]linux下socket编写常用头文件
3 n) S6 o' L# z+ r7 g, R. x
* y# I+ L* F% i[size=13.3333px]#include <sys/socket.h> //connect,send,recv,setsockopt等9 R* D2 |$ k y) A4 [/ x
#include <sys/types.h>
: x$ S) u2 D- k+ G/ c& c+ |. N- ?8 a6 ]7 ~ F) Q9 r8 I
#include <netinet/in.h> // sockaddr_in, "man 7 ip" ,htons; p# Q' d( a) k" R( m
#include <poll.h> //poll,pollfd
- g! H) z9 E. A#include <arpa/inet.h> //inet_addr,inet_aton5 V4 o$ P- T* Q$ {) i0 L
#include <unistd.h> //read,write
2 J) m0 i& \7 k5 _3 f#include <netdb.h> //gethostbyname7 T9 O2 Q5 X" H! l- L
) i; V" Y4 p1 Z: K7 Z
#include <error.h> //perror1 s r* r9 v/ s& g% U
#include <stdio.h>5 ]9 W1 @& S- w) `% v+ O" D
#include <errno.h> //errno+ t9 f; O0 s% K" x B8 |4 r/ Z
~* e2 r5 [0 O# O#include <string.h> // memset0 [& E( `/ N, w3 G h# f! W+ N3 e
#include <string>
% g* M% B. t* {- n. w1 B#include <iostream>" H! Y1 d$ x; D
! N6 w# Y, R' A- n; j; w |