Socket基础编程

地址结构sockaddr_in

其中包含:IP地址端口号协议族
推荐使用sockaddr_in,而不建议使用sockaddr
sockaddr_insockaddr是等价的,但sockaddr_in字段更清晰

/*
 * Socket address, internet style.
 */
struct sockaddr_in {
    __uint8_t   sin_len;
    sa_family_t sin_family;
    in_port_t   sin_port;
    struct      in_addr sin_addr;
    char        sin_zero[8];
};

sockaddr_in字段描述

// 地址家族,通常使用AF_INET代表TCP/CP协议族(AF_INET6代表IPV6)
sockaddr_in.sin_family

// 端口号,必须转化为网络字节序使用htons和ntohs
sockaddr_in.sin_port

// 用于存储ip地址,必须是网络字节序
sockaddr_in.sin_addr

// 为了让sockaddr与sockaddr_in两个数据结构保持大小相同而保留的空字节
sockaddr_in.sin_zero

sockaddr_in字段初始化

struct sockaddr_in servaddr;
bzero(&servaddr, sizeof(servaddr));

servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(8090);

// 若字符串有效则将字符串转换为32位二进制网络字节序的IPV4地址
servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
// 检测转化IP是否出错
if (servaddr.sin_addr.s_addr == INADDR_NONE) {
    perror("ip");
    exit(-1);
}

用到的函数

// 将网络字节序转化为主机字节序
__uint16_t ntohs(__uint16_t);
// 将主机字节序转化为网络字节序
__uint16_t htons(__uint16_t);

__uint32_t ntohl(__uint32_t);
__uint32_t htonl(__uint32_t);
// 若字符串有效则将字符串转换为32位二进制网络字节序的IPV4地址,否则为INADDR_NONE
// 其反函数inet_ntoa
in_addr_t inet_addr(const char* strptr);

可以支持IPV6的IP地址转换方式

// inet_pton能够处理ipv4甚至ipv6(需要改动现在代码)
if (inet_pton(AF_INET, "127.0.0.1", &servaddr.sin_addr) <= 0) {
    perror("pton");
    exit(-1);
}

套接字

创建一个套接字,返回一个套接字描述符

int socket(int domain, int type, int protocol);

套接字描述符(socket file descriptor)
套接字描述符是一个整数类型的值,每个进程的进程空间里都有一个套接字描述符表,表中存放着套接字描述符和套接字数据结构的对应关系。
因此根据套接字描述符就可以找到其对应的套接字数据结构。
每个进程在自己的进程空间里都有一个套接字描述符表,但是套接字数据结构都是在操作系统的内核缓冲里

接收发送函数

ssize_t recv(int sockfd, void *buff, size_t nbytes, int flags);
ssize_t send(int sockfd, const void *buff, size_t nbytes, int flags);

ssize_t write (int fd,const void * buf,size_t count);
ssize_t read(int fd,void * buf ,size_t count);

客户端主要函数

int connect (int sockfd,struct sockaddr * serv_addr,int addrlen);

服务器端主要函数

int bind( int sockfd , const struct sockaddr * my_addr, socklen_t addrlen);
int listen(int s, int backlog);
int accept(int s, struct sockaddr * addr, int * addrlen);

客户端代码

#include <iostream>

#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <arpa/inet.h>
#include <unistd.h>

const int       MAX_LINE    = 512;
const ushort    SERV_PORT   = 7890;
const char*     SERV_IP     = "127.0.0.1";

int main(int argc, const char * argv[])
{
    int socketfd;

    struct sockaddr_in servaddr;
    memset(&servaddr, 0, sizeof(servaddr));

    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(SERV_PORT);

    // 若字符串有效则将字符串转换为32位二进制网络字节序的IPV4地址
//    servaddr.sin_addr.s_addr = inet_addr(SERV_IP);
//    if (servaddr.sin_addr.s_addr == INADDR_NONE) {
//        perror("ip");
//        exit(-1);
//    }

    // 支持IPV6的新的转换方式,推荐
    if (inet_pton(AF_INET, "127.0.0.1", &servaddr.sin_addr) <= 0) {
        perror("pton");
        exit(-1);
    }

    // 创建一个套接字
    if ( (socketfd = socket(AF_INET, SOCK_STREAM, 0)) < 0 ) {
        perror("socket");
        exit(-1);
    }

    // 连接到主机
    if (connect(socketfd, (sockaddr*)&servaddr, sizeof(servaddr)) < 0) {
        perror("connect");
        exit(-1);
    }

    // 接收数据
    char recvline[MAX_LINE+1];
    long bytelen = 0;
    while ((bytelen = read(socketfd, recvline, MAX_LINE)) > 0) {
        recvline[bytelen] = 0;
        printf("recv --> %s\n", recvline);
    }

    close(socketfd);
    return 0;
}

服务器代码

#include <iostream>

#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <arpa/inet.h>
#include <unistd.h>

const ushort SERV_PORT   = 7890;

int main(int argc, const char * argv[])
{
    struct sockaddr_in serveraddr;
    memset(&serveraddr, 0, sizeof(serveraddr));

    serveraddr.sin_family = AF_INET;
    serveraddr.sin_port = htons(SERV_PORT);
    // 系统自动获取本机IP
    serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);

    // 创建一个socket
    int serverfd = socket(AF_INET,SOCK_STREAM,0);
    if (serverfd < 0) {
        perror("socket");
        exit(-1);
    }

    // 绑定端口
    int bind_ret = bind(serverfd, (struct sockaddr *)&serveraddr, sizeof(serveraddr));
    if (bind_ret < 0) {
        perror("bind");
        exit(-1);
    }

    // 开始监听
    // #2: backlog:等待连接队列的最大长度
    if (listen(serverfd, 10) < 0) {
        perror("listen");
        exit(-1);
    }

    printf("---------- listening (127.0.0.1:%d) ----------\n", SERV_PORT);

    // 开始轮询监听请求
    bool toggle = true;
    while (toggle) {
        // 客户端信息结构
        int clientfd;
        struct sockaddr_in clientaddr;
        memset(&serveraddr, 0, sizeof(serveraddr));
        socklen_t len = sizeof(clientaddr);

        // 接收请求,建立连接
        if ( (clientfd = accept(serverfd, (struct sockaddr *)&clientaddr, &len)) < 0 ) {
            perror("accept");
            exit(-1);
        }

        // 向客户端发送信息
        const char *message = "hello world";
        write(clientfd, message, strlen(message)+1);

        // 转为点分十进制ip
        const char *client_ip = inet_ntoa(clientaddr.sin_addr);
        const ushort client_port = ntohs(clientaddr.sin_port);
        printf("client(%s:%u) # send -> %s\n", client_ip, client_port, message);

        // 关闭与客户端的连接
        close(clientfd);
    }

    // 关闭服务监听
    close(serverfd);
    return 0;
}
时间: 2025-01-12 00:17:05

Socket基础编程的相关文章

【Socket】Java Socket基础编程

Socket是Java网络编程的基础,了解还是有好处的, 这篇文章主要讲解Socket的基础编程.Socket用在哪呢,主要用在进程间,网络间通信.本篇比较长,特别做了个目录: 一.Socket通信基本示例 二.消息通信优化 2.1 双向通信,发送消息并接受消息 2.2 使用场景 2.3 如何告知对方已发送完命令 2.3.1 通过Socket关闭 2.3.2 通过Socket关闭输出流的方式 2.3.3 通过约定符号 2.3.4 通过指定长度 三.服务端优化 3.1 服务端并发处理能力 3.2

socket基础编程-1

server端和client端 1.server端: import socket server=socket.socket() server.bind(('localhost',8080)) server.listen() conn,attr=server.connect() data=conn.recv(10240) re=b'.....' conn.send(re) server.close 2.client端 import socket client=socket.socket() cli

Linux应用程序设计之网络基础编程

1.TCP/IP协议概述 1.1.OSI参考模型及TCP/IP参考模型 OSI协议参考模型是基于国际标准化组织(ISO)的建议发展起来的,从上到下工分为7层:应用层,表示层,会话层,传输层,网络层,数据链路层,物理层.与此相区别的TCP/IP协议模型一开始就遵循简单明确的设计思路,它将OSI的7层参考模型简化为4层,从而得到有利于实现和使用.TCP/IP协议参考模型和OSI协议参考模型的对应关系如下图所示: 网络接口层:负责将二进制流转换为数据帧,并进行数据帧的发送和接收.要注意的是数据帧是独立

Unix网络编程 之 socket基础

基本结构 (这部分的地址均为网络地址<网络字节序>) 1.struct sockaddr:通用套接字地址结构 此结构用于存储通用套接字地址. 数据结构定义: typedef unsigned short sa_family_t; struct sockaddr { sa_family_t sa_family; /* address family, AF_xxx */ char sa_data[14]; /* 14 bytes of protocol address */ };    sa_fa

Android网络编程(一)---Socket基础

Socket通常称为""套接字.Socket字面上的中文意思为"插座".一台服务器可能会提供很多服务,每种服务对应一个Socket,而客户的"插头"也是一个Socket.Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口.Socket把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议.Socket用于描述IP地址和端口,是一个通信链的句柄.应

嵌入式 Linux网络编程(一)——Socket网络编程基础

嵌入式 Linux网络编程一--Socket网络编程基础 一.Socket简介 1.网络中进程间通信 本机进程使用进程号区别不同的进程进程间通信方式有管道.信号.消息队列.共享内存.信号量等.网络中进程间的通信首先需要识别进程所在主机在网络中的唯一标识即网络层的IP地址主机上的进程可以通过传输层的协议与端口号识别. 2.Socket原理 Socket是应用层与TCP/IP协议族通信的中间软件抽象层是一种编程接口.Socket屏蔽了不同网络协议的差异支持面向连接(Transmission Cont

Linux程序设计学习笔记----Socket网络编程基础之TCP/IP协议簇

转载请注明出处: ,谢谢! 内容提要 本节主要学习网络通信基础,主要涉及的内容是: TCP/IP协议簇基础:两个模型 IPv4协议基础:IP地址分类与表示,子网掩码等 IP地址转换:点分十进制\二进制 TCP/IP协议簇基础 OSI模型 我们知道计算机网络之中,有各种各样的设备,那么如何实现这些设备的通信呢? 显然是通过标准的通讯协议,但是,整个网络连接的过程相当复杂,包括硬件.软件数据封包与应用程序的互相链接等等,如果想要写一支将联网全部功能都串连在一块的程序,那么当某个小环节出现问题时,整只

Java基础篇Socket网络编程中的应用实例

说到java网络通讯章节的内容,刚入门的学员可能会感到比较头疼,应为Socket通信中一定会伴随有IO流的操作,当然对IO流比较熟练的哥们会觉得这是比较好玩的一章,因为一切都在他们的掌握之中,这样操作起来就显得非常得心应手,但是对于IO本来就不是多熟悉的哥们来说就有一定的困难了,在搞清楚IO流操作机制的同时还必须会应用到Socket通信中去,否则会对得到的结果感到非常郁闷和懊恼,下面就和大家一起分享一下自己遇到一点小麻烦后的感触以及给出的解决办法. 要求:客户端通过Socket通信技术上传本地一

socket网络编程的一些基础知识

源地址:http://blog.csdn.net/roger_77/article/details/1453049 目录: 1) 什么是套接字? 2) Internet 套接字的两种类型 3) 网络理论 4) 结构体 5) 本机转换 6) IP 地址和如何处理它们 7) socket()函数 8) bind()函数 9) connect()函数 10) listen()函数 11) accept()函数 12) send()和recv()函数 13) sendto()和recvfrom()函数