我们在浏览器打开网页,其实是向远端服务器提出页面发送请求,远端服务器在接到请求后,就开始执行请求页面的程序文件,然后将执行结果通过html格式,发送到你的浏览器,再显示出来。以下用百度(www.baidu.com)为例:

我们可以通过GET请求可以获得,百度的http应答格式,详情点我查看。或者通过 "$curl -l www.baidu.com"的方式获得,如果提示没有curl命令则表示你的Linux没有安装curl,安装方法也很简单,拿ubuntu为例,输入以下命令"$sudo apt-get install curl",一路回车即可安装。

查询命令也很简单,输入"$curl -i www.baidu.com"

得到百度的服务器的http应答头部信息后,我们可以仿照这种格式,写一个我们自己的服务端。

百度服务器响应http应答头部信息

HTTP/1.1 200 OK

Accept-Ranges: bytes

Cache-Control: private, no-cache, no-store, proxy-revalidate, no-transform

Connection: Keep-Alive

Content-Length: 2381

Content-Type: text/html

Date: Tue, 29 Oct 2019 01:11:38 GMT

Etag: "588604c8-94d"

Last-Modified: Mon, 23 Jan 2017 13:27:36 GMT

Pragma: no-cache

Server: bfe/1.0.8.18

Set-Cookie: BDORZ=27315; max-age=86400; domain=.baidu.com; path=/

这里我们先简单的模拟其中的部分:

  1. HTTP/1.1 200 OK
  2. Server: myhttp
  3. Content-Length: 330
  4. 发送html文件

核心思想:将浏览器的请求信息解析,提取出相应内容作为返回信息的方法。

服务器代码实现如下:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <fcntl.h> #define PATH "/home/stu/mydir/html_test" /*这里是html文件所在的根目录*/ int create_socket();
int main()
{
int sockfd = create_socket();
assert( sockfd != -1 ); while( 1 )
{
struct sockaddr_in caddr;
int len = sizeof(caddr);
int c = accept(sockfd,(struct sockaddr*)&caddr,&len); /*accept()返回新的连接套接子,表示网络已经存在点对点的连接*/
if ( c < 0 )
{
continue;
} char buff[1024] = {0}; /*buff用于接收浏览器(客户端)的请求*/
int res = recv(c,buff,1023,0); char * s = strtok(buff," "); /*分割字符串函数,用于提取http中的信息*/
if ( s == NULL )
{
close(c);
continue;
} printf("方法:%s\n",s); /*服务端打印请求方法,如:GET*/
printf("read:\n%s\n",buff); /*服务端打印读取到信息*/ s = strtok(NULL," ");
if ( s == NULL )
{
send(c,"404",3,0); //该路经下没有该文件,404错误
close(c);
continue;
} if ( strcmp(s,"/") == 0 ) /*如果客户端没有标明想访问哪一个文件,则默认访问index.html文件*/
{
s = "/index.html";
} char path[256] = {PATH}; /*网页文件路径*/
strcat(path,s); /*文件路径+文件*/ printf("file:%s\n",path); /*服务端打印文件的地址*/ int fd = open(path,O_RDONLY); /*定义文件描述符,读取文件内容,发送回浏览器(客户端)*/
if ( fd == -1 )
{
close(c);
continue;
} int size = lseek(fd,0,SEEK_END); /*返回文件大小*/
lseek(fd,0,SEEK_SET); /*文件指针回正,指向文件开头*/ /*http应答头部信息拼装与head_buff中*/
char head_buff[256] = {"HTTP/1.1 200 OK\r\n"};
strcat(head_buff,"Server: myhttp\r\n");
sprintf(head_buff+strlen(head_buff),"Content-Length: %d\r\n",size);
strcat(head_buff,"\r\n"); /*流式数据,可多次发送,先发送应答头部信息*/
send(c,head_buff,strlen(head_buff),0); /*读取index.html文件,发送回客户端*/
char data[1024] = {0};
int num = 0;
while( ( num = read(fd,data,1024)) > 0 ) /*循环读取,直到读完整个文件*/
{
send(c,data,num,0); /*边读取边发送*/
} close(fd); /*关闭文件描述符*/
close(c); /*短连接,断开连接*/ } close(sockfd);
exit(0); } //封装套接子为create_socker()方法
int create_socket()
{
int sockfd = socket(AF_INET,SOCK_STREAM,0);
if ( sockfd == -1 )
{
return -1;
} struct sockaddr_in saddr;
memset(&saddr,0,sizeof(saddr));
saddr.sin_family = AF_INET;
saddr.sin_port = htons(80);//root
saddr.sin_addr.s_addr = inet_addr("127.0.0.1"); int res = bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));
if ( res == -1 )
{
return -1;
} if ( listen(sockfd,5) == -1 )
{
return -1;
} return sockfd;
}

服务器代码已经实现,现在只需一个index.html文件就可以进行测试。

<html>
<head>
<meta charset=utf-8>
<title>我的html页面</title>
</head> <body background="11.PNG">
<center>
<h3><测试标题> </h3>
<hr>
<br>
这是第一行,<br>
这是第二行. <br>
这是第三行,<br>
这是第四行. <br>
</center>
</body>
</html>

简单的写一个html文档,把文档命名为 index.html ,存放于 我们定义的路径下。

打开浏览器,输入 localhost 或 127.0.0.1 即可看见我们的网页内容了,测试截图如下:

在上图中我们可以看到,浏览器显示的是经过处理的文件内容,我们服务端发送的应答头部信息和原本的内容已经经过处理,如果我们想看到服务端发送的具体内容,我们还可以自己再写一个客户端,让客户端连接自己的服务端并打印出服务端发送的内容。

客户端代码:

#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int main(){
int sockfd;
int len;
struct sockaddr_in address;
int result;
char *strings="GET / HTTP/1.1\r\nHost: 127.0.0.1\r\nConnection: Close\r\n\r\n";
char ch; sockfd = socket(AF_INET, SOCK_STREAM, 0);
address.sin_family = AF_INET;
address.sin_addr.s_addr = inet_addr("127.0.0.1");
address.sin_port = htons(80); len = sizeof(address);
result = connect(sockfd, (struct sockaddr *)&address, len);
if(result == -1){
perror("oops: client1");
return 1;
} write(sockfd,strings,strlen(strings)); while(read(sockfd,&ch, 1)){
printf("%c", ch);
}
close(sockfd); return 0;
}

运行服务端、在运行客户端,我们可以在客户端上看到如下内容:

ps:实现本机访问后也可以试试局域网内互相访问,只需修改服务器端代码,把ip地址改成做服务端的主机的ip地址即可,在另一台主机上输入服务端主机的ip即可访问该网页内容。此博客在上一篇博客Linux套接子(c语言)模拟http请求、应答的基础上做了加强,更加接近真实的http访问协议。

模拟web服务器http请求应答的更多相关文章

  1. [转] c# 模拟Asp.net页面中的某个按钮的点击,向web服务器发出请求

    在没有做题目中所述的内容的时候,感觉这应该是很简单的东西,但是当真正开始做的时候却发现,有很多问题现在在这里写出来,供和我一样水平不高的参考一下. 在写本文之前参照了一下文章 欢迎使用CSDN论坛阅读 ...

  2. 模拟web服务器 (小项目) 搭建+部署

    模拟web服务器:可以从浏览器中访问到自己编写的服务器中的资源,将其资源显示在浏览器中. 技术选型: corejava 线程池 同任务并发执行 IO流 传递数据 客户端也会像服务端发送数据, 服务器像 ...

  3. web服务器获取请求客户端真实地址的方法

    服务器获取客户端或者网页的请求,获取IP时需要注意,因为一个请求到达服务器之前,一般都会经过一层或者多层代理服务器,比如反向代理服务器将http://192.168.1.10:port/ 的URL反向 ...

  4. 第一篇 先用socket模拟web服务器

    一.用socket来模拟网站访问 socket为python2.7 #!/usr/bin/env python # -*- coding:utf-8 -*- import socket def han ...

  5. Socket模拟Web服务器

    效果如下: 源码下载地址:https://github.com/doyoulaikeme/DotNetSample/tree/master/DotNetSample3/SocketWebServer

  6. C#中使用Socket请求Web服务器过程

    最开始我们需要明白一件事情,因为这是这篇文章的前提: HTTP协议只是一个应用层协议,它底层是通过TCP进行传输数据的.因此,浏览器访问Web服务器的过程必须先有“连接建立”的发生. 而有人或许会问: ...

  7. 详谈socket请求Web服务器过程

    最开始我们需要明白一件事情,因为这是这篇文章的前提: HTTP协议只是一个应用层协议,它底层是通过TCP进行传输数据的.因此,浏览器访问Web服务器的过程必须先有“连接建立”的发生. 而有人或许会问: ...

  8. 详谈socket请求Web服务器过程(转)

    最开始我们需要明白一件事情,因为这是这篇文章的前提: HTTP协议只是一个应用层协议,它底层是通过TCP进行传输数据的.因此,浏览器访问Web服务器的过程必须先有“连接建立”的发生. 而有人或许会问: ...

  9. 【网络开发】详谈socket请求Web服务器过程

    最开始我们需要明白一件事情,因为这是这篇文章的前提: HTTP协议只是一个应用层协议,它底层是通过TCP进行传输数据的.因此,浏览器访问Web服务器的过程必须先有"连接建立"的发生 ...

随机推荐

  1. 由浅入深---MyBatis的全局配置文件

    从我开始接触代码,我就很怕写配置文件,一般的配置文件我都是直接从上一个项目复制到这个项目来改改,可能有部分同学也有我这种痛吧: 我目前一般的做法,先去找找例子(从网上,从github,从官网)之后再改 ...

  2. 【FAQ】接入HMS Core推送服务过程中一些常见问题总结

    HMS Core 推送服务(Push Kit)是华为提供的消息推送平台,建立了从云端到终端的消息推送通道.开发者通过集成推送服务,可以向客户端应用实时推送消息,构筑良好的用户关系,提升用户的感知度和活 ...

  3. quartz框架(六)-ThreadPool

    ThreadPool 本篇博文,博主将介绍Quartz框架中ThreadPool线程池相关的内容.线程池顾名思义,就是一个可以帮助我们来进行线程资源管理的对象.在web开发中,常见的就有数据库连接池, ...

  4. laravel 7 登录

    1:路由,展示登录表单 Route::group(['prefix'=>'day','namespace'=>'day18'],function (){ // 登录 Route::get( ...

  5. 图解|用好MySQL索引,你需要知道的一些事情

    我是蝉沐风. 这一篇文章来聊一聊如何用好MySQL索引. 为了更好地进行解释,我创建了一个存储引擎为InnoDB的表user_innodb,并批量初始化了500W+条数据.包含主键id.姓名字段(na ...

  6. 13、mysql锁

    mysql锁 事务的隔离性是通过锁来实现的.为保证数据的一致性,需要锁对并发事务操作进行控制.同时锁机制也为实现MySQL的各个隔离级别提供了保证. mysql并发事务访问相同的记录会出现什么问题(在 ...

  7. Go值类型和引用类型+作用域+空白标识符+常量

    值类型和引用类型 所有像 int.float.bool 和 string 这些基本类型都属于值类型,使用这些类型的变量直接指向存在内存中的值: 当使用等号 = 将一个变量的值赋值给另一个变量时,如:j ...

  8. Termux镜像在阿里云镜像站首发上线

    镜像下载.域名解析.时间同步请点击阿里云开源镜像站 简介 Termux 是 Android 平台上的一个终端模拟器,它将众多 Linux 上运行的软件和工具近乎完美的移植到了手机端. 无需任何复杂的安 ...

  9. docker是干什么的,docker常用命令

    镜像下载.域名解析.时间同步请点击 阿里云开源镜像站 一.百度百科 Docker是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖到一个可移植的镜像中,然后发布到任何流行的Linux或Win ...

  10. 九鼎RK3399笔记一:Linux平台手册

    @ 目录 一.git下载九鼎SDK 二.安装所需的软件包: 三.安装 kernel 及 u-boot 编译需要依赖的软件包 四.安装文件系统需要依赖的软件包 五.安装 Buildroot 编译需要依赖 ...