IIS或者其他Web服务器究竟做了哪些工作,让浏览器请求一个URL地址后显示一个漂亮的网页?要想弄清这个疑问,我想我们可以自己写一个简单的web服务器。

思路:

  1. 创建socket监听浏览器请求。
  2. 连接成功,接受浏览器的请求数据。
  3. 响应浏览器的请求。(我们只响应静态文件) 。

好的,思路很清晰。下面就跟着我动手一步步用代码实现。

1、创建socket监听浏览器请求。

当浏览器请求域名,DNS将域名解析为IP地址。带着缺省的端口80访问我们的服务器。所以Socket的监听,也需要绑定IP和端口。

代码:

// 窗体加载
private void Form1_Load(object sender, EventArgs e)
{
// 关闭线程操作检测
Control.CheckForIllegalCrossThreadCalls = false;
Log("服务器启动中 >>>");
IPAddress ip = IPAddress.Parse(txtIP.Text);
IPEndPoint p = new IPEndPoint(ip, Convert.ToInt32(txtPort.Text));
socketWatch.Bind(p);
socketWatch.Listen();
// 为了不阻塞主线程,开启后台线程,监听浏览器请求。
Thread t = new Thread(Listen);
t.IsBackground = true;
t.Start();
Log("服务器启动成功。");
}

Listen 进程只需要无限循环接受浏览器发送过来的请求。

private void Listen()

{

   while (true)

   {

     Socket sender=socketWatch.Accept();

     Log(sender.RemoteEndPoint.ToString() + "连接成功");

}

}

辅助方法,显示提示信息。(不重要)

// 显示提示信息
private void Log(string msg)
{
txtMsg.AppendText(msg + "\r\n");
}

执行结果

打开我们的winform程序。

打开浏览器访问: http://192.168.95.1:3099/  地址

就能看到以下结果:

2、连接成功,接受浏览器的请求数据。

分析:

浏览器的请求到达服务器后,我们要接待他,就要问他是过来干嘛的。

要和浏览器交流,那么服务器和浏览器就应该说一种语言。这个语言就是http协议。

(对HTTP协议还不太了解的园友可以参考:http协议知识整理

根据HTTP协议,浏览器请求服务器的格式都是规定好的,

所以我们只需要根据格式进行拆分,获取我们感兴趣的数据。

这件事情,已经不属于form1.cs这个类的能力范围了。所以我们找一个管家HttpManager来帮我们处理。至于怎么处理form1不用管,他只需要告诉管家就可以了。

代码:

在form1.cs的Listen中创建管家

HttpManager manager = new HttpManager(sender, Log);

管家接收基础的数据后,再交给专门处理请求数据的对象HttpRequest。

class HttpManager
{
private Socket socket;
private Action<string> Log;
public HttpManager(Socket socket, Action<string> act)
{
this.socket = socket;
this.Log = act;
// 接收消息
string msg = ReceiveMsg();
Log(msg); try
{
// 去找请求对象,拿到我们需要的数据。
HttpRequest request = new HttpRequest(msg);
}
catch (Exception ex)
{ Log(ex.Message);
}
} // 接受浏览器请求报文
private string ReceiveMsg()
{
byte[] date = new byte[ * ];
int count = socket.Receive(date);
return Encoding.UTF8.GetString(date, , count);
}
}

核心的HttpResponse:

class HttpRequest
{
// 请求原始地址 /xxx/xxx.xxxx
public string RawUrl { get; set; }
// 请求类型 Post/Get
public string Method;
public HttpRequest(string msg)
{
string[] lines = msg.Split(new string[] { "\r\n" }, StringSplitOptions.RemoveEmptyEntries);
string[] arrays = lines[].Split(' ');
Method = arrays[];
RawUrl = arrays[];
}
}

3、响应浏览器的请求。

  响应过程其实也非常简单,IO读取浏览器需要请求的数据,拼接响应报文返回给浏览器。

在HttpManager类接收完数据后,调用响应对象。来帮我们处理响应。

// 构造响应报文,响应请求。
HttpResponse response = new HttpResponse(request.RawUrl, socket, Log);

核心HttpResponse代码

 class HttpResponse
{
public int StatusCode { get; set; }
public Dictionary<int, string> Dic = new Dictionary<int, string>();
public string ContentType { get; set; }
public int ContentLength { get; set; }
public byte[] Data { get; set; }
public Socket Socket { get; set; }
public Action<string> Log { get; set; }
public HttpResponse(string rawUrl, Socket socket, Action<string> log)
{
this.Socket = socket;
FillDic();
this.Log = log;
// 判断是否为静态资源
if (Judge(rawUrl))
{
// IO 读取资源
Data = ReadFile(rawUrl);
if (Data == null)
{
// 404 没有这个文件
StatusCode = ;
string msg = "没有找到这个文件。";
ContentType = "text/html";
Data = Encoding.Default.GetBytes(msg);
ContentLength = Data.Length;
ContentType = "text/html";
SendData();
}
else
{
ContentLength = Data.Length;
StatusCode = ;
ContentType = GetContentType(rawUrl);
// 响应请求
SendData();
}
}
else
{
// 不处理
string msg = "只处理静态文件。";
Data = Encoding.Default.GetBytes(msg);
ContentLength = Data.Length;
StatusCode = ;
ContentType = "text/html";
// 响应请求
SendData();
} } // 判断文件是否为静态资源
private bool Judge(string rawUrl)
{
bool isStatic = false;
string ext = Path.GetExtension(rawUrl);
switch (ext)
{
case ".html":
case ".htm":
case ".jpg":
case ".png":
case ".gif":
case ".js":
case ".css":
case ".ico": isStatic = true; break;
}
return isStatic;
} // 响应
private void SendData()
{
StringBuilder sb = new System.Text.StringBuilder();
sb.Append("HTTP/1.1 " + StatusCode + " " + Dic[StatusCode] + "\r\n");
sb.Append("Server: ksn/1.0\r\n");
sb.Append("Content-Type: " + ContentType + "\r\n");
sb.Append("Content-Length: " + ContentLength + "\r\n");
sb.Append("\r\n");
Log("响应报文: \r\n" + sb.ToString()); byte[] head = Encoding.UTF8.GetBytes(sb.ToString());
List<byte> res = new List<byte>();
res.AddRange(head);
if (this.Data != null)
{
res.AddRange(this.Data);
}
Socket.Send(res.ToArray()); } private string GetContentType(string rawUrl)
{
string ct = "text/html";
string ext = Path.GetExtension(rawUrl);
switch (ext)
{
case ".js": ct = "text/javascript";
break;
case ".css": ct = "text/css";
break;
case ".jpg": ct = "image/jpeg";
break;
case ".png": ct = "image/png";
break;
case ".gif": ct = "image/gif";
break;
case ".ico": ct = "image/x-ico";
break;
}
return ct;
} // 初始化状态码
private void FillDic()
{
Dic.Add(, "Ok");
Dic.Add(, "Not Found");
Dic.Add(, "Internal Error");
} // 读取请求的资源
private byte[] ReadFile(string rawUrl)
{
string path = AppDomain.CurrentDomain.BaseDirectory;
path = path + "web" + rawUrl;
if (File.Exists(path))
{
using (FileStream fs = new FileStream(path, FileMode.Open))
{
byte[] data = new byte[fs.Length];
fs.Read(data, , data.Length);
return data;
}
}
else
{
return null;
} }
}

  这个类的代码比较多,但是不难,都是按照顺序一步步往下走的。其实web服务器大致就做了这些事情。只不过封装的东西更多,功能更完善。

资源:

项目源码下载地址

一步步搭建自己的web服务器的更多相关文章

  1. linux系统下搭建自己的web服务器

    之前在windows 2008 server上搭建了一个用于测试的web服务器,但是在打开网站的时候特别的慢,尤其是图片的加载都会失败,当时以为是路径的问题,但是在服务器上自己打开都特别慢,自己实在找 ...

  2. [阿里云部署] Ubuntu+Flask+Nginx+uWSGI+Mysql搭建阿里云Web服务器

    部署地址:123.56.7.181 Ubuntu+Flask+Nginx+uWSGI+Mysql搭建阿里云Web服务器 这个标题就比之前的"ECS服务器配置Web环境的全过程及参考资料&qu ...

  3. 基于python2【重要】怎么自行搭建简单的web服务器

    基本流程:1.需要的支持     1)python本身有SimpleHTTPServer     2)ForkStaticServer.py支持,该文件放在python7目录下     3)将希望共享 ...

  4. nginx搭建前端项目web服务器以及利用反向代理调试远程后台接口

    前端同学用nginx搭建自己的web服务器,后台程序专门部署在一台服务器上(我们之前公司就有三套环境,开发/测试/生产),这样做的好处是 1.前端代码基本都是静态文件,重启一次很快,也就几秒钟时间. ...

  5. 使用 Nginx 搭建静态资源 web 服务器

    在搭建网站的时候,往往会加载很多的图片,如果都从 Tomcat 服务器来获取静态资源,这样会增加服务器的负载,使得服务器运行 速度非常慢,这时可以使用 Nginx 服务器来加载这些静态资源,这样就可以 ...

  6. 用go-module作为包管理器搭建go的web服务器

    本篇博客主要介绍了如何从零开始,使用Go Module作为依赖管理,基于Gin来一步一步搭建Go的Web服务器.并使用Endless来使服务器平滑重启,使用Swagger来自动生成Api文档. 源码在 ...

  7. 使用 Nodejs 搭建简单的Web服务器

    使用Nodejs搭建Web服务器是学习Node.js比较全面的入门教程,因为要完成一个简单的Web服务器,你需要学习Nodejs中几个比较重要的模块,比如:http协议模块.文件系统.url解析模块. ...

  8. qingshow “不积跬步无以至千里,不积小流无以成江海”。--荀子《劝学篇》 用tomcat+花生壳搭建自己的web服务器+域名(参考)

    链接地址:http://www.blogjava.net/qingshow/archive/2010/01/17/309846.html 用tomcat搭建web服务器 目标:免费拥有自己的网站及域名 ...

  9. 【小姿势】如何搭建ipa下载web服务器(直接在手机打开浏览器安装)

    前提: 1) 有个一个现成的web服务器,我用是nodejs. 2) 有个能在用你手机安装的ipa 3) 有个github账号 开搞: 1.用http://plist.iosdev.top/plist ...

随机推荐

  1. scrapy之Pymongo

    用Pymongo保存数据 爬取豆瓣电影top250movie.douban.com/top250的电影数据,并保存在MongoDB中. items.py class DoubanspiderItem( ...

  2. fork有关趣味题

    第一题: //fork1.c #include <unistd.h> #include <stdio.h> int main(void) { ; ; i < ; i++) ...

  3. Codeforces Round #341 (Div. 2) C. Mike and Chocolate Thieves 二分

    C. Mike and Chocolate Thieves time limit per test 2 seconds memory limit per test 256 megabytes inpu ...

  4. hibernate:inverse、cascade,一对多、多对多详解

    1.到底在哪用cascade="..."? cascade属性并不是多对多关系一定要用的,有了它只是让我们在插入或删除对像时更方便一些,只要在cascade的源头上插入或是删除,所 ...

  5. 从git获取项目代码

    1.先复制项目的SSH链接备用 2.在你要放置项目的地方git bash here 3.按照以下步骤走: $ git clone YourSSHAddress // clone项目 $ ls // 查 ...

  6. Java中的数据结构及排序算法

    (明天补充) 主要是3种接口:List Set Map List:ArrayList,LinkedList:顺序表ArrayList,链表LinkedList,堆栈和队列可以使用LinkedList模 ...

  7. Python在线教程(廖雪峰)

    http://www.liaoxuefeng.com/wiki/001374738125095c955c1e6d8bb493182103fac9270762a000

  8. mysql的基本的数据库的查询

    学习一个数据库我们要学习哪些东西: sql数据库的话, curd. 对于查询,要注意表的关联的查询. 索引,触发器,对于控制连接量,脚本, 数据库的可视化工具,权限管理. http://www.360 ...

  9. Knockout结合Bootstrap创建动态UI--产品列表管理

    本篇文章结合Bootstrap创建一个比较完整的应用,对产品列表进行管理,包括产品的增加.删除.修改. 需要的引用 <script type='text/javascript' src='htt ...

  10. Java的序列化机制

    1. 所有实现序列化的类都必须实现Serializable接口,序列化有如下两个特点: 如果一个类可以被序列化,那么它的子类也可以被序列化 由于static代表类成员,trasient代表对象的临时数 ...