linux 网络编程-基础篇

#Socket简介

是一个编程接口
是一种特殊的文件描述符(everything in Unix is a file)
并不仅限于TCPIP协议
面向连接(Transmission Control Protocol - TCPIP)
无连接(User Datagram Protocol-UDP 和 Inter-Network Packet Exchange-IPX)

#Socket类型
流式套接字(SOCK_STREAM)
提供了一个面向连接,可靠的数据传输服务,数据无差错,无重复的发送且
按发送顺序接收。内设置流量控制,避免数据流淹没慢的接收方。数据被看作是
字节流,无长度限制
数据报套接字(SOCK_DGRAM)
提供无连接服务。数据包以独立数据包的形式被发送,不提供无差错保证,
数据可能丢失或重复,顺序发送,可能乱序接收
原始套接字(SOCK_RAW)
可以对较低层次协议如IP,ICMP直接访问

#IP地址
IP地址是Internet中主机的标识
Internet中的主机要与别的机器通信必须具有一个IP地址
IP地址为32位(IPV4)或者128位(IPV6)
每个数据包都必须携带目的IP地址和源IP地址,路由器依靠此信息为数据包
选择路由
表示形式:常用点分形式,如202.38.64.10,最后都会转换为一个32位的无符号
整数。

#IP地址的转换
inet_aton()
将strptr所指的字符串转换成32位的网络字节序二进制值
int inet_aton(const char *strptr, struct in_addr *addrptr);
inet_addr()
功能同上,返回转换后的地址
in_addr_t inet_addr(const char *strptr);
inet_ntoa()
将32位网络字节序二进制地址转换成点分十进制的字符串。
char *inet_ntoa(struct in_addr inaddr);

#端口号
为了区分一台主机接收到的数据包应该转交给哪个进程来进行处理,使用端口号
来区分
TCP端口号与UDP端口号独立(协议不同可以使用同一个端口)
三无组:协议,IP,端口
端口号一般由IANA(Internet Assigned Numbers Authority)管理
众所周知端口:1~1023(1~255之间为众所周知端口,256~1023端口通常由
UNIX系统占用
已登记端口:1024~49151
动态或私有端口:49152~65535

#字节序
不同类型CPU的主机中,内存存储多字节整数序列有两种方法,称为主机字节序(HBO):
小端序(little-endian)-低字节存储在低地址
将低字节存储在起始地址,称为"Little-Endian"字节序,Intel,AMD等
采用的是这种方式
大端序(big-endian)-高序字节存储在低地址
将高字节存储在起始地址,称为"Big-Endian"字节序,由ARM,Motorola
等所采用
网络中传输的数据必须按网络字节序,即大端字节序
在大部分PC机上,当应用进程将整数送入socket前,需要转化成网络字节序;当
应用进程从socket取出整数后,要转化成小端字节序

#字节序转换函数
把给定系统所采用的字节序称为主机字节序,为了避免不同类别主机之间在数据
交换时由于对于字节序的不用而导致的差错,引入了网络字节序
主机字节序到网络字节序:
u_long htonl(u_long hostlong);
u_short htons(u_short short);
网络字节序到主机字节序
u_long ntohl(u_long hostlong);
u_short ntohs(u_short short);

#网络编程相关API
int Socket(int domain, int type, int protocol);
int bind(int Sockfd, struct sockaddr *my_addr, int addrlen);
int listen(int sockfd, in backlog);
int accept(int sockfd, struct sockaddr *addr, Socklen_t *addrlen);
int connect(int sockfd, const struct sockaddr *addr, Socklen_t addrlen);
ssize_t send(int sockfd, const void *buf, size_t len, int flags);
ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,
        const struct sockaddr *dest_addr, Socklen_t addrlen);
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,
        struct sockaddr *src_addr, Socklen_t *addrlen);
int shutdown(int sockfd, int how);

#command:
netstat -anpt

#地址结构的一般用法
1.定义一个struct sockaddr_in类型的变量并清空
struct sockaddr_in myaddr;
memset(&myaddr, 0, sizeof(myaddr));
2.填充地址信息
myaddr.sin_family = PF_INET;
myaddr.sin_port = htons(8888);
myaddr.sin_addr.s_addr = inet_addr("192.168.1.100");
3.将该变量强制转换为struct sockaddr类型在函数中使用
bind(listenfd,(struct sockaddr *)&myaddr, sizeof(myaddr));

#地址转换函数
unsigned long inet_addr(const char *address);
int inet_aton(const char *cp, struct in_addr *inp);
char *inet_ntoa(struct in_addr in);

socket流程图:

 

 

附:代码

client.c

//int socket(int domain, int type, int protocol);
//int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
//int listen(int sockfd, int backlog);
//int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
//uint16_t htons(uint16_t hostshort);
//int atoi(const char *nptr);
//in_addr_t inet_addr(const char *cp);
//void bzero(void *s, size_t n);
//int listen(int sockfd, int backlog);
//int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
//int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

#include <stdio.h>
#include <sys/types.h>//socket():bind();listen():accept();listen();accept();connect();
#include <sys/socket.h>//socket():bind();listen():accept();inet_addr();listen():accept();connect();
#include <arpa/inet.h>//htons();inet_addr():
#include <netinet/in.h>//inet_addr():
#include <strings.h>//bzero();
#include <stdlib.h>//atoi();exit();
#include <unistd.h>//close():
#include <string.h>

#define N 64

int main(int argc, char *argv[])
{
    if(argc < 3)
    {
        printf("Using: %s ip port\n",argv[0]);
        exit(-1);
    }
    int sockfd;
    struct sockaddr_in myaddr,peeraddr;
    char buf[N] = {0};
    size_t n;
    socklen_t mylen,peerlen;
    mylen = sizeof(myaddr);
    peerlen = sizeof(peeraddr);
    /*  creat socket  */
    if(-1 == (sockfd = socket(AF_INET,SOCK_DGRAM,0)))
    {
        perror("socket");
        exit(-1);
    }
    /* bind initialization */
    bzero(&myaddr,sizeof(myaddr));
    myaddr.sin_family = AF_INET;
    myaddr.sin_port = htons(atoi(argv[2])+1);
    myaddr.sin_addr.s_addr = inet_addr(argv[1]);
    /*      bind      */
    if(-1 == bind(sockfd, (struct sockaddr *)&myaddr, mylen))
    {
        perror("bind");
        exit(-1);
    }
    /*    connect    */
    bzero(&peeraddr, sizeof(peeraddr));
    peeraddr.sin_family = AF_INET;
    peeraddr.sin_port = htons(atoi(argv[2]));
    peeraddr.sin_addr.s_addr = inet_addr(argv[1]);
    //connect(sockfd,(struct sockaddr *)&peeraddr, peerlen);
    while(1){
    fgets(buf, N, stdin);
    printf("buf:%s",buf);
    if(-1 == (n = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&peeraddr, peerlen)))
    {
        perror("sendto");
        exit(-1);
    }
    if(strncmp(buf, "quit", 4) == 0)
            break;
    printf("n = %d\n",n);
    }
    close(sockfd);
    exit(0);
}

server.c

#include <stdio.h>
#include <sys/types.h>//socket():bind();listen():accept();listen();accept();connect();
#include <sys/socket.h>//socket():bind();listen():accept();inet_addr();listen():accept();connect();
#include <arpa/inet.h>//htons();inet_addr():
#include <netinet/in.h>//inet_addr():
#include <strings.h>//bzero();
#include <stdlib.h>//atoi();exit();
#include <unistd.h>//close():
#include <string.h>

#define N 64

typedef struct sockaddr SA;

int main(int argc, char *argv[])
{
    if(argc < 3)
    {
        printf("Using: %s ip port\n",argv[0]);
        exit(-1);
    }

    int listenfd, connfd;
    struct sockaddr_in myaddr, peeraddr;
    char buf[N] = {0};
    size_t n;
    socklen_t mylen, peerlen;
    mylen = sizeof(myaddr);
    peerlen = sizeof(peeraddr);
    /*    create    socket    */
    if ((listenfd = (socket(PF_INET, SOCK_DGRAM, 0))) < 0)
    {
        perror("socket");
        exit(-1);
    }
    /*    bind initialization    */
    bzero(&myaddr, mylen);
    myaddr.sin_family = PF_INET;
    myaddr.sin_port = htons(atoi(argv[2]));
    inet_aton(argv[1], &myaddr.sin_addr);

    if(bind(listenfd,(SA *)&myaddr, sizeof(myaddr)) < 0)
    {
        perror("bind");
        exit(-1);
    }
    /*    listen        */
    listen(listenfd, 5);
    /*    accept        */
    if(-1 == (connfd = accept(listenfd, (SA *)&peeraddr, &peerlen)))
    {
        perror("accept");
        exit(-1);
    }
    printf("connect from %s: %d\n",inet_ntoa(peeraddr.sin_addr), ntohs(peeraddr.sin_port));
    /*    recv-send    */
    while(1){
    n = recv(connfd, buf, N, 0);
    buf[n] = 0;
    printf("n = %d---%s\n",n,buf);
    if(0 == strncmp(buf,"quit",4))    break;
    send(connfd, buf, strlen(buf), 0);
    }
    close(connfd);
    close(listenfd);
    exit(0);
}
时间: 2024-09-12 12:44:19

linux 网络编程-基础篇的相关文章

Linux网络编程一步一步学【转】

转自:http://blog.chinaunix.net/uid-10747583-id-297982.html Linux网络编程一步一步学+基础  原文地址:http://blogold.chinaunix.net/u1/48325/showart_413841.html ·Linux网络编程基础(一)·Linux网络编程基础(二)·Linux网络编程基础(三) • Linux网络编程一步一步学-简单客户端编写   • Linux网络编程一步一步学-绑定IP和端口 • Linux网络编程一步

Linux网络编程入门

(一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍 客户端和服务端          网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户端         在网络程序中,如果一个程序主动和外面的程序通信,那么我们把这个程序称为客户端程序. 比如我们使用ftp程序从另外一         个地方获取文件的时候,是我们的ftp程序主动同外面进行通信(获取文件), 所以这个地方我们的ftp程序就是客户端程序.  服务端        

处理Linux网络编程中的IP地址

 Linux网络服务能力非常强大,它的TCP/IP代码是最高级的.Linux的网络实现是模仿FreeBSD的,它支持FreeBSD的带有扩展的Sockets(套接字)和TCP/IP协议.它支持两个主机间的网络连接和Sockets通讯模型,实现了两种类型的Sockets:BSD Sockets和INET Sockets.它为不同的通信模型和服务质量提供了两种传输协议,即不可靠的.基于消息的UDP传输协议和可靠的.基于流的传输协议TCP,并且都是在IP网络协议上实现的.INET sockets是在以

Proxy源代码分析--谈谈如何学习linux网络编程

原创地址: http://fanqiang.chinaunix.net/a4/b7/20010810/1200001101.html   Linux是一个可靠性非常高的操作系统,但是所有用过Linux的朋友都会感觉到,Linux和Windows这样的"傻瓜"操作系统(这里丝毫没有贬低Windows的意思,相反这应该是Windows的优点)相比,后者无疑在易操作性上更胜一筹.但是为什么又有那么多的爱好者钟情于Linux呢,当然自由是最吸引人的一点,另外Linux强大的功能也是一个非常重要

linux网络编程libpcap获取网络接口信息失败

问题描述 linux网络编程libpcap获取网络接口信息失败 #include typedef u_int32_t in_addr_t; struct in_addr{ in_addr_t s_addr;}; void main(){ char error_content[PCAP_ERRBUF_SIZE]; struct in_addr net_ip_address; struct in_addr net_mask_address; char *net_interface; char *ne

Linux网络编程 epoll中EPOLLIN EPOLLOUT信号无法触发

问题描述 Linux网络编程 epoll中EPOLLIN EPOLLOUT信号无法触发 比如 if(events[i].events&EPOLLIN) { int bytes_read=read(m_sockfd,read_buf,READ_BUFFER_SIZE); } 这样会触发EPOLLIN 但如果把read()封装到比如service类的sread()函数中 if(events[i].events&EPOLLIN) { service.sread() } 便不会触发EPOLLIN事

Linux 网络编程 epoll中的EPOLLIN EPOLLOUT如何触发

问题描述 Linux 网络编程 epoll中的EPOLLIN EPOLLOUT如何触发 代码很长只截取关键部分 //服务器端 #include #include #include #include #include #include #include #include #include #include"locker.h" #include"threadpool.h" #include"http_conn.h" #include"htt

Linux 网络编程 之 TCP状态转换

                                               Linux 网络编程 之 TCP状态装换                                                                       从上面的图中可以看出,TCP共有11状态.由TCP发送和接收的数据有:ACK, FIN, SYN,RST.对于一个还未调用connect的client和未调用listen的server来说,它们都处于CLOSED状态.ACK

Linux网络编程socket选项之SO_LINGER,SO_REUSEADDR

                       Linux网络编程socket选项之SO_LINGER,SO_REUSEADDR Linux网络编程中,socket的选项很多.其中几个比较重要的选项有:SO_LINGER(仅仅适用于TCP,SCTP), SO_REUSEADDR.   SO_LINGER 在默认情况下,当调用close关闭socke的使用,close会立即返回,但是,如果send buffer中还有数据,系统会试着先把send buffer中的数据发送出去,然后close才返回.