刚刚接触ASP.NET编程,为了更好的屡清楚服务器的处理过程,就用Socket模拟服务器来处理请求。用Socket来模拟服务器的时候,同样是自己来封装一些对应的类文件。包括 HttpRequest、HttpResponse、HttpContext、HttpApplication、IHttpHandel。主要的执行流程是:先用Socket来创建一个简单的服务器,进行监听,当监听到请求后将请求交给处理程序去处理,应用程序中根据请求的是静态资源还是动态资源做出不同的处理。然后通过Socket对象将响应类容给发送回去。这只是为了更好的了解服务器的处理,进行了一个简单的模拟,程序中有很多的bug,但是能够实现基本的功能

HttpRequest类

主要包含几个主要的属性:请求方式,请求地址,请求协议的版本号

     public class HttpRequest
{
public HttpRequest (string str)
{
if (string.IsNullOrEmpty(str))
{
return;
}
string head = str.Replace("\r\n", "$").Split('$')[];
string[] heads = head.Split(' ');
Method = heads.Length>? heads[]:"";
//得到请求的绝对路径
Url = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), heads.Length>? heads[].Substring():heads[]);
Protocol = heads.Length > ? heads[] : ""; } public string Method { get; set; }
public string Url { get; set; }
public string Protocol { get; set; }
}

HttpRequest类

HttpResponse类

包含响应头和响应体,在网络上传输的是二进制文件,故将这两个属性定义为byte[],中间还包含其他的一些属性。其中响应长度是动态的,在处理程序里面设置

     public class HttpResponse
{
public HttpResponse (HttpRequest request)
{
this.request = request;
Type = GetType(Path.GetExtension(request.Url));
} private HttpRequest request;
private string Type { get; set; }
public int Length { get; set; }
public byte[] Head {
get
{ // -----响应报文------------
//HTTP/1.1 200 OK
//Connection: keep-alive
//Date: Thu, 26 Jul 2007 14:00:02 GMT
//Server: Microsoft-IIS/6.0
//X-Powered-By: ASP.NET
//Content-Length: 190
//Content-Type: text/html
//Set-Cookie: ASPSESSIONIDSAATTCSQ=JOPPKDCAMHHBEOICJPGPBJOB; path=/
//Cache-control: private
////--空行--
//响应体(正文) StringBuilder sb = new StringBuilder();
sb.AppendFormat("{0} {1}\r\n", request.Protocol, "200 OK");
sb.AppendLine("Date:" + DateTime.Now.ToString());
sb.AppendLine("Server:QIGANG-PC");
sb.AppendLine("Content-Length:" + Length);
sb.AppendLine("Content-Type:" + Type);
sb.AppendLine();
return Encoding.UTF8.GetBytes(sb.ToString()); }
}
public byte[] Body { get; set; } ///根据请求来得到响应的类型
private string GetType (string ext)
{
string type1 = "text/html;charset=utf-8";
switch (ext)//mime-type
{
case ".aspx":
case ".html":
case ".htm":
type1 = "text/html;charset=utf-8";
break;
case ".png":
type1 = "image/png";
break;
case ".gif":
type1 = "image/gif";
break;
case ".jpg":
case ".jpeg":
type1 = "image/jpeg";
break;
case ".css":
type1 = "text/css";
break;
case ".js":
type1 = "application/x-javascript";
break;
default:
type1 = "text/plain;charset=gbk";
break;
}
return type1;
}
}

HttpResponse类

HttpContext类

在这个类里面简单的封装了两个成员对象,就是HttpRequest和HttpResponse两个成员,其它的就从简了

     public class HttpContext
{
public HttpContext (string str)
{
Request = new HttpRequest(str);
Response = new HttpResponse(Request);
}
public HttpRequest Request { get; set; }
public HttpResponse Response { get; set; }
}

HttpContext类

IHttpHandel接口

由于在客户端请求的数据中可能请求的是一个动态网页,这是就需要交给.NET Framework 来进行处理,为了方便处理,故要求所有的动态网页都需要实现一个接口,只有实现了这个接口的程序才能够被浏览器给请求到

     public interface IHttpHandel
{
void ProcessRequest (HttpContext context);
}

IHttpHandel接口

服务器端Socket程序

在这主要是启动一个Socket对象,来进行连接的监听,然后把监听到的对象交给处理程序 HttpApplication进行处理

         private void button1_Click (object sender, EventArgs e)
{
socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
socket.Bind(new IPEndPoint(IPAddress.Parse("127.0.0.1"), ));
socket.Listen();
Thread thread = new Thread((obj) =>
{
Socket server = obj as Socket;
while (true)
{
Socket client = server.Accept();
Thread cth = new Thread((obj2) =>
{
Socket cSocket = obj2 as Socket; byte[] by = new byte[cSocket.Available];
cSocket.Receive(by, , by.Length, SocketFlags.None);
//拿到请求头文件
string str = Encoding.UTF8.GetString(by); /*--------- 调用请求处理函数进行处理 ---------*/
HttpContext context = new HttpContext(str);
HttpApplication app = new HttpApplication(context, cSocket);
});
cth.IsBackground = true;
cth.Start(client);
}
});
thread.IsBackground = true;
thread.Start(socket);
}

Socket服务端程序

HttpApplication类

这个类需要传递一个参数,上下文对象。然后进行请求的解析,根据请求的是静态资源还是动态资源去进行不同的处理。如果是静态资源就直接冲磁盘文件中读取返回,如果是动态资源,就交给对应的类。当然前提是请求的资源名称就对应了一个类文件。

     public class HttpApplication
{
//构造函数
public HttpApplication (HttpContext context,Socket socket)
{
string url =context.Request.Url;
if(string.IsNullOrEmpty(url)){
return;
}
string ext = Path.GetExtension(url);
//请求的是动态资源文件
if (ext == ".aspx")
{
//下面的代码中也就没有进行错误的处理了,主要是模拟,没有考虑其他的情况
//拿到请求资源的文件名(不包含后缀)
string cs = Path.GetFileNameWithoutExtension(url);
//得到当前程序的程序集
Assembly ass = Assembly.GetExecutingAssembly();
//拿到请求的文件对应的类
Type type = ass.GetType(ass.GetName().Name + "." + cs, true, true);
//创建对象,进行调用
IHttpHandel handel = Activator.CreateInstance(type) as IHttpHandel;
handel.ProcessRequest(context);
//上面几句话可以合并到一起,拿到程序集后直接CreateInstance();
}
else if (ext == ".ashx")
{
}
else//访问静态资源
{
//直接从磁盘中读取请求的资源文件
byte[] by = File.ReadAllBytes(url);
context.Response.Length = by.Length;
context.Response.Body = by; }
socket.Send(context.Response.Head);
socket.Send(context.Response.Body);
}
}

HttpApplication类

整个请求的模型基本上就差不多完成了,如果是请求的动态文件b.aspx,那么这个b.cs文件需要实现IHttpHandel接口,同时在规定的方法里面进行处理,代码如下:

 using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace IIS_Two
{
public class b:IHttpHandel {
public void ProcessRequest (HttpContext context)
{
//读取模板HTML文件
string html = File.ReadAllText("temp.html");
string temp = "<table border='1'>{0}</table>";
//读取文本文件类容,拼接一个table表格
using (StreamReader reader = new StreamReader("1.txt"))
{
string str = "";
List<string> list = new List<string>();
while ((str = reader.ReadLine()) != null)
{
string[] strs = str.Split(':');
list.Add("<tr><td>" + strs[] + "</td><td>" + strs[] + "</td></tr>"); }
temp = string.Format(temp, string.Join("", list));
}
//模板内容的替换
html = html.Replace("$body", temp);
byte[] by = Encoding.UTF8.GetBytes(html);
context.Response.Length = by.Length;
context.Response.Body = by;
}
}
}

b.cs文件,对应b.aspx资源

附上b.cs中用到的一个HTML模板和一个txt文件

 <!DOCTYPE html>

 <html lang="en" xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta charset="utf-8" />
<title></title>
</head>
<body>
$body
</body>
</html>

temp.html

 aaa:22
bbb:23
ccc:18

1.txt

整个简单的模拟就完成了,bug很多,这不重要。如果请求一个简单的html页面或者是jpg等图片都能够成功,当然这些文件要存在代码中所写的目录下面才行

用Socket来简单实现IIS服务器的更多相关文章

  1. Socket 实现简单的多线程服务器程序

    **********服务器端************* public class ServerSocket{ public static void main(String[] args) throws ...

  2. 第27篇 重复造轮子---模拟IIS服务器

    在写程序的时候,重复造轮子是程序员的一个大忌,很多人对重复造轮子持有反对的态度,但是我觉得这个造轮子的过程,是对于现有的知识的一个深入的探索的过程,虽然我们不可能把轮子造的那么的完善,对于现在有的东西 ...

  3. 运用socket实现简单的服务器客户端交互

    Socket解释: 网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个socket. Socket的英文原义是“孔”或“插座”.作为BSD UNIX的进程通信机制,取后一种意 ...

  4. Visual Studio写的项目在 IIS 服务器上运行的两种简单方法

    首先需要PC上开启了IIS服务,相关方法网上很多,也很简单 第一种:直接在项目中操作 1.创建一个项目,然后右击选中项目,右击,单击属性,打开项目属性标签页面 如图,选择Web标签,在服务器栏目中选中 ...

  5. IIS服务器简单搭建

     概况 系统:WIN10企业版 开发工具:VS2013 - VS2017 IIS版本:IIS6.0 安装 安装步骤如图: 这里需要注意一点信息服务器下默认web管理工具只有iis控制台勾上了,也就意味 ...

  6. 第37章 socket编程 之练习:实现简单的web服务器

    一.参考网址 1.linux C学习之实现简单的web服务器 2.C语言实现简单Web服务器(一)

  7. 自己动手模拟开发一个简单的Web服务器

    开篇:每当我们将开发好的ASP.NET网站部署到IIS服务器中,在浏览器正常浏览页面时,可曾想过Web服务器是怎么工作的,其原理是什么?“纸上得来终觉浅,绝知此事要躬行”,于是我们自己模拟一个简单的W ...

  8. socket.io简单入门(一.实现简单的图表推送)

    引子:随着nodejs蓬勃发展,虽然主要业务系统因为架构健壮性不会选择nodejs座位应用服务器.但是大量的内部系统却可以使用nodejs试水,大量的前端开发人员转入全堆开发也是一个因素. 研究本例主 ...

  9. 20145216 20145330 《信息安全系统设计基础》 实验五 简单嵌入式WEB 服务器实验

    20145216 20145330 <信息安全系统设计基础> 实验五 简单嵌入式WEB 服务器实验 实验报告封面 实验步骤 1.阅读理解源码 进入/arm2410cl/exp/basic/ ...

随机推荐

  1. 嵌入式操作系统VxWorks中网络协议存储池原理及实现

    嵌入式操作系统VxWorks中网络协议存储池原理及实现 周卫东 蔺妍 刘利强 (哈尔滨工程大学自动化学院,黑龙江 哈尔滨,150001) 摘  要  本文讨论了网络协议存储池的基本原理和在嵌入式操作系 ...

  2. spring 学习 二 IOC/DI

    中文名称:控制反转 英文名称:( Inversion of Control ) 1 控制反转作用: 一般在编写java程序时,需要程序员自己创建对象的实例,例如 A a=new A();语句,就是程序 ...

  3. 哈希与字典树与KMP

    hash讲解 主要记录hash的公式: ; i<=len; i++) { Hash[i]=(Hash[i-]*)%mod)%mod; } 求hash的公式是这个,怎么求一小段的hash值呢? ; ...

  4. tflite笔记

    固化模型 方法一:freeze_graph方法 把tf.train.write_graph()生成的pb文件与tf.train.saver()生成的chkp文件固化之后重新生成一个pb文件 with ...

  5. mysql知识积累

    验证mysql工作状态 systemctl status mysql.service 启动 sudo systemctl start mysql 停止 service mysql stop 重启mys ...

  6. 安装了nodejs后在命令行运行npm报错:Error: Cannot find module 'internal/util/types'

    报错如下图所示: 解决方法:删除目录“C:\Users\wls\AppData\Roaming\npm\node_modules”下的npm文件夹

  7. 好文推荐系列--------(3)GruntJS 在线重载 提升生产率至新境界

    好文原文地址:http://segmentfault.com/a/1190000000354555 本文将首先介绍grunt-markdown插件如何配合HTML模板使用,接着我将介绍如何使用grun ...

  8. BAT文件语法和技巧(bat文件的编写及使用)

    源文链接:http://www.jb51.net/article/5828.htm 比较有用的东比较有用的东西 首先,批处理文件是一个文本文件,这个文件的每一行都是一条DOS命令(大部分时候就好象我们 ...

  9. 模板引擎(smarty)知识点总结四

    /*   smarty 引入对象 */ require_once 'libs/Smarty.class.php';  require 'MySmarty.class.php';  $msma = ne ...

  10. 1143 Lowest Common Ancestor

    The lowest common ancestor (LCA) of two nodes U and V in a tree is the deepest node that has both U ...