|
[size=13.3333px]一 三种类型的套接字:
+ o7 j% Y0 H: U& x; E6 X1.流式套接字(SOCKET_STREAM)/ G( z, }2 a6 z5 t9 P, a
提供面向连接的可靠的数据传输服务。数据被看作是字节流,无长度限制。例如FTP协议就采用这种。
( u6 ]* r4 q! t) Z# e2 m2.数据报式套接字(SOCKET_DGRAM)
' X6 S j% G9 K: ?9 I 提供无连接的数据传输服务,不保证可靠性。
; l$ W1 p# J% e0 |6 g, c7 r1 }& P3.原始式套接字(SOCKET_RAW)
- m4 k! a9 A2 h. e. q4 | 该接口允许对较低层次协议,如IP,ICMP直接访问。, Q$ l" d+ l3 a( L3 `
- U* q. }6 H# H6 x+ {二 基本套接字系统调有有如下一些:$ k" w- E6 \2 g" N. V
创建套接字: socket()
) X+ ]: ^ B& q% P 绑定本机端口: bind()0 S/ Z3 u8 D! n( T2 f+ V" e+ p
建立连接: connect(),accept()8 B; y$ S5 \- q/ Y, w) R7 p0 F) V8 r
侦听端口: listen()$ y, d; [' {6 t* V3 n* G, E) ~
数据传输: send(), recv()9 W* x1 E" y; J5 R
输入/输出多路复用: select()
6 c% {6 o% f' b# X' K9 k 关闭套接只: closesocket()! C2 a' F6 L2 ]3 ~; O5 y% _
$ _" K2 V" _; e2 C: }) m
三 数据类型) H$ N# D# c* \6 g l) D4 L: m! t7 r `
struct sockaddr+ m: o6 R( n* s5 `( c5 a/ P8 Z
{
3 y8 E0 I- b6 w9 q& n unsigned short sa_family; //地址族, 一般为AF_INET
: g' s% U, i. `0 f+ i4 m6 `4 ] char sa_data[14]; //14字节的协议地址
" i h( B1 z$ D4 n } F }
1 |! ] k% }' c, Q3 W( u" @2 B. r4 v( E# z- H
struct sockaddr_in* X: V7 {6 x/ x( J( {5 w* I
{
" h l2 {6 S0 O7 e short int sin_family; //地址族
' o7 a3 |: c i unsigned short int sin_port; //端口号' k# X5 ?5 ~8 X# z; M/ O8 y
struct in_addr in_addr; //ip地址% o& i' P9 C7 [7 b
unsigned char sin_zero[8]; //填充" A% ]' ^. s; O$ }+ L: R- u( G
}8 b0 M2 L6 w- y/ X# r. Y. ]% t
8 {. c* |2 [: V' i5 V, }四 常用函数
; H1 R; k8 |* t: V) \ w! r 1 socket(): t6 M2 L* `0 E: N
头文件: 6 B- `; j p* P1 a. ^0 g' Q
#include <sys/types.h>
0 f* s+ G& o0 o- z: B' O0 P* I #include <sys/socket.h>
- ?& X" t4 d: i6 S. v 函数原型:
3 G& y9 m1 z$ G6 y9 q% R int socket(int domain, int type, int protocol)
1 k) H1 x. ^0 N domain: 协议类型,一般为AF_INET6 e& B) i) c9 x
type: socket类型' S5 N; b F0 F! Z8 k4 p
protocol:用来指定socket所使用的传输协议编号,通常设为0即可5 t+ N( A3 B. n$ M3 o2 [+ O
; ?. l- k, Q5 v
2 bind()/ H, i3 p( b+ K
头文件:
8 _. y9 c$ r5 c9 _ #include <sys/types.h>3 I9 {' {' D% l
#include <sys/socket.h>
4 W8 q: Y# \3 W- I2 ] 函数原型:
" ?' v: T/ n% O2 v int bind(int sockfd, struct sockaddr *my_addr, int addrlen)& C: ~- \8 k" I7 Q
sockfd: socket描述符
- h4 g E) N$ K" y' I0 y my_addr:是一个指向包含有本机ip地址和端口号等信息的sockaddr类型的指针3 X* @% j* a8 X. n
addrlen:常被设为sizeof(struct sockaddr)
1 C( k' o7 j! E5 |. i; L- M/ [) p5 ?- L2 k
3 connect()
5 G! }9 v7 s$ F% u" M( ~ 头文件:
- [$ g" z4 Z! ^6 r p' v #include <sys/types.h> / `% g; A0 M" L, \
#include <sys/socket.h>; _# D- `5 C* e; Q0 A, b8 A& v) T
函数原型:( m. P/ F1 J0 z) E+ H# b; [* o4 [
int connect(int sockfd, struct sockaddr *serv_addr, int addrlen)) F, s0 r( [1 S& v: \2 m3 _4 n
sockfd: 目的服务器的socket描述符
7 |3 m# W. {$ J/ u+ D F: z8 G% Q! A, } serv_addr:包含目的机器ip地址和端口号的指针
8 X3 H4 J; F' h4 p2 |8 Z addrlen:sizeof(struct sockaddr)
8 ^) W* A, g: R# ]" y* t- Z# K1 g9 A/ w
4 listen()
( o+ O* Z* s: R' X, u3 r 头文件:
* T0 s5 i/ g+ ~% A/ i #include <sys/socket.h>( E$ z9 `" m& O6 M
函数原型:
8 d5 S. u2 A# W' q# h2 N8 n int listen(int sockfd, int backlog);
) k3 p/ k! n+ n4 Q1 d2 Z) z) c sockfd:socket()系统调用返回的socket描述符
2 v% b+ y' Q8 R: c) `9 `0 e backlog:指定在请求队列中的最大请求数,进入的连接请求将在队列中等待accept()它们。& X/ y; q* d: o5 S# E/ j: @
$ a0 z: \7 m' f- H$ t$ z! D8 S
5 accept()
+ K$ C% Y8 r& e 头文件:
. W2 c/ b6 Y% ^& u #include <sys/types.h>& V9 G1 V8 f1 Z! \* d6 i$ D
#inlcude <sys/socket.h>! j) A8 l! e- `, s n$ Y9 E' O
函数原型:. K6 V' L( u- |% c
int accept(int sockfd, void *addr, int addrlen)
& C: n% ]8 Z- M' R0 }) j sockfd:是被监听的socket描述符) g1 m4 o# \; K. y' n3 H9 y
addr:通常是一个指向sockaddr_in变量的指针,该变量用来存放提出连接请求服务的主机的信息 X+ A- ]* n2 U: [
addrlen:sizeof(struct sockaddr_in)3 j, d: @2 |6 A- I
. p% K( b2 N# q7 T( Z. J6 N
6 send()% c; m) D; C1 y+ L$ ~! H
头文件:+ r3 i; Q6 {# P4 \/ R
#include <sys/socket.h>7 z7 Y8 K, \6 {: e" u+ r
函数原型:, G8 y. Y# i9 h0 I
int send(int sockfd, const void *msg, int len, int flags);
1 |0 J" K0 D2 t* _/ y) c* q sockfd:用来传输数据的socket描述符
5 l1 ?* w1 O2 B0 z msg:要发送数据的指针 " T$ Z& Y6 |# }1 [9 \# s' C7 L! X# L# c
flags: 0
) G- Z1 M7 J: E' ^- H# v% y0 F 9 w7 a2 }) j% k
7 recv()
" a/ w) i# t( I/ Q 头文件:! x8 k: w8 z3 B, ?+ T! f
#include <sys/types.h>
- X: v( V3 K* E #include <sys/socket.h>
- O N6 f' V8 w 函数原型:, |3 z" b4 T/ \: v" m, x( a3 ?3 D
int recv(int sockfd, void *buf, int len, unsigned int flags)
% E7 c2 \7 w8 g sockfd:接收数据的socket描述符: Q6 s5 h& g+ O8 q
buf:存放数据的缓冲区/ w; B' q2 c# g6 |, J/ C- P
len:缓冲的长度
) g& G$ k4 b5 L7 Y2 d) L# f flags:0
5 I9 n, ?1 h9 f0 |( u7 D; c4 K6 r8 X. @" g* A- m" G7 k( q% y. i8 k
8 sendto()
+ u) ?# i8 [7 `4 v3 G 头文件:
: j9 d' @3 u4 u* b #include <sys/types.h>
6 i7 j# m. V- B; z# Y# v# C #include <sys/socket.h>% L x7 P; w$ B+ O
函数原型:
5 ^* ~$ t$ V# |1 \3 c: K4 z int sendto(int sockfd, const void *msg, int len, unsigned int flags, const struct sockaddr *to, int tolen);
+ ]5 @0 f; }9 l& d f , y! |* _6 V" Y, }
. _& @3 r5 y% n F$ t
9 recvfrom()
0 B9 g Z) z# Z9 G- B 头文件:0 K$ Z: D1 U# B2 D# Z
#include <sys/types.h>
* Q, P8 t i1 K- `% `( Y #include <sys/socket.h>
+ \2 N8 {- z0 Y" j, Z 函数原型:
, e( i: U, W6 J) A int recvfrom(int sockfd, void *buf, int len, unsigned int flags, struct sockaddr *from, int fromlen)
x: c2 w( V5 @! b4 R% R$ i4 \0 P4 [5 j4 D( Y" I- j4 I! @2 O8 d/ c
' u; s/ R& f7 Y- A4 B
10 read() write()1 L: N+ U4 f1 j+ R' i/ O% Y J
int read(int fd, char *buf, int len)8 o8 \7 H, z/ Q( ^# V* s/ \3 u
int write(int fd, char *buf, int len) f( n# T3 l4 Y5 i C) } P
% @8 D2 L$ k+ n 11 shutdown()# u- {6 ]. |3 p$ q: n
close(sockfd)
4 \ @1 |) N H; j5 @8 A' X# c int shutdown(int sockfd, int how)6 c# X m3 F+ f$ g: N* N) ~2 v5 e
----------------------------------- [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等) 9 h2 @8 g/ x! I5 j4 I
<sys/socket.h> //与套接字相关的函数声明和结构体定义,如socket()、bind()、connect()及struct sockaddr的定义等' U+ U% p: {( w+ P# q( V/ D
<sys/ioctl.h> //I/O控制操作相关的函数声明,如ioctl()3 {; f) K% Z* C- o+ V3 X. f5 s
<stdlib.h> //某些结构体定义和宏定义,如EXIT_FAILURE、EXIT_SUCCESS等7 [- `8 F6 R. G t2 h9 _
<netdb.h> //某些结构体定义、宏定义和函数声明,如struct hostent、struct servent、gethostbyname()、gethostbyaddr()、herror()等7 d+ g5 f1 e, E% O' @
<arpa/inet.h> //某些函数声明,如inet_ntop()、inet_ntoa()等
3 g2 q0 Q# Q5 j- e<netinet/in.h> //某些结构体声明、宏定义,如struct sockaddr_in、PROTO_ICMP、INADDR_ANY等 [size=13.3333px]------------------------------ [size=13.3333px]linux下socket编写常用头文件
$ N9 G% H, G! Y
' e/ Y; N$ {+ u# j4 D[size=13.3333px]#include <sys/socket.h> //connect,send,recv,setsockopt等
; t: D, H+ m4 V#include <sys/types.h> ; d( @+ {; u+ M$ M h
' l7 ^& C3 p3 P
#include <netinet/in.h> // sockaddr_in, "man 7 ip" ,htons
2 l! E8 X) E: Q3 s& S- ~3 b2 U#include <poll.h> //poll,pollfd2 y: h. \% |& d( W2 V: h: q; Z
#include <arpa/inet.h> //inet_addr,inet_aton
& h" ~ _2 ~2 O& `7 Y#include <unistd.h> //read,write
3 s# j) H& X% I) W; ?' @#include <netdb.h> //gethostbyname, r$ [/ e9 L& z, A- t
6 {* L% ?. @' N- H
#include <error.h> //perror
& E. l# @# l B+ k) u! v#include <stdio.h>- @# N7 M) y+ ?1 D8 G+ g( P. R( I
#include <errno.h> //errno E+ l% K! F6 D7 p) o
2 Q9 r/ [2 i O: {+ Y" ?; q
#include <string.h> // memset/ \6 H% e( G/ ~. b
#include <string>( X7 k; ]) \% B- K2 P
#include <iostream>* V8 t; k. ?: n& K
4 X8 }( v3 {$ m |