|
[size=13.3333px]一 三种类型的套接字:
2 }1 T% ]+ Y6 n! ~ u3 e2 z1.流式套接字(SOCKET_STREAM)
9 e- k# p, N- H7 U 提供面向连接的可靠的数据传输服务。数据被看作是字节流,无长度限制。例如FTP协议就采用这种。% C0 g: ^5 }7 L* }( f: z
2.数据报式套接字(SOCKET_DGRAM)% U G* ^3 U: [* ?4 t5 Q3 o H/ H
提供无连接的数据传输服务,不保证可靠性。% j. e* ]0 V$ x9 q1 l* U7 M$ J
3.原始式套接字(SOCKET_RAW)5 G2 n5 B* I9 ]8 ]
该接口允许对较低层次协议,如IP,ICMP直接访问。; H$ @+ V1 [7 X# a" X
( F6 u( a& b8 F- Q4 e# r
二 基本套接字系统调有有如下一些:. J, S8 V% _& k# {: [
创建套接字: socket()" S. u m; i3 b& D' o# C/ |& y+ ]
绑定本机端口: bind()
7 i6 `2 P7 N. i5 R1 t* b 建立连接: connect(),accept()
4 |: w9 m) q, F; Q3 E. h2 e& L+ @ 侦听端口: listen()
3 B7 Y T( b r" M7 H4 g% t 数据传输: send(), recv()
4 ?8 Y6 [& m, \$ R8 X7 K 输入/输出多路复用: select()( J+ D3 C2 f! T4 c+ h
关闭套接只: closesocket()4 v( i9 J9 w+ K1 a L
V1 h* S- @9 k& H1 ?( c" ?2 R三 数据类型
/ D7 d3 o6 p) _2 K1 N4 l2 ], i4 ~. @' H struct sockaddr
' z) h3 t6 l/ l/ O* j" } r {8 V3 N0 w+ T& N( @
unsigned short sa_family; //地址族, 一般为AF_INET
$ V# g" @" W5 E1 @) O+ k- w- D" g/ e char sa_data[14]; //14字节的协议地址
4 J* w- O8 b/ `, b% a# b }
4 ?5 H9 u/ x1 i, k
; F3 l: ]7 z9 Y5 ?! G$ B3 f struct sockaddr_in0 Q$ j! T7 \1 B4 z4 d- ^
{/ u+ n- q5 E2 b$ W% W2 u2 z: P9 {
short int sin_family; //地址族
$ e+ n8 Y) D7 k% ] unsigned short int sin_port; //端口号( [; e+ q" g* f2 q4 s
struct in_addr in_addr; //ip地址+ q& z- D; a7 W2 d6 H( n
unsigned char sin_zero[8]; //填充
" ]8 T' c$ r3 G }; v4 V) D' ?& U. v u; n
5 C' {9 H3 z+ H/ v# L5 s- z四 常用函数
+ a* n% ~" ~0 b4 j1 \; g! p 1 socket()
! M& p# }) d- B' O- s 头文件:
6 H/ ?* f, D, }. |8 i #include <sys/types.h>
( ]9 V8 W+ `, ^7 [ #include <sys/socket.h>
- {. ]' G4 W# d: Z- c 函数原型:& C5 r: I: f" U3 N; E
int socket(int domain, int type, int protocol)" [; ?0 v2 _! Z1 L d& O
domain: 协议类型,一般为AF_INET
* \% U( U: f1 _- T! f% V9 L) l type: socket类型
3 ?" P- w% [8 u7 j4 u: {+ D! W protocol:用来指定socket所使用的传输协议编号,通常设为0即可
1 v* T: o2 H! m5 I- Q5 ]. V7 n6 b4 o; D: `7 n, N9 E0 J" D3 P
2 bind()( l) D& Z I! V# Z: c' n
头文件:, `& D4 L) z; ^3 B- D- U
#include <sys/types.h>% _% G: n5 l) R
#include <sys/socket.h>. l" L3 W, Q5 E$ a
函数原型:
9 `2 Q% ?" Z* L int bind(int sockfd, struct sockaddr *my_addr, int addrlen): g( F2 m5 r2 F7 W
sockfd: socket描述符$ `2 D& V$ E4 D& E9 T8 K4 x; o( v
my_addr:是一个指向包含有本机ip地址和端口号等信息的sockaddr类型的指针
8 ^6 w" f; @9 H' q) d addrlen:常被设为sizeof(struct sockaddr)- G) n q0 I6 H" N2 H
' F$ p. I: |5 ^: s2 ?2 S 3 connect()
# T: B# h' @6 E 头文件:; Q! @6 U. o3 T0 D6 l9 D
#include <sys/types.h>
. R- ]0 b* C. {# R$ D6 m7 _ #include <sys/socket.h>
9 t# _! \1 U/ H$ h2 x U. N 函数原型:
/ x2 y6 ~! J# Z0 y2 L int connect(int sockfd, struct sockaddr *serv_addr, int addrlen)
$ s4 N* U' W) j0 @2 ] ] sockfd: 目的服务器的socket描述符1 g. y2 V' u: M' k" U
serv_addr:包含目的机器ip地址和端口号的指针2 l4 B0 A4 @. Q% ~' _
addrlen:sizeof(struct sockaddr)
' H- t1 t" @7 G$ x; {% P
( N w7 ?1 X' P 4 listen()
( j7 d9 a1 J I% _8 b+ k 头文件:6 c5 m7 g9 U. B1 I" e
#include <sys/socket.h>
( |" [; p# |2 |8 K8 Z1 N+ t 函数原型:) f q' W6 {& a; ^ @5 S B& ?
int listen(int sockfd, int backlog);" J* y, K0 z5 q
sockfd:socket()系统调用返回的socket描述符' {0 Y* ?- s5 `) @
backlog:指定在请求队列中的最大请求数,进入的连接请求将在队列中等待accept()它们。
% |* W0 S5 J1 I' M! w# U- w3 t& L& @. c0 S
5 accept()7 i+ O% d3 R4 _) Z# M
头文件:
7 }/ L3 ~4 C& A g6 }* Y: U# q, l #include <sys/types.h>
4 _( l8 Z1 z+ q% K L& a. A) O# @4 K #inlcude <sys/socket.h>, Y" @/ h' |5 A% o" l
函数原型:- d7 c6 p& V8 g4 d1 P) j' `, n& d
int accept(int sockfd, void *addr, int addrlen)
3 w1 }/ {! _# b" K+ R W sockfd:是被监听的socket描述符
3 E, ^1 v/ H, ]& L7 s addr:通常是一个指向sockaddr_in变量的指针,该变量用来存放提出连接请求服务的主机的信息
/ [2 ?' R( C) t9 d+ B3 W" u0 ]) U addrlen:sizeof(struct sockaddr_in)
0 Q j, f3 N" U/ _ . [2 ~) \" T$ D7 g9 n+ \
6 send()$ H" a+ I: e4 D
头文件:
3 }0 p2 O# f, p; A7 u0 v+ \ #include <sys/socket.h>3 i [) ]( ?$ y( g
函数原型:- x+ g' b! k8 A: t R7 i
int send(int sockfd, const void *msg, int len, int flags);
5 {$ j9 e- d5 i- Y& c7 g% l) }% {8 ~ t sockfd:用来传输数据的socket描述符( o, d: r$ L& V& ?* D
msg:要发送数据的指针 . L% q; C# {6 m+ b' n7 E2 e
flags: 03 \6 b; J9 \' g) f' w) i
2 d( m; H8 x, |8 |$ \
7 recv(). k9 v3 a% G% U" @
头文件:
7 P6 c7 t; a5 ]2 K #include <sys/types.h>' o+ C# x) y0 m
#include <sys/socket.h>
- H% j" L G# Z9 _ 函数原型:+ x' f4 T' b `0 B
int recv(int sockfd, void *buf, int len, unsigned int flags)( h; i0 C+ ~: Q& ^; I6 P- F
sockfd:接收数据的socket描述符9 ]+ q' d' r( s0 z$ [0 A
buf:存放数据的缓冲区
1 _% q: ?. r- m7 @2 ]. @: v/ A len:缓冲的长度; U& H8 \7 j h4 t' P/ Y( V; r
flags:02 K) \. G0 Y8 N/ b, j' K" L
/ K7 \+ @1 I: g' _5 {4 r# N3 O9 \
8 sendto()
, ?: m8 X: H" U$ t2 ^6 i 头文件:; V. Z; z2 ?$ T3 O4 U& t+ G1 G
#include <sys/types.h>
- G; ~- \% b! r" L$ H #include <sys/socket.h>% b) X8 d6 ^! h4 _! g% u
函数原型:
# h' _: h/ `5 K9 u int sendto(int sockfd, const void *msg, int len, unsigned int flags, const struct sockaddr *to, int tolen);5 w, {$ A# m$ J0 _2 z1 f4 l
4 o" K/ a9 @9 m- b* p+ n8 J3 t 3 H% `+ V: M+ x+ V/ A# I
9 recvfrom()# R8 l1 P0 ]# z2 [0 U' _
头文件:
% R; R5 m& v+ t0 X0 l; t #include <sys/types.h>
' X0 {) D4 g; o" y4 b #include <sys/socket.h>
; I. l& o7 y4 s& E( I% Z, V9 l" ^% l 函数原型:
5 ?+ f9 J& s6 U8 |/ r+ e int recvfrom(int sockfd, void *buf, int len, unsigned int flags, struct sockaddr *from, int fromlen)
; Y# t- B( V8 C2 p9 t3 n8 L* [* ?& G I( J& S3 c
* Q1 w+ Y8 k) \+ y" O0 ]$ k 10 read() write()3 d' ^5 p1 l" m* C6 X; B0 m
int read(int fd, char *buf, int len)
* g* L0 s& w2 ]$ Z# W# b" { int write(int fd, char *buf, int len)
" i9 ~. B* i- M5 M 3 n; I2 `* b0 @, S/ V
11 shutdown()" g! v: z, \. Q+ y, N1 W
close(sockfd)7 ^$ r8 x# }( u6 F. D3 I. s5 @
int shutdown(int sockfd, int how)
/ N; G- ]0 s* A; ~# A3 b9 }$ a+ d" I C----------------------------------- [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 }2 x" [$ H5 y: y<sys/socket.h> //与套接字相关的函数声明和结构体定义,如socket()、bind()、connect()及struct sockaddr的定义等8 y/ R6 Y! h! b3 m% x
<sys/ioctl.h> //I/O控制操作相关的函数声明,如ioctl()1 l ^& h j1 d$ E
<stdlib.h> //某些结构体定义和宏定义,如EXIT_FAILURE、EXIT_SUCCESS等
. X1 o8 M# s' c. e/ Y<netdb.h> //某些结构体定义、宏定义和函数声明,如struct hostent、struct servent、gethostbyname()、gethostbyaddr()、herror()等
& h/ I8 h& k h" B' Q; l<arpa/inet.h> //某些函数声明,如inet_ntop()、inet_ntoa()等
, d- W5 Q! z _9 F. ?" N<netinet/in.h> //某些结构体声明、宏定义,如struct sockaddr_in、PROTO_ICMP、INADDR_ANY等 [size=13.3333px]------------------------------ [size=13.3333px]linux下socket编写常用头文件
' @5 }. v2 ?0 L5 z% K) U
- Q5 d1 a' K' _/ H0 V) R. W[size=13.3333px]#include <sys/socket.h> //connect,send,recv,setsockopt等
0 F7 U0 `3 R& a* a$ @#include <sys/types.h>
9 x0 _, h/ L7 j% O
7 I8 @$ N3 R- ~, I+ S% {#include <netinet/in.h> // sockaddr_in, "man 7 ip" ,htons
: W1 X9 J6 O1 k$ S2 R& t" \5 }3 ]#include <poll.h> //poll,pollfd
* ? j4 c A! I0 i' p#include <arpa/inet.h> //inet_addr,inet_aton% @1 R* F" M) A" X j# u. C' {5 D
#include <unistd.h> //read,write
& @4 K! N6 {6 h! G( ^, u#include <netdb.h> //gethostbyname
5 W' \3 k, u3 l9 o4 K; I
% U# n( Y1 u# ]' H$ c0 e#include <error.h> //perror
+ i, B9 _/ z/ P% O#include <stdio.h>
, `) v; i5 B0 ~#include <errno.h> //errno
/ m0 b/ {- B; ^' t
- R1 j+ p5 Y! I* H0 g#include <string.h> // memset/ S7 a5 V( ~/ }& ]- }
#include <string>
2 E( P9 C1 h% l- x7 [# e$ E1 Q#include <iostream>$ b% o- T! f4 X+ }
% U% F# M/ c6 e+ X0 m5 ]' G
|