惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

SecWiki News
SecWiki News
I
InfoQ
The Cloudflare Blog
人人都是产品经理
人人都是产品经理
博客园 - Franky
T
Tailwind CSS Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
量子位
博客园_首页
罗磊的独立博客
V
V2EX
李成银的技术随笔
大猫的无限游戏
大猫的无限游戏
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
T
True Tiger Recordings
Vercel News
Vercel News
Cyberwarzone
Cyberwarzone
Cisco Talos Blog
Cisco Talos Blog
F
Fox-IT International blog
D
Darknet – Hacking Tools, Hacker News & Cyber Security
M
Microsoft Research Blog - Microsoft Research
Know Your Adversary
Know Your Adversary
爱范儿
爱范儿
The Register - Security
The Register - Security
G
Google Developers Blog
The Hacker News
The Hacker News
Malwarebytes
Malwarebytes
S
Securelist
博客园 - 三生石上(FineUI控件)
Jina AI
Jina AI
T
Threat Research - Cisco Blogs
T
The Exploit Database - CXSecurity.com
S
SegmentFault 最新的问题
博客园 - 叶小钗
F
Fortinet All Blogs
Apple Machine Learning Research
Apple Machine Learning Research
宝玉的分享
宝玉的分享
博客园 - 聂微东
T
Threatpost
博客园 - 【当耐特】
D
Docker
P
Privacy & Cybersecurity Law Blog
www.infosecurity-magazine.com
www.infosecurity-magazine.com
G
GRAHAM CLULEY
V
Visual Studio Blog
C
Cisco Blogs
IT之家
IT之家
S
Security Archives - TechRepublic
Latest news
Latest news
阮一峰的网络日志
阮一峰的网络日志

ZDDHUB

PixelsMeasure 开发第二年总结 PixelsMeasure 开发一年总结 Swift/SwiftUI 踩坑记 为什么说 GPT 利好程序员 ChatGPT 编程实现 Web 数字水印 Web 数字水印探究 Micro Frontends for Mobile URL 加载系统(URL Loading System) Protocol Buffers GraphQL 从 0 到 1 开发一款 IOS 应用 - Swift MV* 软件设计架构 学习一个新技巧需要多久? 不停机数据库迁移 Rspec 如何 mock update 方法更新自己? Rails 使用 mysql2 出现的段错误 使用 Docker-compose 部署 Rails 应用到生产环境 Cocoa troubleshooting 独孤九剑 Dit (0x05) - 终端篇 Gem-based Jekyll theme 开发小记 Miscellaneous 前端手记 TodoMVC 之 Redux 篇 前端手记 TodoMVC 之 Server 篇 前端手记 TodoMVC 之 React 篇 前端手记 TodoMVC 之 CSS 篇 独孤九剑 Dit (0x04) - 测试篇 独孤九剑 Dit (0x03) - 缓存篇 英语小抄 LLDB debug Golang Make mistakes 大牛俱乐部上线啦 独孤九剑 Dit (0x02) - 数据结构篇 独孤九剑 Dit (0x01) - 总决 独孤九剑 Dit (0x00) - 我为什么要做 Dit 零值强制类型转换的使用 终端颜色输出重定向 Go语法简略 - 正则表达式 Makefile Go语法简略 - 依赖注入 Go语法简略 - web应用框架 Go语法简略 - 反射 Go语法简略 - 面向对象 Go语法简略 - 方法和接口 Go语法简略 - goroutine Go语法简略 - 基础篇 大牛 | 轻松科研 为 Android Studio 创建图标 未来这几年 Shell Git Vim 金庸答百问 论拖延症 Flex, A fast scanner generator 有理想的人 从虚拟到现实 常用视频转接口 Recognizer configuration on CentOS 整个世界清静了 《Python源码剖析》读书笔记
Go语法简略 - Duck框架探索
2015-07-11 · via ZDDHUB

这个周末,《奔跑吧,兄弟》没有了,《那时那你》看完了,《万万没想到》剧组也没有新的项目,挠人的《花千骨》也要等到周日才更新,这是一个剧荒的周末,没办法,只能看看代码了。

本文打算参考Martini进一步实现我的[Duck框架]。

Duck 计划

想想,一个经典的web框架,都有什么行为呢?web应用是通过http协议实现的,客户端通常由浏览器实现,web框架处在服务器端,主要完成对服务器请求的应答任务。首先,我希望它能自动解析路由,最好支持REST风格API的实现,可以方便的添加丰富的中间件,具有日志功能。能完美的支持MVC/MVVC结构,能解析静态文件,支持session功能和数据库访问能力。

打算支持路由,中间件处理,日志。其它使用中间件注入。

先研究一下Go的http包是如何实现的。

Go http实现

Go在net/http包里实现了http协议,http使用TCP实现:在服务器端监听,请求到来时新建一个线程,响应客户端的请求。想知道更多的细节点这里:build-web-application-with-golang

有两点需要说明一下:

  • Go的变量和结构可以重名。😒,好吧,第一次知道这个,请原谅我的无知。
// A Server defines parameters for running an HTTP server.
// The zero value for Server is a valid configuration.
type Server struct {
  Addr           string        // TCP address to listen on, ":http" if empty
  Handler        Handler       // <- 看这里,看这里 handler to invoke, http.DefaultServeMux if nil
  ...
}
  • 在处理每个request时,调用的handler的ServeHTTP方法
serverHandler{c.server}.ServeHTTP(w, w.req)

// 具体实现如下,其中sh.srv.Handler就是我们传给http.ListenAndServe的第二个参数
func (sh serverHandler) ServeHTTP(rw ResponseWriter, req *Request) {
handler := sh.srv.Handler
if handler == nil { // 为空时默认使用DefaultServeMux
  handler = DefaultServeMux
}
if req.RequestURI == "*" && req.Method == "OPTIONS" {
  handler = globalOptionsHandler{}
}
handler.ServeHTTP(rw, req) // 实际处理请求的地方
}

DefaultServeMux是ServeMux对象的实例,ServeMux就是一个内置的路由器,最后绕了一圈:

// ServeHTTP calls f(w, r).
func (f HandlerFunc) ServeHTTP(w ResponseWriter, r *Request) {
  f(w, r) // 调用我们传过来的匿名方法
}

可以这样理解web路由:通过给定路由的pattern,把用户定义的匿名方法,注入到合适的服务环境中(w,r)环境,来实现用户希望的操作。

接下来就开始实现我们的Web框架吧。

最简单的实现

package main

import (
  "fmt"
  "net/http"
  // "reflect"
)

type Handler interface{}

type Duck struct {
  *Injector
  handlers []Handler // handler all fanc
  IP       string
  Port     string
}

func Incubate() *Duck {
  return &Duck{Injector: New(), IP: "", Port: "3030"}
}

func (d *Duck) Run() {
  fmt.Println("[Duck] listening on", d.IP+":"+d.Port)
  http.ListenAndServe(d.IP+":"+d.Port, d) // 加上我们的
}

// 实现ServeHTTP方法,作为一个http.Handler被http server调用
// 每次请求调用一次
func (d *Duck) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  // 一次请求后的响应环境
  fmt.Println("[Duck] start ServeHTTP")

  // 设置环境
  d.SetMap(w)
  d.SetMap(r)

  // 根据request的信息,调用不同的handler方法,实现用户定义的功能
  for i := 0; i < len(d.handlers); i++ {

    if ret, err := d.Invoke(d.handlers[i]); err == nil {
      if len(ret) != 0 {
        fmt.Fprintf(w, ret[0].Interface().(string))
      }
    } else {
      fmt.Println(err)
    }
  }

  fmt.Println("[Duck] end   ServeHTTP")
}

func (d *Duck) Get(pattern string, handler Handler) {
  fmt.Println("[Duck] Get")
  d.handlers = append(d.handlers, handler)
}

上面的内容很简单,但是原理清楚,接下来的事情就容易了,通过不同的pattern和requst method来选用不同的handler,整个框架的核心部分就有了。

这是我实现的Duck雏形,zddhub/duck。还不够完善,但是自认为原理很清楚。有人愿意一起做吗?

如果你喜欢这篇文章,欢迎赞赏作者以示鼓励