Go语言中的装饰器模式:优雅增强功能,提升代码可维护性

一、GoLang装饰器

1、简单timer装饰器
package main

import (
    "fmt"
    "time"
)

func timer(fn func()) func() {
    return func() {
        startTime := time.Now().Unix()
        fn()
        endTime := time.Now().Unix()
        fmt.PrintIn("运行时间:",endTime - startTime)
    }
}

func testFunc() {
    fmt.PrintIn("运行 testFunc")
    time.Sleep(time.Second * 2)
}

func main() {
    testFunc := timer(testFunc)
    testFunc()
}
/*
运行 testFunc
运行时间: 2
 */
2、项目中认证实现
package main

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

// DecoratorHandler 是一个用于包装 http.HandlerFunc 的装饰器类型
type DecoratorHandler func(http.HandlerFunc) http.HandlerFunc

// MiddlewareHandlerFunc 将多个 DecoratorHandler 应用于给定的 http.HandlerFunc,并返回新的 http.HandlerFunc。
// 这个函数的目的是创建一个中间件链,每个中间件都能够修改或验证请求。
func MiddlewareHandlerFunc(hp http.HandlerFunc, decors ...DecoratorHandler) http.HandlerFunc {
	// 遍历所有的 DecoratorHandler,并将其应用于 hp
	for _, fn := range decors {
		dp := fn
		hp = dp(hp) // dp(hp) => VerifyHeader(Pong)
	}
	return hp
}

// VerifyHeader 是一个装饰器函数,用于验证请求中的 token 是否为空。
// 如果 token 为空,将返回错误响应;否则,将调用下一个处理函数。
func VerifyHeader(h http.HandlerFunc) http.HandlerFunc {
	return func(w http.ResponseWriter, r *http.Request) {
		token := r.URL.Query().Get("token")
		if token == "" {
			fmt.Fprintf(w, r.URL.Path+" VerifyHeader:token不能为空")
			return
		}
		h(w, r)
	}
}

// VerifyHeader2 是另一个装饰器函数,用于验证请求中的 token 是否为 "mimi"。
// 如果 token 不为 "mimi",将返回错误响应;否则,将调用下一个处理函数。
func VerifyHeader2(h http.HandlerFunc) http.HandlerFunc {
	return func(w http.ResponseWriter, r *http.Request) {
		token := r.URL.Query().Get("token")
		if token != "mimi" {
			fmt.Fprintf(w, r.URL.Path+" VerifyHeader:token必须为:mimi")
			return
		}
		h(w, r)
	}
}

// Pong 是一个简单的 http.HandlerFunc,用于返回成功响应。
func Pong(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, r.URL.Path+"--> 验证通过,返回数据")
	return
}

func main() {
	// 在 "/test" 路径上注册 MiddlewareHandlerFunc,应用 VerifyHeader 和 VerifyHeader2 装饰器
	// http://127.0.0.1:8080/test?token=mimi
	http.HandleFunc("/test", MiddlewareHandlerFunc(Pong, VerifyHeader, VerifyHeader2))

	// 启动 HTTP 服务器,监听在 8080 端口
	err := http.ListenAndServe(":8080", nil)
	if err != nil {
		log.Fatal("ListenAndServe: ", err)
	}
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Go语言,可以使用`os/exec`包来调用系统程序。下面是一个简单的例子,演示如何使用`os/exec`包来调用Windows系统的`calc.exe`程序: ```go package main import ( "fmt" "os/exec" ) func main() { cmd := exec.Command("calc.exe") err := cmd.Start() if err != nil { fmt.Println("Error starting calculator:", err) return } fmt.Println("Calculator started successfully.") err = cmd.Wait() if err != nil { fmt.Println("Error waiting for calculator to exit:", err) return } fmt.Println("Calculator exited.") } ``` 这段代码,`exec.Command`函数用于创建一个`cmd`变量,它表示要执行的命令。在这个例子,我们将命令设置为`calc.exe`,也就是Windows系统自带的计算器程序。 `cmd.Start()`函数用于启动命令。如果启动失败,会返回一个非空的错误值。如果启动成功,则可以通过`cmd.Wait()`函数等待命令执行完毕。如果等待期间发生错误,也会返回一个非空的错误值。 最后,这段代码会输出一些信息,以告诉用户命令是否成功执行。如果启动失败,会输出一条错误信息。如果命令执行成功,会输出一条成功信息和一条命令退出的信息。 ### 回答2: 要使用Golang编写一段代码来调用系统的calc.exe程序,您可以使用`exec.Command`函数来实现。以下是一个简单的代码示例: ```go package main import ( "fmt" "os/exec" ) func main() { cmd := exec.Command("calc.exe") err := cmd.Start() if err != nil { fmt.Println("无法启动calc.exe程序:", err) return } err = cmd.Wait() if err != nil { fmt.Println("calc.exe程序执行错误:", err) return } fmt.Println("calc.exe程序已成功执行") } ``` 在代码,我们导入了`os/exec`包,该包可以用于执行外部命令。然后,我们使用`exec.Command`函数创建一个命令对象,并指定要执行的命令,这里是calc.exe。接下来,我们使用`cmd.Start`方法来启动该命令,并检查是否有错误发生。最后,我们使用`cmd.Wait`方法等待calc.exe程序执行完毕,并再次检查是否有错误发生。 请注意,这段代码假设您的操作系统是Windows,并且calc.exe程序位于系统的可执行文件路径。如果您的操作系统或calc.exe程序路径不同,您需要相应地进行修改。 希望这个代码示例对您有帮助! ### 回答3: 使用golang语言调用系统的calc.exe程序可以使用os包的Exec函数来实现。具体代码如下: ```go package main import ( "fmt" "os" "os/exec" ) func main() { cmd := exec.Command("calc.exe") err := cmd.Start() if err != nil { fmt.Println("启动calc.exe程序失败:", err) return } err = cmd.Wait() if err != nil { fmt.Println("等待calc.exe程序退出失败:", err) } } ``` 首先,我们导入了fmt、os和os/exec包。然后,在main函数,创建一个cmd变量,通过exec.Command函数来创建一个表示calc.exe程序的命令。 接下来,我们通过cmd.Start()方法启动calc.exe程序,并将启动结果赋值给err变量。如果启动失败,err将不为nil,我们将打印启动失败的错误信息并返回。 如果启动成功,我们使用cmd.Wait()方法等待calc.exe程序退出,并将等待结果赋值给err变量。如果等待失败,err将不为nil,我们将打印等待失败的错误信息。 这段代码的作用是调用系统的calc.exe程序,并等待calc.exe程序退出。如果你的计算机上存在calc.exe程序,执行这段代码将启动计算器程序,并等待计算器程序退出。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值