从零开始一个http服务器 (一)

代码地址 : https://github.com/flamedancer/cserver

git checkout step1

一个简单的socket server

  • 从helloworld开始
  • 回顾c语言的socket 通信
  • 一个简单的socket server
  • 用telent测试

从helloworld 开始

先来回顾下c语言的,c语言的helloword程序如下

// main.c
#include<stdio.h> int main() {
printf("hello world");
}

编译 gcc main.c

运行 ./a.out

输出 hello world

回顾c语言的socket 通信

socket (server端) 通信的一般 步骤

* 创建 socket , 返回socket 文件描述符,需指明域(本地文件socket还是网络socket),类型(TCP 还是 UDP)
* 绑定 bind, 绑定socket地址(本地socket文件地址 或 网络地址 IP + port)
* 监听 listen, 为socket创建监听队列, 连接到socket的链接将会进入这个队列, 需要指明队列最大长度
* 接收链接 accept, 接收客户端链接,返回接收到的 客户socket文件描述符
* 读写 read/write, 对 客户socket文件描述符 进行 读写操作来进行通信
* close, 通信结束, 关闭 客户socket文件描述符, 整个server结束,也要关闭 server socket文件描述符

c 语言 socket通信有关的函数及结构原型

  1. creating a socket
    #include <sys/types.h> #include <sys/socket.h>
int socket(int domain, int type, int protocol); *** domains
AF_UNIX: 本地文件socket (file system sockets) AF_INET: 网络socket (UNIX network sockets)
...
*** type
SOCK_STREAM: TCP 协议
SOCK_DGRAM: UDP 协议
*** protocol
一般选默认值 0
  1. struct: socket Address socket 地址结构体

    本地文件socket地址:
    AF_UNIX socket_un    defind in sys/un.h
struct sockaddr_un {
sa_family_t sun_family; // AF_UNIX
char sun_path[]; // pathname
}; 网络socket 地址:
AF_INET sockaddr_in defind in netinet/in.h
struct sockaddr_in {
short int sin_family; // AF_INET
unsigned short in sin_port; // Port number
struct in_addr sin_addr; // Inernet address
};
其中代表ip地址的结构体in_addr:
struct in_addr {
unsigned long int s_addr;
}
  1. bind

    成功返回0,失败返回-1,失败信息见 errno
    #include <sys/socket.h>
int bind(int socket, const struct sockaddr *address, size_t address_len);
  1. Creating a socket queue
    #include <sys/socket.h>
int listen(int socket, int backlog); // backlog : the maximum number of pending connections
  1. Accept connections

    这里的address和address_len 都是指client端的地址,如果成功连接client,则address被填充

    返回连接后client 的 socket 文件描述符
    #include <sys/socket.h>
int accept(int socket, struct sockaddr *address, size_t *address_len);
  1. Host and Network Byte Ordering

    有可能本地字节编码顺序和网络字节编码顺序不同,本地字节编码要转成网络字节编码
    #include <netinet/in.h>
unsigned long int htonl(unsigned long int hostlong);
unsigned short int htons(unsigned short int hostshort);

一个简单的socket server

#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <netinet/in.h> int main() {
int server_sockfd, client_sockfd;
int server_len, client_len;
struct sockaddr_in server_address;
struct sockaddr_in client_address;
server_sockfd = socket(AF_INET, SOCK_STREAM, 0); // 创建socket,选择地址类型为网络地址,选择 TCP 通信 server_address.sin_family = AF_INET;
server_address.sin_addr.s_addr = inet_addr("127.0.0.1"); // 设置网络地址的ip, inet_addr 会自动 转为 网络字节顺序
server_address.sin_port = htons(9734); // 设置端口号,注意这里的 htons 方法
server_len = sizeof(server_address);
bind(server_sockfd, (struct sockaddr *)&server_address, server_len); listen(server_sockfd, 5);
while(1) {
char ch[5000];
char send_str[] = "hello world !\n"; // 准备给连接过来的客户端发送的字符串
client_len = sizeof(client_address);
client_sockfd = accept(server_sockfd,
(struct sockaddr *)&client_address, &client_len);
read(client_sockfd, &ch, 5000); // 接收 客户端传来的字符
printf("%s", ch); // 打印我们接收到的字符
write(client_sockfd, &send_str, sizeof(send_str)/sizeof(send_str[0])); // 向客户端发送数据,这里的 read write 和 和文件读写时没什么区别
close(client_sockfd);
}
}

和之前helloword一样编译运行我们的第一个版本!

用 telnet 测试

看看效果吧!新启一个终端,然后用telnet 尝试连接我们的服务器。

执行命令 telnet 127.0.0.1 9734

随便输入几个字符按回车

屏幕输出大概为这样:

Trying 127.0.0.1...
Connected to localhost.
Escape character is '^]'.
dsfsd
hello world !
Connection closed by foreign host.

再返回查看我们的服务器屏幕打印,能看到我们刚才随意输入的字符,说明我们的服务器能成功接收并返回数据了。

从零开始一个http服务器(一)-开始的更多相关文章

  1. 从零开始一个http服务器(六)-多路复用和压力测试

    从零开始一个http服务器(六)-多路复用和压力测试 代码地址 : https://github.com/flamedancer/cserver git checkout step6 运行: make ...

  2. 从零开始一个http服务器(五)-模拟cgi

    从零开始一个http服务器-模拟cgi(五) 代码地址 : https://github.com/flamedancer/cserver git checkout step5 运行: make cle ...

  3. 从零开始一个http服务器(三)-返回response 构造

    从零开始一个http服务器(三) 代码地址 : https://github.com/flamedancer/cserver git checkout step3 运行: gcc request.h ...

  4. 从零开始一个http服务器(二)-请求request解析

    从零开始一个http服务器 (二) 代码地址 : https://github.com/flamedancer/cserver git checkout step2 解析http request 观察 ...

  5. 从零开始一个http服务器(四)-动态返回

    从零开始一个http服务器(四) 代码地址 : https://github.com/flamedancer/cserver git checkout step4 运行: make clean &am ...

  6. 在阿里云服务器(ECS)上从零开始搭建nginx服务器

    本文介绍了如何在阿里云服务器上从零开始搭建nginx服务器.阿里云服务器(ECS)相信大家都不陌生,感兴趣的同学可以到http://www.aliyun.com/product/ecs去购买,或到体验 ...

  7. Linux下配置一个VNC服务器

    在Linux下配置一个VNC服务器,并设置2个用户,要求其中一个用户登录时不需要输入密码. 然后在客户端使用ssh+vncview的方式访问. 1确认vnc安装 2配置vncserver 3测试vnc ...

  8. 在 Linux 上配置一个 syslog 服务器

    syslog服务器可以用作一个网络中的日志监控中心,所有能够通过网络来发送日志的设施(包含了Linux或Windows服务器,路由器,交换机以及其他主机)都可以把日志发送给它. 通过设置一个syslo ...

  9. nodejs创建一个HTTP服务器 简单入门级

    const http = require('http');//请求http.createServer(function(request, response){    /*createServer该函数 ...

随机推荐

  1. 【Leetcode】【Medium】Flatten Binary Tree to Linked List

    Given a binary tree, flatten it to a linked list in-place. For example,Given 1 / \ 2 5 / \ \ 3 4 6 T ...

  2. 基于腾讯云centos简单搭建VSFTP

    基于腾讯云centos7.3搭建VSFTP 环境分析: 基于vsftp服务在于云主机上,所以推荐使用FTP的PASV模式: FTP协议有两种工作方式:PORT方式和PASV方式,中文意思为主动式和被动 ...

  3. gogs配置及迁移

    工作需要迁移gogs,粗略记下笔记 操作系统:CentOS Linux release 7.4.1708 (Core) 防火墙:关闭状态,如有需要开启默认的3000端口 一.配置 首先安装git [r ...

  4. 邮件营销巧妙添加GIF让您的邮件动起来

    动态图片远比静态图片要吸引人,因此近年来,一些营销人员也开始越来越频繁的使用GIF动画图片,适当的穿插和点缀动态图片,能够生动形象的表达出 主题,并且时不时令读者忍俊不禁.尤其是做邮件营销的,如果能在 ...

  5. 使用Swagger处理Api的显示与隐藏

    一.在SwaggerConfig.cs中配置如下: c.DocumentFilter<ShowApiFilter>(); c.DocumentFilter<HideApiFilter ...

  6. commons-lang(一)

    写在开头的话: 在工作中,经常会要用到一些公用的组件包,比如日期,String这些等,换过几次工作的朋友对这事肯定比较深刻,如果自己有积累的还好,对于没有积累的同学,尤其是规模较小的公司,怕是又要重头 ...

  7. ejb3persistence.jar javax.persistence的注解配置

    JPA注解持久化类很方便,需要jar包:ejb3-persistence.jar.我用以下三个类来说明用法.  sh原创 转载请注明: http://67566894.iteye.com/blog/6 ...

  8. Java连接MQ的实例, 测试类

    package cjf.mq.mqclient; import com.ibm.mq.MQC; import com.ibm.mq.MQEnvironment; import com.ibm.mq.M ...

  9. 牛客网多校训练第一场 J - Different Integers(树状数组 + 问题转换)

    链接: https://www.nowcoder.com/acm/contest/139/J 题意: 给出n个整数的序列a(1≤ai≤n)和q个询问(1≤n,q≤1e5),每个询问包含两个整数L和R( ...

  10. JQuery 学习总结及实例 !! (转载)

    出自 new:http://www.jianshu.com/users/1967b163cb61/latest_articles 1.JQuery简介 普通JavaScript的缺点:每种控件的操作方 ...