新手socket编程入门详解指南

开发环境

运行平台:Ubantu 14.04 LTS

疑问引导

问题1:头文件的疑问:

#include <sys/socket.h>与#include <linux/socket.h>有何区别?

解答:

1. 使用diff查看:adc分别表示添加、删除、修改

2. 其实是路径的不同导致有不同的socke.h文件

3. <sys/socket.h> 是 Internet Protocol family,也就是tcpip协议的应用层接口

4. <linux/socket.h>目前暂时未弄懂,但不是接口函数,估计是系统函数。它应该是被操作系统使用,猜测该文件在tcpip的传输层

问题2:大小端字节序问题:

1.c语言检测:利用指针取值和取址的交叉应用,为了增强网络移植性

2. 而socket提供了字节序转换函数:h:host;n:network;l:long32位;s:short16位

3. htonl:将主机的32位主机字节序(ip地址),转换为网络字节序(一列数据)。

问题3:就一个服务器、一个客户端来说,有如下的对应角色说法:

对象first 对象second
服务器 客户端
监听者 广播者
提供服务 请求服务

解析socket编程整体过程:
建立与删除
服务器和客户端通过同一的socket信道通信,而创建一个socket信道,提供socket连接。

int socket(int domain,int type,int protocol);
domain(域):各个域以AF_XXX命令,意指地址族。决定使用何种的地址类型,确定通信特性:包括地址格式
type:确定套字节的类型,(还)可以自由增加类型。
常用:SOCK_STREAM (即:TCP)和 SOCK_DGRAM(即:UDP)
protocol:指定socket使用的传输协议编号,一般直接设置为0即可,以此表示为给定的域和套接字类型选择默认的传输协议。
返回值:正确返回套接字处理代码(我称之为套接字文件描述符),错误返回-1。该数值将存储使用。

服务器和客户端通都可以,关闭socket通信IO

int shutdown(int s,int how);
s:代表socket_fd,需要关闭的套接字文件描述符
how:为一种方式
shutdown是使socket信道处于不活动状态。可以让该信道关闭写端,而继续接收该套接字读端以此确定数据何时结束,然后再使用close来关闭这个信道。

连接关系

创建和销毁或关闭IO之后,需要知道如何标识一个目标通信进程。
原因:网络有多个计算机,某台计算机上运行着多个程序(进程)。下面是两层关系:
1)目标计算机的网络地址
2)目标计算机上的目标进程的所代表的端口号

所以,目前你需要了解到的有下面几点:

1. 字节序:直接看上面的问题2即可,简单的转换关系。
 2. 地址格式:根据不同的因特网地址,在<netinet/in.h>定义不同的结构体,部分socket函数参数调用。如下   
 3. 定义地址结构体,根据实际装入数值作为socket API实参   
 4. 地址进制转换:对地址进行二进制与文本字符串格式之间的转换。inet_ntop或inet_pton

绑定接着,对于服务端来说,需要绑定(关联)地址和套接字。为给定的sockfd关联一个sockaddr结构数据。只有服务端将套接字绑定在(域)地址上,客户端才能够连接(connect)成功。

int bind(int sockfd,struct sockaddr * my_addr,int addrlen);
sockfd:套接字文件描述符,是socket返回的值
my_addr:(服务器)网络地址信息
返回值:判断是否正确绑定地址和套接字

连接在此之前,我们创建了套接字(socket)、建立连接基础(bind)。那么,就这就是为了在通信之前,将socket信道连接起来。

int connect (int sockfd,struct sockaddr * serv_addr,int addrlen);
sockfd:套接字文件描述符,是socket返回的值
serv_addr :网络地址信息
返回值:判断是否正确连接,客户端程序必须要能够处理connect返回的错误。

到目前,你或许已经发现了,connect函数的参数类型与个数都跟bind是一样的(他们的值并不一样,我所说的是形式),结合一起去理解,会更好。
毕竟,根据TCPIP协议,需要连接的信息:IP地址,端口号,就已经足够了。至于其余的MAC地址等等,在socket里面,我们不需要理会。

监听需要注意的是,这种连接,服务器还需要确定是哪个客户端请求连接。所以,服务器首先进入运行请求客户端(任意一个)连接的状态,进入listen(监听)状态。使用函数:

int listen(int s,int backlog);
s:服务器套接字描述符,是socket返回的值
backlog:指定同时能够处理的最大连接要求
函数返回值:是否正确进入监听状态

连接这时候,服务器已经进入了listen状态,然后紧接着调用:

int accept(int s,struct sockaddr * addr,int * addrlen);
s:服务器套接字描述符,是socket返回的值
addr:某一被连接的客户端的套接字数据
addrlen:某一被连接的客户端的套接字数据长度
返回:某一被连接的客户端的文件描述符

读取与发送数据

到目前为止,服务器和客户端都已经做好了双向通信的基础准备。
send与recv暂时不提及,读者自己去查API

int recv(int s,void *buf,int len,unsigned int flags);
int send(int s,const void * msg,int len,unsigned int falgs);

以下直接与代码相关:

//常用包含头文件 and socket编程的作用
#include <stdio.h> //
#include <stdlib.h> //
#include <errno.h> //errno错误信息变量
#include <unistd.h> //
#include <stddef.h> //

#include <sys/socket.h> //提供socket API
#include <sys/un.h> //
#include <sys/types.h> //socket API参数的类型定义文件
#include <arpa/inet.h> //地址转换函数
#include <netinet/in.h> //字节序函数(宏)、域地址类型定义

客户端代码:

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <netinet/in.h>
#include <pthread.h>

#include "wrap.h"

#define MAXLINE 80
#define SERV_PORT 8000

int main(int argc, char *argv[])
{
 struct sockaddr_in servaddr;
 char buf[MAXLINE];
 int sockfd, n;
 sockfd = Socket(AF_INET, SOCK_STREAM, 0);
 bzero(&servaddr, sizeof(servaddr));
 servaddr.sin_family = AF_INET;
 inet_pton(AF_INET, "192.168.191.6", &servaddr.sin_addr);
 servaddr.sin_port = htons(SERV_PORT);
 Connect(sockfd, (struct sockaddr *) &servaddr, sizeof(servaddr));
 while (fgets(buf, MAXLINE, stdin) != NULL) {
 Write(sockfd, buf, strlen(buf));
 n = Read(sockfd, buf, MAXLINE);
 if (n == 0)
 printf("the other side has been closed.\n");
 else
 Write(STDOUT_FILENO, buf, n);
 }
 Close(sockfd);
 return 0;
}

服务器代码:

#include <stdio.h>
#include <string.h>
#include <netinet/in.h>
#include "wrap.h"

#define MAXLINE 80
#define SERV_PORT 8000

int main(void)
{
 struct sockaddr_in servaddr, cliaddr;
 socklen_t cliaddr_len;
 int listenfd, connfd;
 char buf[MAXLINE];
 char str[INET_ADDRSTRLEN];
 int i, n;

 listenfd = Socket(AF_INET, SOCK_STREAM, 0);

 bzero(&servaddr, sizeof(servaddr));
 servaddr.sin_family = AF_INET;
 servaddr.sin_addr.s_addr = inet_addr("192.168.191.6");
 servaddr.sin_port = htons(SERV_PORT);

 Bind(listenfd, (struct sockaddr *) &servaddr, sizeof(servaddr));
 Listen(listenfd, 20);
 printf("Accepting connections ...\n");

 while (1) {
 cliaddr_len = sizeof(cliaddr);
 connfd = Accept(listenfd,
 (struct sockaddr *) &cliaddr,
 &cliaddr_len);
 while (1) {
 n = Read(connfd, buf, MAXLINE);
 if (n == 0) {
 printf("the other side has been closed.\n");
 break;
 }
 printf("received from %s at PORT %d\n",
 (char *)inet_ntop(AF_INET,
 &cliaddr.sin_addr, str,
 sizeof(str)),
 (int)ntohs(cliaddr.sin_port));
 for (i = 0; i < n; i++)
 buf[i] = toupper(buf[i]);
 Write(connfd, buf, n);
 }
 Close(connfd);
 }
}

运行结果:

hhc@my:~/sharefile/socket/tcp$ ./server &
[1] 15371
hhc@my:~/sharefile/socket/tcp$ Accepting connections ...
hhc@my:~/sharefile/socket/tcp$ ./client
this is a test!
received from 192.168.191.6 at PORT 53685
THIS IS A TEST!

个人封装的socket接口函数:

#include "wrap.h"

void perr_exit(const char *s)
{
 perror(s);
 exit(1);
}

int Accept(int fd, struct sockaddr *sa, socklen_t * salenptr)
{
 int n;
 again:
 if ((n = accept(fd, sa, salenptr)) < 0) {
 if ((errno == ECONNABORTED) || (errno == EINTR))
 goto again;
 else
 perr_exit("accept error");
 }
 return n;
}

void Bind(int fd, const struct sockaddr *sa, socklen_t salen)
{
 if (bind(fd, sa, salen) < 0)
 perr_exit("bind error");
}

void Connect(int fd, const struct sockaddr *sa, socklen_t salen)
{
 if (connect(fd, sa, salen) < 0)
 perr_exit("connect error");
}

void Listen(int fd, int backlog)
{
 if (listen(fd, backlog) < 0)
 perr_exit("listen error");
}

int Socket(int family, int type, int protocol)
{
 int n;
 if ((n = socket(family, type, protocol)) < 0)
 perr_exit("socket error");
 return n;
}

ssize_t Read(int fd, void *ptr, size_t nbytes)
{
 ssize_t n;
 again:
 if ((n = read(fd, ptr, nbytes)) == -1) {
 if (errno == EINTR)
 goto again;
 else
 return -1;
 }
 return n;
}

ssize_t Write(int fd, const void *ptr, size_t nbytes)
{
 ssize_t n;
 again:
 if ((n = write(fd, ptr, nbytes)) == -1) {
 if (errno == EINTR)
 goto again;
 else
 return -1;
 }
 return n;
}

void Close(int fd)
{
 if (close(fd) == -1)
 perr_exit("close error");
}

ssize_t Readn(int fd, void *vptr, size_t n)
{
 size_t nleft;
 ssize_t nread;
 char *ptr;
 ptr = vptr;
 nleft = n;
 while (nleft > 0) {
 if ((nread = read(fd, ptr, nleft)) < 0) {
 if (errno == EINTR)
 nread = 0;
 else
 return -1;
 } else if (nread == 0)
 break;
 nleft -= nread;
 ptr += nread;
 }
 return n - nleft;
}

ssize_t Writen(int fd, const void *vptr, size_t n)
{
 size_t nleft;
 ssize_t nwritten;
 const char *ptr;
 ptr = vptr;
 nleft = n;
 while (nleft > 0) {
 if ((nwritten = write(fd, ptr, nleft)) <= 0) {
 if (nwritten < 0 && errno == EINTR)
 nwritten = 0;
 else
 return -1;
 }
 nleft -= nwritten;
 ptr += nwritten;
 }
 return n;
}

ssize_t my_read(int fd, char *ptr)
{
 static int read_cnt;
 static char *read_ptr;
 static char read_buf[100];
 if (read_cnt <= 0) {
 again:
 if ((read_cnt = read(fd, read_buf, sizeof(read_buf))) < 0) {
 if (errno == EINTR)
 goto again;
 return -1;
 } else if (read_cnt == 0)
 return 0;
 read_ptr = read_buf;
 }
 read_cnt--;
 *ptr = *read_ptr++;
 return 1;
}

ssize_t Readline(int fd, void *vptr, size_t maxlen)
{
 ssize_t n, rc;
 char c, *ptr;
 ptr = vptr;
 for (n = (ssize_t)1; n < (ssize_t)maxlen; n++) {
 if ((rc = my_read(fd, &c)) == 1) {
 *ptr++ = c;
 if (c == '\n')
 break;
 } else if (rc == 0) {
 *ptr = 0;
 return n - 1;
 } else
 return -1;
 }
 *ptr = 0;
 return n;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • java实现Socket通信之单线程服务

    前言 使用基于TCP 协议的双向通信时,网络中的两个应用程序之间必须首先建立一个连接,这两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个socket. 建立网络通信连接至少要一对端口号(socket).socket本质是编程接口(API),对TCP/IP的封装,TCP/IP也要提供可供程序员做网络开发所用的接口,这就是Socket编程接口:HTTP是轿车,提供了封装或者显示数据的具体形式;Socket是发动机,提供了网络通信的能力. Socket的英文原义是"孔"或

  • Java多线程编程实现socket通信示例代码

    流传于网络上有关Java多线程通信的编程实例有很多,这一篇还算比较不错,代码可用.下面看看具体内容. TCP是Tranfer Control Protocol的 简称,是一种面向连接的保证可靠传输的协议.通过TCP协议传输,得到的是一个顺序的无差错的数据流.发送方和接收方的成对的两个socket之间必须建 立连接,以便在TCP协议的基础上进行通信,当一个socket(通常都是server socket)等待建立连接时,另一个socket可以要求进行连接,一旦这两个socket连接起来,它们就可以

  • python多线程socket编程之多客户端接入

    Python中实现socket通信的服务端比较复杂,而客户端非常简单,所以客户端基本上都是用sockct模块实现,而服务 端用有很多模块可以使用,如下: 1.客户端 #!/usr/bin/env python #coding:utf-8 ''' file:client.py date:9/9/17 3:43 PM author:lockey email:lockey@123.com desc:socket编程客户端,python3.6.2 ''' import socket,sys HOST =

  • 从零开始的Socket编程学习

    "一切皆socket!" 话虽些许夸张,但是事实也是,现在的网络编程几乎都是用的socket. 本地的进程间通信(IPC)有很多种方式,但可以总结为下面4类: 1.网络中进程之间如何通信? 2.Socket是什么? 3.socket的基本操作 3.1.socket()函数 3.2.bind()函数 3.3.listen().connect()函数 3.4.accept()函数 3.5.read().write()函数等 3.6.close()函数 4.socket中TCP的三次握手建立

  • 如何实现socket网络编程的多线程

    TCP 是可靠的连接.这个可靠的意思就是得有明确的连接对象才行,就像是打电话,拨打的号码必须得在服务中有人接,这个网络会话才算是建立了. UDP: 不可靠的连接.不可靠的意思就是不太确定这个会话最后是不是真的送达你要连接的对象那里去了,就像是寄快递,把快递地址填上了,但是说不好半路会出点啥幺蛾子,能不能安全送达不一定. IP地址: 就是计算机的身份证.身份证号前六位数字是地址码(可以知道是哪个省哪个城市哪个县城),接着八位数字是出生日期码,IP地址也是这样的,它是由网络地址(确定是哪个网络)和主

  • node实现socket链接与GPRS进行通信的方法

    业务背景 最近接到一个需求,在微信公众号界面设计一个独立界面,界面上有 A 电机进.A 电机退.B 电机进.B 电机退 4 个按钮,点击对应按钮,云平台发送不同的代码给电机本地的控制器,控制电机执行不同的动作,电机本地控制器具备GPRS网络功能.服务器与电机本地控制器(客户端)采用 TCP 协议连接,客户端发送心跳包给服务器保持长连接,客户端每次收到服务器下发的代码指令后作出回复主要的实现原理是前端访问后台的接口传输数据.后台采用用socket与GPRS模块进链接,暴露出一个IP+PORT给GP

  • Java编程利用socket多线程访问服务器文件代码示例

    这篇文章将向大家展示Java编程利用socket多线程访问服务器文件代码示例,如果您想先了解Java多线程socket编程的基础知识,可以看下这篇文章:Java多线程编程实现socket通信示例代码. 接下来进入正文,我们看看利用socket多线程访问服务器代码: ServerMain.java package com.ysk.webServer; import java.io.File; import java.io.IOException; import java.net.ServerSoc

  • 如何基于C语言socket编程实现TCP通信

    TCP/IP协议(Transmission Control Protocol/Internet Protocol)叫做传输控制/网际协议,又叫网络通信协议.实际上,它包含上百个功能的协议,如ICMP(互联网控制信息协议).FTP(文件传输协议).UDP(用户数据包协议).ARP(地址解析协议)等.TCP负责发现传输的问题,一旦有问题就会发出重传信号,直到所有数据安全正确的传输到目的地. 套接字(socket):在网络中用来描述计算机中不同程序与其他计算机程序的通信方式.socket其实是一种特殊

  • 新手socket编程入门详解指南

    开发环境 运行平台:Ubantu 14.04 LTS 疑问引导 问题1:头文件的疑问: #include <sys/socket.h>与#include <linux/socket.h>有何区别? 解答: 1. 使用diff查看:adc分别表示添加.删除.修改 2. 其实是路径的不同导致有不同的socke.h文件 3. <sys/socket.h> 是 Internet Protocol family,也就是tcpip协议的应用层接口 4. <linux/sock

  • C++ 中 socket编程实例详解

    C++ 中 socket编程实例详解 sockets(套接字)编程有三种,流式套接字(SOCK_STREAM),数据报套接字(SOCK_DGRAM),原始套接字(SOCK_RAW):基于TCP的socket编程是采用的流式套接字.在这个程序中,将两个工程添加到一个工作区.要链接一个ws2_32.lib的库文件. 服务器端编程的步骤: 1:加载套接字库,创建套接字(WSAStartup()/socket()): 2:绑定套接字到一个IP地址和一个端口上(bind()): 3:将套接字设置为监听模式

  • Java 网络编程socket编程等详解

    网络编程是指编写运行在多个设备(计算机)的程序,这些设备都通过网络连接起来. java.net包中J2SE的API包含有类和接口,它们提供低层次的通信细节.你可以直接使用这些类和接口,来专注于解决问题,而不用关注通信细节. java.net包中提供了两种常见的网络协议的支持: TCP: TCP是传输控制协议的缩写,它保障了两个应用程序之间的可靠通信.通常用于互联网协议,被称TCP / IP. UDP:UDP是用户数据报协议的缩写,一个无连接的协议.提供了应用程序之间要发送的数据的数据包. 本教程

  • Python socket编程实例详解

    本文实例形式较为详细的讲述了Python socket编程.分享给大家供大家参考.具体如下: 复制代码 代码如下: sock=socket.socket(socket.AF_INET,socket.SOCK_STREAM) 上面的代码创建了一个socket对象.type参数代表套接字类型,可为SOCK_STREAM(流套接字)和SOCK_DGRAM(数据报套接字).AF_INET表示创建的是ip v4的类型. 复制代码 代码如下: socket(address_family,type,proto

  • Python多线程编程入门详解

    目录 一.任务.进程和线程 任务 进程 线程 进程和线程的关系 二.Python既支持多进程,又支持多线程 Python实现多进程 Process进程类的说明 Python实现多线程 线程类Thread 总结 一.任务.进程和线程 现代操作系统比如Mac OS X, Linux,Windows等,都是支持"多任务"的操作系统. 什么叫"多任务"(multitasking)呢?简单地说,就是操作系统可以同时运行多个任务.例如你一边在用浏览器上查资料,一边在听MP3,一

  • C++中Socket网络编程实例详解

    C++中Socket网络编程实例详解 现在几乎所有C/C++的后台程序都需要进行网络通讯,其实现方法无非有两种:使用系统底层socket或者使用已有的封装好的网络库.本文对两种方式进行总结,并介绍一个轻量级的网络通讯库ZeroMQ.  1.基本的Scoket编程 关于基本的scoket编程网络上已有很多资料,作者在这里引用一篇文章中的内容进行简要说明. 基于socket编程,基本上就是以下6个步骤: 1.socket()函数 2.bind()函数 3.listen().connect()函数 4

  • 新手必看Android Studio入门详解

    上篇文章已经说过了Android Studio的安装配置,从这里开始我们就来完成第一个Android项目吧! 如何安装配置还不太熟悉的可以参考这篇文章:Android Studio安装配置详细步骤(超详细) 让我们开始第一个Android项目吧 1.建立项目 选一个Empty Activity,然后Next 默认即可,点击 Finish Name:文件名 Save location:文件的保存位置 Language:默认Java,会用Kotlin的也可以更改 API level:默认即可,级别低

  • OpenMP 共享内存的并行编程框架入门详解

    目录 简介 认识 openmp 的简单易用性 C 语言实现 C++ 实现 OpenMP 实现 opnemp 基本原理 积分例子 总结 简介 OpenMP 一个非常易用的共享内存的并行编程框架,它提供了一些非常简单易用的API,让编程人员从复杂的并发编程当中释放出来,专注于具体功能的实现.openmp 主要是通过编译指导语句以及他的动态运行时库实现,在本篇文章当中我们主要介绍 openmp 一些入门的简单指令的使用. 认识 openmp 的简单易用性 比如现在我们有一个任务,启动四个线程打印 he

  • C++元编程语言初步入门详解

    目录 模板 泛型初步 函数模板 友元 模板参数 元编程的基本概念 可变参数模板 模板 由于模板元编程需要以面向对象为基础,所以如有疑问之处可以先补充一点C++面向对象的知识: C++面向对象这一篇就够了 泛型初步 由于C++是静态强类型语言,所以变量一经创建,则类型不得更改.如果我们希望创建一种应用广泛地复数类型,那么相应地需要基于int.float.double这些基础类型逐一创建,十分麻烦.泛型编程便是为了简化这一过程而生. 能够容纳不同数据类型作为成员的类被成为模板类,其基本方法为在类声明

  • linux下的C\C++多进程多线程编程实例详解

    linux下的C\C++多进程多线程编程实例详解 1.多进程编程 #include <stdlib.h> #include <sys/types.h> #include <unistd.h> int main() { pid_t child_pid; /* 创建一个子进程 */ child_pid = fork(); if(child_pid == 0) { printf("child pid\n"); exit(0); } else { print

随机推荐