ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

网络编程

2019-08-12 09:41:54  阅读:231  来源: 互联网

标签:sockaddr struct read 编程 网络 fd include addr


网络编程:是按网络互联的不同计算机上运行的程序进行数据的交换

网络编程的三个要素
IP地址:(InetAddress)网络中设备的标识,不易记忆,可用主机名
InetAddress类的介绍:
作用:为了方便我们对IP地址进行获取和操作

       常见的功能:
       ​	    public String getHostAddress()//获取IP   
            public String getHostName()//获取主机名

IP的概述:所谓的IP就是给每个连接在Internet上的分配一个32位的地址
IP的组成: IP地址=网络地址+主机地址
IP地址的分类:

A类IP地址:第一段号码为网络地址,剩下的三段号码为本地计算机的号码
B类IP地址:前二段号码为网络地址,剩下的二段号码为本地计算机的号码
C类IP地址:前三段号码为网络地址,剩下的一段号码为本地计算机的号码


端口号:用于标示进程的逻辑地址,不同的进程标示;
端口的分类:物理端口 和 逻辑端口

物理端口:网卡口
逻辑端口:就是我们网络编程所用的

    a:每个网络程序都会有一个逻辑端口
    b:用于标识进程的逻辑地址,不同进程的标识
    c:有效端口:0~65535(两个字节),其中0~1023系统使用或保留端口。

端口号的作用:

  1. 一台拥有IP地址的主机可以提供许多服务,比如Web服务、FTP服务、SMTP服务等

  2. 这些服务完全可以通过1个IP地址来实现。那么,主机是怎样区分不同的网络服务呢?显然不能只靠IP地址,因为IP 地址与网络服务的关系是一对多的关系。

  3. 实际上是通过“IP地址+端口号”来区 分不同的服务的。

  4. 端口提供了一种访问通道,

  5. 服务器一般都是通过知名端口号来识别的。例如,对于每个TCP/IP实现来说,FTP服务器的TCP端口号都是21,每个Telnet服务器的TCP端口号都是23,每个TFTP(简单文件传送协议)服务器的UDP端口号都是69。

传输协议:UDP和TCP

TCP/UDP对比:

  1. TCP面向连接(如打电话要先拨号建立连接);UDP是无连接的,即发送数据之前不需要建立连接
  2. TCP提供可靠的服务。也就是说,通过TCP连接传送的数据,无差错,不丢失,不重复,且按序到达;UDP尽最大努力交付,即不保证可靠交付
  3. TCP面向字节流,实际上是TCP把数据看成一连串无结构的字节流;UDP是面向报文的
    UDP没有拥塞控制,因此网络出现拥塞不会使源主机的发送速率降低(对实时应用很有用,如IP电话,实时视频会议等)
  4. 每一条TCP连接只能是点到点的;UDP支持一对一,一对多,多对一和多对多的交互通信
  5. TCP首部开销20字节;UDP的首部开销小,只有8个字节
  6. TCP的逻辑通信信道是全双工的可靠信道,UDP则是不可靠信道

字节序

在这里插入图片描述
Little endian 小端字节序
Big endian 大端字节序

网络字节序=大端字节序

在这里插入图片描述

在这里插入图片描述

socket服务器和客户端的开发步骤

在这里插入图片描述

API介绍

  1. 指定讲“汉语”(连接协议)

在这里插入图片描述
2.准备好地址
在这里插入图片描述

在这里插入图片描述

地址转换API

int inet_aton(const char* straddr,struct in_addr *addrp);
把字符串形式的“192.168.1.123”转为网络能识别的格式

char* inet_ntoa(struct in_addr inaddr);
把网络格式的ip地址转为字符串形式

  1. 监听
    在这里插入图片描述
  2. 连接
    在这里插入图片描述

数据收发

在这里插入图片描述
在这里插入图片描述

客户端的connect函数

在这里插入图片描述
socket服务端代码实现:

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
int main()
{
//socket
int s_fd;

s_fd=socket(AF_INET,SOCK_STREAM,0);
if(s_fd==-1){
    perror("socket");
    exit(-1);
}

struct sockaddr_in s_addr;
s_addr.sin_family=AF_INET;
s_addr.sin_port=htons(7878);// 返回网络字节序的值
inet_aton("192.168.175.128",&s_addr.sin_addr);//把字符串形式的“192.168.1.123”
                                //转为网络能识别的格式


//bind
bind(s_fd,(struct sockaddr *)&s_addr,sizeof(struct sockaddr_in));

//listen
listen(s_fd,10);
//accept
int c_fd=accept(s_fd,NULL,NULL);
//read
//write

while(1);
return 0;

}

结果:

在这里插入图片描述

socket服务端代码实现(用window的cmd代替客户端):

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
int main()
{
//socket
int s_fd;
int n_read;
char readbuf[128]
char *msg=“I get your connnext”;

struct sockaddr_in s_addr;
struct sockaddr_in c_addr;

memset(&s_addr,0,sizeof(struct sockaddr_in));
memset(&c_addr,0,sizeof(struct sockaddr_in));

s_fd=socket(AF_INET,SOCK_STREAM,0);
if(s_fd==-1){
    perror("socket");
    exit(-1);
}

s_addr.sin_family=AF_INET;
s_addr.sin_port=htons(7878);// 返回网络字节序的值
inet_aton("192.168.175.128",&s_addr.sin_addr);//把字符串形式的“192.168.1.123”
                                //转为网络能识别的格式


//bind
bind(s_fd,(struct sockaddr *)&s_addr,sizeof(struct sockaddr_in));

//listen
listen(s_fd,10);
//accept
int clen=sizeof(struct sockaddr_in);
int c_fd=accept(s_fd,(struct sockaddr *)&c_addr,&clen);
//read
n_read=read(c_fd,readbuf,128);
if(n_read==-1){
    perror("read");
}else{
    printf("get message:%d,%s\n",n_read,readbuf);
}
//write
write(c_fd,msg,strlen(msg));
while(1);
return 0;

}

cmd:
telnet 192.168.175.128 7878

结果:
在这里插入图片描述

客户端:

代码:
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
int main()
{
//socket
int c_fd;
int n_read;
char readbuf[128];
char *msg=“msg from client”;

struct sockaddr_in c_addr;

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

c_fd=socket(AF_INET,SOCK_STREAM,0);
if(c_fd==-1){
    perror("socket");
    exit(-1);
}

c_addr.sin_family=AF_INET;
c_addr.sin_port=htons(7878);// 返回网络字节序的值
inet_aton("192.168.175.128",&c_addr.sin_addr);//把字符串形式的“192.168.1.123”
                                //转为网络能识别的格式

//connext
int a;
a=connect(c_fd,(struct sockaddr *)&c_addr,sizeof(struct sockaddr));
if(a==-1){
    perror("connext");
    exit(-1);
}
//send
write(c_fd,msg,strlen(msg));

//read
n_read=read(c_fd,readbuf,128);
if(n_read==-1){
    perror("read");
}else{
    printf("get message from serve:%d,%s\n",n_read,readbuf);
}
while(1);
return 0;

}

联机结果:

在这里插入图片描述

客户端和服务端互相发消息代码:

服务端:
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char **argv) {
// socket
int s_fd;
int c_fd;
int n_read;
char readbuf[128];
char msg[128] = {0};
// char *msg = “I get your connnext”;

struct sockaddr_in s_addr;
struct sockaddr_in c_addr;

if (argc != 3) {

    printf("param is not good\n");
    exit(-1);
}

memset(&s_addr, 0, sizeof(struct sockaddr_in));
memset(&c_addr, 0, sizeof(struct sockaddr_in));

s_fd = socket(AF_INET, SOCK_STREAM, 0);
if (s_fd == -1) {
    perror("socket");
    exit(-1);
}

s_addr.sin_family = AF_INET;
s_addr.sin_port = htons(atoi(argv[2]));  // 返回网络字节序的值
inet_aton(argv[1], &s_addr.sin_addr);  //把字符串形式的“192.168.1.123”
                                       //转为网络能识别的格式

// bind
bind(s_fd, (struct sockaddr *)&s_addr, sizeof(struct sockaddr_in));

// listen
listen(s_fd, 10);
// accept

int clen = sizeof(struct sockaddr_in);
while (1) {
    c_fd = accept(s_fd, (struct sockaddr *)&c_addr, &clen);
    if (c_fd == -1) {
        perror("accept");
    }
    printf("get connect:%s\n", inet_ntoa(c_addr.sin_addr));
    // read
    if (fork() == 0) {
        if (fork() == 0) {
            while (1) {
                memset(msg, 0, sizeof(msg));
                printf("input:\n");
                gets(msg);
                // write
                write(c_fd, msg, strlen(msg));
            }
        }
        while (1) {
            memset(readbuf,0,sizeof(readbuf));
            n_read = read(c_fd, readbuf, 128);
            if (n_read == -1) {
                perror("read");
            } else {
                printf("get message:%d,%s\n", n_read, readbuf);
            }
        }
        break;
    }
}

return 0;

}

客户端:
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char **argv) {
// socket
int c_fd;
int n_read;
char readbuf[128];
// char *msg=“msg from client”;
char msg[128] = {0};
struct sockaddr_in c_addr;
if (argc != 3) {

    printf("param is not good\n");
    exit(-1);
}

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

c_fd = socket(AF_INET, SOCK_STREAM, 0);
if (c_fd == -1) {
    perror("socket");
    exit(-1);
}

c_addr.sin_family = AF_INET;
c_addr.sin_port = htons(atoi(argv[2]));  // 返回网络字节序的值
inet_aton(argv[1], &c_addr.sin_addr);  //把字符串形式的“192.168.1.123”
                                       //转为网络能识别的格式

// connext
int a;
a = connect(c_fd, (struct sockaddr *)&c_addr, sizeof(struct sockaddr));
if (a == -1) {
    perror("connext");
    exit(-1);
}
while (1) {
    if (fork() == 0) {
        while (1) {
            printf("input :\n");
            memset(msg, 0, sizeof(msg));
            gets(msg);
            // send
            write(c_fd, msg, strlen(msg));
        }
    }
    while (1) {
        // read
        memset(readbuf,0,sizeof(readbuf));
        n_read = read(c_fd, readbuf, 128);
        if (n_read == -1) {
            perror("read");
        } else {
            printf("get message from serve:%d,%s\n", n_read, readbuf);
        }
    }
}

return 0;

}

结果:

在这里插入图片描述

服务器自动回复解决多个客户端竞争资源代码:

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char **argv) {
// socket
int s_fd;
int c_fd;
int n_read;
char readbuf[128];
char msg[128] = {0};
int mark=0;//没有初始化会数据溢出
// char *msg = “I get your connnext”;

struct sockaddr_in s_addr;
struct sockaddr_in c_addr;

if (argc != 3) {

    printf("param is not good\n");
    exit(-1);
}

memset(&s_addr, 0, sizeof(struct sockaddr_in));
memset(&c_addr, 0, sizeof(struct sockaddr_in));

s_fd = socket(AF_INET, SOCK_STREAM, 0);
if (s_fd == -1) {
    perror("socket");
    exit(-1);
}

s_addr.sin_family = AF_INET;
s_addr.sin_port = htons(atoi(argv[2]));  // 返回网络字节序的值
inet_aton(argv[1], &s_addr.sin_addr);  //把字符串形式的“192.168.1.123”
                                       //转为网络能识别的格式

// bind
bind(s_fd, (struct sockaddr *)&s_addr, sizeof(struct sockaddr_in));

// listen
listen(s_fd, 10);
// accept

int clen = sizeof(struct sockaddr_in);
while (1) {
    c_fd = accept(s_fd, (struct sockaddr *)&c_addr, &clen);
    if (c_fd == -1) {
        perror("accept");
    }
    printf("get connect:%s\n", inet_ntoa(c_addr.sin_addr));
    mark++;
    // read
    if (fork() == 0) {
        if (fork() == 0) {
            while (1) {
                sprintf(msg,"welcome to NO.%d client\n",mark);
                // write
                write(c_fd, msg, strlen(msg));
                sleep(3);
            }
        }
        while (1) {
            memset(readbuf,0,sizeof(readbuf));
            n_read = read(c_fd, readbuf, 128);
            if (n_read == -1) {
                perror("read");
            } else {
                printf("get message:%d,%s\n", n_read, readbuf);
            }
        }
        break;
    }
}

return 0;

}

标签:sockaddr,struct,read,编程,网络,fd,include,addr
来源: https://blog.csdn.net/weixin_43664986/article/details/99222217

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有