《DotNet Web应用单文件部署系列》二、打包wwwroot文件夹
在这篇文章中,你将学到web缓存规则,文件传输中用到的压缩格式,以及如何手写代码响应请求。最后还能学到快速打包wwwroot文件夹组件用法。
一、了解Response Header
当第一次加载程序时,浏览器将打开页面并下载所有的资源连接。假如页面没有错误返回都是正确那么就是返回文件数据和Http Status为200 -OK的状态
我们看下这个jquery.min.js文件Http请求对应的Response Header,这里会包含ETag值。HTTP内容如下:
ETag: 1d7a4ae31f17d74
ETag :HTTP响应头是资源的特定版本的标识符。这可以让缓存更高效,并节省带宽,因为如果内容没有改变,Web服务器不需要发送完整的响应。而如果内容发生了变化,使用ETag有助于防止资源的同时更新相互覆盖。如果给定URL中的资源更改,则一定要生成新的Etag值。 因此Etags类似于指纹,也可能被某些服务器用于跟踪。 比较etags能快速确定此资源是否变化,但也可能被跟踪服务器永久存留。
如果再次请求这个地址的话,浏览器将发送ETag到服务端,如果两个值没有变化,那么服务端会发送304状态到浏览器,那么浏览器将使用之前的资源而不是重新下载一份。
If-None-Match: 1d7a4ae31f17d74
将文件都缓存到了客户端,这样就提高了浏览器的响应性能,并且通过304状态,浏览器与服务端的请求流量得以减少。
加快浏览器的响应性能,还可以设置两种方案,
1)减少浏览器与服务端的请求次数;可以在响应头内设置Expires,Cache-Control两个参数。
Expires:响应头包含日期/时间, 即在此时候之后,响应过期。
Cache-Control:通用消息头字段,被用于在http请求和响应中,通过指定指令来实现缓存机制。缓存指令是单向的,这意味着在请求中设置的指令,不一定被包含在响应中。
2)压缩文件,减少返回流量,从而提高响应性能。
在HTTP 请求时,客户端会发送Accept-Encoding请求头给服务端,服务端就可以根据Accept-Encoding请求头匹配到压缩方式,将文件压缩后返回。
二、DotNet 5 手写代码响应请求
先放主代码
1 [HttpGet("_/js/jquery.min.js")]
2 public IActionResult __js_jquery_min_js()
3 {
4 if (SetResponseHeaders("4F252523D4AF0B478C810C2547A63E19") == false) { return StatusCode(304); }
5 const string s = "base64编码";
6 var bytes = UseCompressBytes(s);
7 return File(bytes, "text/javascript");
8 }
第一行,我们定义请求方式及请求地址。
第二行,我们将请求地址转成可用的方法名。
第四行,我们设置返回头,如果返回头有相同的ETAG值,就返回304。ETAG值可使用Hash值,如Md5。
第五行,获取文件的base64编码,注意文件先经Brotli 算法压缩,然后转成base64编码的。这样有效地减少文件大小。
第六行,我们尝试以压缩的编码返回。
第七行,我们返回文件内容及Content-Type类型。
SetResponseHeaders方法如下
1 SetResponseHeaders方法如下
2 private bool SetResponseHeaders(string etag)
3 {
4 if (Request.Headers["If-None-Match"] == etag) { return false; }
5 Response.Headers["Cache-Control"] = "max-age=315360000";
6 Response.Headers["Etag"] = etag;
7 Response.Headers["Date"] = DateTime.Now.ToString("r");
8 Response.Headers["Expires"] = DateTime.Now.AddYears(100).ToString("r");
9 return true;
10 }
我们先对比etag值是否相同,如果相同返回false。不同就设置etag值以及过期时间100年有效。
UseCompressBytes方法如下:
1 private byte[] UseCompressBytes(string s)
2 {
3 var bytes = Convert.FromBase64String(s);
4 var sp = Request.Headers["Accept-Encoding"].ToString().Replace(" ", "").ToLower().Split(',');
5 if (sp.Contains("br")) {
6 Response.Headers["Content-Encoding"] = "br";
7 } else {
8 using (MemoryStream stream = new MemoryStream(bytes)) {
9 using (BrotliStream zStream = new BrotliStream(stream, CompressionMode.Decompress)) {
10 using (var resultStream = new MemoryStream()) {
11 zStream.CopyTo(resultStream);
12 bytes = resultStream.ToArray();
13 }
14 }
15 }
16 if (sp.Contains("gzip")) {
17 Response.Headers["Content-Encoding"] = "gzip";
18 using (MemoryStream stream = new MemoryStream()) {
19 using (GZipStream zStream = new GZipStream(stream, CompressionMode.Compress)) {
20 zStream.Write(bytes, 0, bytes.Length);
21 zStream.Close();
22 }
23 bytes = stream.ToArray();
24 }
25 }
26 }
27 return bytes;
28 }
第一步,判断是否支持Brotli 算法压缩,如果支持就马上返回。
第二步,使用Brotli 算法解压。
第三步,判断是否支持gzip算法压缩,如果支持使用gzip算法压缩,然后返回。
第四步,返回原bytes。
三、快速压缩打包wwwroot文件夹
打包wwwroot文件夹很简单,但文件一个一个手写就会觉得很麻烦。这时就需要ToolGood.WwwRoot组件,在Nuget上直接获取。
3.1、快速上手
1)新建一个控制台应用程序,
2)从Nuget上引用ToolGood.WwwRoot组件,
3)添加以下代码
1 WwwRootSetting setting = new WwwRootSetting();
2 setting.NameSpace = "ToolGood.TextFilter.Controllers";
3 setting.InFolderPath = @"你的项目路径\wwwroot";
4 setting.OutFolderPath = @"你的项目路径\Controllers\wwwroot";
5
6 setting.ExcludeFileSuffixs.Add(".old.js");
7 setting.ExcludeFileSuffixs.Add(".map");
8 setting.BuildControllers();
4)运行控制台应用程序,就会生成相应的cs文件。
2.2、WwwRootSetting参数简介
NameSpace参数设置命名空间
InFolderPath参数指定wwwroot目录
OutFolderPath参数指定输出目录
ExcludeFileSuffixs参数依据文件后缀排除文件
ExcludeFiles参数 依据文件排除文件
2.3、其他相关
1)每个生成的文件都有 #if RELEASE 和 #endif,保证调试模式下不会被编译。
2)app.UseStaticFiles();前面添加 #if DEBUG 后面添加 #endif ,保证生成后不会使用本地静态文件。
3)程序更新后,静态文件过期问题。网上有很多成熟的方案,这里介绍一个最简单的方法,使用静态文件+程序版本号来解决,如:
<script src="_/js/ok.js?v=20210911"></script>
后记:
.net 5 单文件运行时会将dll文件释放到内存内,技术高超的人还是能从内存截取dll文件。提高dll文件反编译成本,我们可以将dll文件混淆。
dll文件混淆后会带来一系列问题,主要是操作麻烦,所以下几篇将介绍dll文件混淆、VS调时使用项目源文件,只在生成时使用dll混淆文件。
《DotNet Web应用单文件部署系列》二、打包wwwroot文件夹的更多相关文章
- 《DotNet Web应用单文件部署系列》一、pubxml文件配置
很多人想用DotNet开发软件赚点外快子补添家用,但心里总放不下心来,担心被人破解了.好消息是去年发布的DotNet 5支持单文件部署,不同于DotNet 3运行时将文件释放到临时文件夹内,DotNe ...
- linux内核中的文件描述符(二)--socket和文件描述符
http://blog.csdn.net/ce123_zhouwei/article/details/8459730 Linux内核中的文件描述符(二)--socket和文件描述符 Kernel ve ...
- 《DotNet Web应用单文件部署系列》三、混淆dll文件
众所周知,C#编译后的dll文件可被反编译,网上搜索"C# 反编译"会出现一大堆资料.为了提高反编译成本,我们必须对dll文件进行混淆处理. 目前,C#混淆工具很多,我推荐obfu ...
- C# 文件读写系列二
读取文件原则上非常简单,但它不是通过FileInfo和DirectoryInfo来完成的,关于FileInfo和DirectoryInfo请参考C# 文件操作系列一,在.Net Framework4. ...
- Python小白学习之路(二十)—【打开文件的模式二】【文件的其他操作】
打开文件的模式(二) 对于非文本文件,我们只能使用b模式,"b"表示以字节的方式操作(而所有文件也都是以字节的形式存储的,使用这种模式无需考虑文本文件的字符编码.图片文件的jgp格 ...
- Windows phone开发之文件夹与文件操作系列(一)文件夹与文件操作
Windows phone7中文件的存储模式是独立的,即独立存储空间(IsolatedStorage).对文件夹与文件操作,需要借助IsolatedStorageFile类. IsolatedStor ...
- ASP.NET Core部署系列二:发布到CentOS上
前言: 在上一节中,通过一系列的步骤,已经将项目部署到IIS上,虽然遇到了一些问题,但最终解决并成功运行了.而在这一节中,将尝试通过linux系统的环境下,部署项目,实现Net Core跨平台的亮点. ...
- web前端知识大纲:系列二 css篇
web前端庞大而复杂的知识体系的组成:html.css和 javascript 二.css 1.CSS选择器 CSS选择器即通过某种规则来匹配相应的标签,并为其设置CSS样式,常用的有类选择器.标签选 ...
- 小白日记40:kali渗透测试之Web渗透-SQL手工注入(二)-读取文件、写入文件、反弹shell
SQL手工注入 1.读取文件[load_file函数] ' union SELECT null,load_file('/etc/passwd')--+ burpsuite 2.写入文件 ' unio ...
随机推荐
- 浅谈Java类中的变量初始化顺序
一.变量与构造器的初始化顺序 我们知道一个类中具有类变量.类方法和构造器(方法中的局部变量不讨论,他们是在方法调用时才被初始化),当我们初始化创建一个类对象时,其初始化的顺序为:先初始化类变量,再执行 ...
- SoundPool概率性无声问题
public class SoundManager { private static SoundManager instance; private SoundPool mSoundPool; priv ...
- uTools电脑软件快速启动工具
uTools电脑软件快速启动工具 http://www.autoahk.com/archives/16112 https://gitee.com/weiyunw ...
- 用QT写的简单Todo记事本-附源码(浮动窗口)
去年边学边写了搞了很久, 已经好久没继续开发了, 先放出来供大家参考吧. 发现自己的学习能力还是不错的. 技术点: 使用QT, QML技术 代码参考: https://github.com/cnscu ...
- linux笔记1(不全,无图版)随笔
1.ls 查看当前目录下的所有内容 黑色的是文件,蓝色的是文件夹,也就是目录 2.rm -f anaconda-ks. cfg 彻底删除文件(如不确定,则需要先保存备份,也就是快照) 3.ifconf ...
- Apache解析漏洞
多解析特性 在Apache1.x,2.x中Apache 解析文件的规则是从右到左开始判断解析,如果后缀名为不可识别文件解析,就再往左判断.因此对于apache而言,一个test.php.qwea文件依 ...
- BUUCTF-[极客大挑战 2019]BabySQL(联合注入绕过waf)+[极客大挑战 2019]LoveSQL(联合注入)
BUUCTF-[极客大挑战 2019]BabySQL(联合注入绕过waf) 记一道联合注入的题,这道题存在过滤. 经过手工的测试,网站会检验用户名和密码是否都存在,如果在用户名处插入注入语句,语句后面 ...
- AbstractRoutingDataSource -- Spring提供的轻量级数据源切换方式
AbstractRoutingDataSource 只支持单库事务,也就是说切换数据源要在开启事务之前执行. spring DataSourceTransactionManager进行事务管理,开启事 ...
- Adaptive AUTOSAR 学习笔记 15 - 持久化 Persistency
本系列学习笔记基于 AUTOSAR Adaptive Platform 官方文档 R20-11 版本 AUTOSAR_EXP_PlatformDesign.pdf.作者:Zijian/TENG 原文地 ...
- C# 调用DOS 命令
class NetWorkDeviceInfo { public static string GetDeviceInfo() { System.Diagnostics.Process p = new ...