服务器搭建(TCP套接字)-基础版(客户端)

一、socket

1.1、vim man查看socket

cpp 复制代码
:!man socket


1.2、 依赖的头文件

cpp 复制代码
 #include <sys/types.h>     
 #include <sys/socket.h>

1.3、原型

cpp 复制代码
int socket(int domain, int type, int protocol);
domain 说明
AF_INET IPV4协议
AF_INET6 IPV6协议
AF_LOCAL Unix域协议
type 说明
SOCK_STREAM 字节流套接字(TCP/SCTP)
SOCK_DGRAM 数据报套接字(UDP)
SOCK_RAM 原始套接字 (IPv4/IPv6)
SOCK_SEQPACKET 有序分组套接字(SCTP)
protocol 说明
IPPROTO_CP TCP传输协议
IPPROTO_UDP UDP传输协议
IPPROTO_SCTP SCTP传输协议

socket函数在成功时会返回一个小的非负整数值 ,它与文件描述符类似,就称它为套接字描述符。只需要指定协议族(IPV4、IPV6或Unix)和套接字类型即可

  • 代码实现
cpp 复制代码
socket_fd=socket(AF_INET,SOCK_STREAM,0);

二、connect

2.1、原型

cpp 复制代码
int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

connect() 函数用于在客户端应用程序中与服务器建立连接。需要注意的是,connect() 函数可能会阻塞,直到连接成功建立或发生错误。因此,您可以使用适当的超时机制或非阻塞 Socket 设置来控制连接过程的行为。

入参:

  • sockfd:要连接的客户端 Socket 的文件描述符。
  • addr:指向服务器地址结构的指针,其中包含服务器的 IP 地址和端口号。
  • addrlen:服务器地址结构的长度。

返回值:

  • 0 表示连接成功
  • 返回值为 -1 表示连接失败,并设置相应的错误代码(可以使用 perror() 函数打印错误信息)。

2.2、代码

cpp 复制代码
#define SERVER_PORT 8596 
#define MESSAGE_LENGTH 1024

int ret = -1;
int socket_fd;
//server addr
struct sockaddr_in serverAddr;

serverAddr.sin_family = AF_INET;
serverAddr.sin_port = htons(SERVER_PORT);
//inet_addr()函数,将点分十进制IP转换成网络字节序IP
serverAddr.sin_addr.s_addr = inet_addr(args[1]);
if(connect(socket_fd, (struct sockaddr *)&serverAddr, sizeof(serverAddr)) < 0){
    perror("connect error");
    return 1;
}

三、send

3.1、原型

cpp 复制代码
ssize_t send(int sockfd, const void *buf, size_t len, int flags);

send 是一个在网络编程中使用的函数,用于发送数据到已连接的套接字或发送数据报套接字中。

入参:

  • sockfd:整型参数,表示要发送数据的套接字的文件描述符。
  • buf:指向要发送数据的缓冲区的指针。
  • len:无符号整型参数,表示要发送数据的长度。
  • flags:整型参数,用于指定发送操作的可选标志。常见的标志包括 0(无特殊标志)和 MSG_DONTWAIT(非阻塞模式发送)等。

返回值

  • 如果成功发送数据,send 返回发送的字节数。
  • 如果连接关闭(对于流套接字)或发送超时(对于数据报套接字),send 返回 0。
  • 如果发生错误,send 返回 -1,并设置相应的错误码,可以通过 errno 全局变量获取具体的错误信息。

3.2、源码

cpp 复制代码
#define MESSAGE_LENGTH 1024

char sendbuf[MESSAGE_LENGTH];

 while(1)
  {
    memset(sendbuf, 0, MESSAGE_LENGTH);

    printf("send message:");

    fgets(sendbuf,MESSAGE_LENGTH,stdin);
    //printf("\n");
    ret = send(socket_fd, sendbuf, strlen(sendbuf), 0);
    if(ret <= 0 ){
      printf("the connection is disconnection!\n");
      break;
    }

  }

四、recv

4.1、原型

cpp 复制代码
ssize_t recv(int sockfd, void *buf, size_t len, int flags);

recv 是一个在网络编程中使用的函数,用于接收数据从已连接的套接字或接收数据报套接字中。

需要注意的是,recv 函数是一个阻塞调用,即在没有数据到达之前,它会一直等待。如果需要非阻塞的操作,可以使用非阻塞 I/O 或多线程/多进程的方式处理接收操作。

入参:

  • sockfd:整型参数,表示要接收数据的套接字的文件描述符。
  • buf:指向接收数据缓冲区的指针,用于存储接收到的数据。
  • len:无符号整型参数,表示接收数据缓冲区的长度。
  • flags:整型参数,用于指定接收操作的可选标志。常见的标志包括 0(无特殊标志)、MSG_DONTWAIT(非阻塞模式接收)和 MSG_WAITALL(阻塞模式接收,直到接收到指定长度的数据)等。

返回值:

  • 如果成功接收数据,recv 返回接收到的字节数。
  • 如果连接关闭(对于流套接字)或接收超时(对于数据报套接字),recv 返回 0。
  • 如果发生错误,recv 返回 -1,并设置相应的错误码,可以通过 errno 全局变量获取具体的错误信息。

五、完整代码

cpp 复制代码
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>

#include <errno.h>
#include <netdb.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define SERVER_PORT 8596 
#define MESSAGE_LENGTH 1024

int main(int argc,char* args[])
{

  int ret = -1;
  int socket_fd;

  //server addr
  struct sockaddr_in serverAddr;

  char sendbuf[MESSAGE_LENGTH];
  char recvbuf[MESSAGE_LENGTH];

  int data_len;

  if((socket_fd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
  {
    perror("socket");
    return 1;
  }

  serverAddr.sin_family = AF_INET;
  serverAddr.sin_port = htons(SERVER_PORT);

  //inet_addr()函数,将点分十进制IP转换成网络字节序IP
  serverAddr.sin_addr.s_addr = inet_addr(args[1]);

  if(connect(socket_fd, (struct sockaddr *)&serverAddr, sizeof(serverAddr)) < 0)
  {
    perror("connect");
    return 1;
  }

  printf("success to connect server...\n");

  while(1)
  {
    memset(sendbuf, 0, MESSAGE_LENGTH);

    printf("send message:");

    fgets(sendbuf,MESSAGE_LENGTH,stdin);
    //printf("\n");
    ret = send(socket_fd, sendbuf, strlen(sendbuf), 0);
    if(ret <= 0 ){
      printf("the connection is disconnection!\n");
      break;
    }

    if(strcmp(sendbuf, "quit") == 0){
      break;
    }

    printf("receive message:");

    recvbuf[0] = '\0';
    data_len = recv(socket_fd, recvbuf, MESSAGE_LENGTH, 0);

    recvbuf[data_len] = '\0';

    printf("%s\n", recvbuf);

  }

  close(socket_fd);

  return 0;

}
相关推荐
鸠摩智首席音效师12 分钟前
.NET Core 应用程序如何在 Linux 中创建 Systemd 服务 ?
linux·运维·.netcore
汤米粥32 分钟前
通过PHP创建AWS的CloudFront并绑定证书添加备用域名
服务器·云计算·aws
Aiden_SHU1 小时前
Wireshark中的length栏位
服务器·网络·wireshark
清风fu杨柳1 小时前
麒麟服务器工作站SP1 arm环境qt5.6.3源码编译
服务器·arm开发·qt
叫我龙翔1 小时前
【计网】实现reactor反应堆模型 --- 多线程方案优化 ,OTOL方案
linux·运维·网络
mit6.8241 小时前
[Docker#9] 存储卷 | Volume、Bind、Tmpfs | -v/mount | MySQL 灾难恢复 | 问题
linux·运维·docker·容器·架构
Z1eaf_complete1 小时前
Docker的基础使用
运维·docker·容器·云计算
群联云防护小杜2 小时前
服务器被挂马怎么办?——解决服务器被挂马的方法和步骤
运维·服务器·网络协议·tcp/ip·安全·ddos
明志致远淡泊宁静2 小时前
记录一次服务器redis被入侵
运维·服务器·redis
Jtti2 小时前
Jtti:服务器总是自动重启怎么办?
运维·服务器