热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

golang视角下的设计模式

单利模式funcNewSingleton()*singleton{ifinstancenil{instance&single

单利模式

func NewSingleton() *singleton {
   if instance == nil {
        instance = &singleton{}
   }
   return instance
}
func NewSingleton() *singleton {
   l.Lock()                   // lock
   defer l.Unlock()
   if instance == nil {  // check
       instance = &singleton{}
   }
   return instance
}
func NewSingleton() *singleton {
   if instance == nil {    // check
       l.Lock()            // lock
       defer l.Unlock()  
       if instance == nil {    // check
           instance = &singleton{}
       }
   }
   return instance
}
func NewSingleton() *singleton {
   if atomic.LoadUInt32(&initialized) == 1 {
       return instance
   }
   mu.Lock()
   defer mu.Unlock()
   if initialized == 0 {
       instance = &singleton{}
       atomic.StoreUint32(&initialized, 1)
   }
   return instance
}
func NewSingleton() *singleton {
   once.Do(func() {
       instance = &singleton{}
   })
   return instance
}

工厂模式:

    工厂根据条件产生不同功能的类。工厂模式使用经常使用在替代new的场景中,让工厂统一根据不同条件生产不同的类。工厂模式在解耦方面将使用者和产品之间的依赖推给了工厂,让工厂承担这种依赖关系。工厂模式又分为简单工厂,抽象工厂。golang实现一个简单工厂模式如下:

package main
import (
   "fmt"
)
type Op interface {
   getName() string
}
type A struct {
}
type B struct {
}
type Factory struct {
}
func (a *A) getName() string {
   return "A"
}
func (b *B) getName() string {
   return "B"
}
func (f *Factory) create(name string) Op {
   switch name {
   case `a`:
       return new(A)
   case `b`:
       return new(B)
   default:
       panic(`name not exists`)
   }
   return nil
}
func main() {
   var f = new(Factory)
   p := f.create(`a`)
   fmt.Println(p.getName())
   p = f.create(`b`)
   fmt.Println(p.getName())
}

依赖注入:

    具体含义是:当某个角色(可能是一个实例,调用者)需要另一个角色(另一个实例,被调用者)的协助时,在传统的程序设计过程中,通常由调用者来创建被调用者的实例。但在这种场景下,创建被调用者实例的工作通常由容器(IoC)来完成,然后注入调用者,因此也称为依赖注入。Golang利用函数f可以当做参数来传递,同时配合reflect包拿到参数的类型,然后根据调用者传来的参数和类型匹配上之后,最后通过reflect.Call()执行具体的函数。下面的代码来自:https://www.studygolang.com/articles/4957 这篇文章上。

package main
import (
   "fmt"
   "reflect"
)
var inj *Injector
type Injector struct {
   mappers map[reflect.Type]reflect.Value // 根据类型map实际的值
}
func (inj *Injector) SetMap(value interface{}) {
   inj.mappers[reflect.TypeOf(value)] = reflect.ValueOf(value)
}
func (inj *Injector) Get(t reflect.Type) reflect.Value {
   return inj.mappers[t]
}
func (inj *Injector) Invoke(i interface{}) interface{} {
   t := reflect.TypeOf(i)
   if t.Kind() != reflect.Func {
       panic("Should invoke a function!")
   }
   inValues := make([]reflect.Value, t.NumIn())
   for k := 0; k        inValues[k] = inj.Get(t.In(k))
   }
   ret := reflect.ValueOf(i).Call(inValues)
   return ret
}
func Host(name string, f func(a int, b string) string) {
   fmt.Println("Enter Host:", name)
   fmt.Println(inj.Invoke(f))
   fmt.Println("Exit Host:", name)
}
func Dependency(a int, b string) string {
   fmt.Println("Dependency: ", a, b)
   return `injection function exec finished ...`
}
func main() {
   // 创建注入器
   inj = &Injector{make(map[reflect.Type]reflect.Value)}
   inj.SetMap(3030)
   inj.SetMap("zdd")
   d := Dependency
   Host("zddhub", d)
   inj.SetMap(8080)
   inj.SetMap("www.zddhub.com")
   Host("website", d)
}

装饰器模式

   

 装饰器模式:允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能。我们使用最为频繁的场景就是http请求的处理:对http请求做COOKIE校验。

package main
import (
   "fmt"
   "log"
   "net/http"
)
func autoAuth(h http.HandlerFunc) http.HandlerFunc {
   return func(w http.ResponseWriter, r *http.Request) {
       COOKIE, err := r.COOKIE("Auth")
       if err != nil || COOKIE.Value != "Authentic" {
           w.WriteHeader(http.StatusForbidden)
           return
       }
       h(w, r)
   }
}
func hello(w http.ResponseWriter, r *http.Request) {
   fmt.Fprintf(w, "Hello, World! "+r.URL.Path)
}
func main() {
   http.HandleFunc("/hello", autoAuth(hello))
   err := http.ListenAndServe(":5666", nil)
   if err != nil {
       log.Fatal("ListenAndServe: ", err)
   }
}

推荐阅读
  • golang 解析磁力链为 torrent 相关的信息
    其实通过http请求已经获得了种子的信息了,但是传播存储种子好像是违法的,所以就存储些描述信息吧。之前python跑的太慢了。这个go并发不知道写的有没有问题?!packag ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • 从零学Java(10)之方法详解,喷打野你真的没我6!
    本文介绍了从零学Java系列中的第10篇文章,详解了Java中的方法。同时讨论了打野过程中喷打野的影响,以及金色打野刀对经济的增加和线上队友经济的影响。指出喷打野会导致线上经济的消减和影响队伍的团结。 ... [详细]
  • 集合的遍历方式及其局限性
    本文介绍了Java中集合的遍历方式,重点介绍了for-each语句的用法和优势。同时指出了for-each语句无法引用数组或集合的索引的局限性。通过示例代码展示了for-each语句的使用方法,并提供了改写为for语句版本的方法。 ... [详细]
  • 目录在Go语言项目中使用Zap日志库介绍默认的GoLogger日志库实现GoLogger设置Logger使用LoggerLogger的运行GoLogger的优势和劣势优势劣势Ube ... [详细]
  • Go冒泡排序练习
    package main要求:随机生成5个元素的数组,并使用冒泡排序对其排序  从小到大思路分析:随机数用mathrand生成为了更好 ... [详细]
  • Go 快速入门指南命令行参数
    命令行参数个数调用os包即可。获取参数个数,遍历参数packagemainimport(fmtos)funcmain(){fmt.Printf(Numberofargsi ... [详细]
  • 集成第三方库,自检测读取配置文件。文件读取,结构体定义,接口实现,错误返回,库解析,适合新同学练手。思路文件读取获取字节流文件类型分析,确定解析api集成第三方解析api管理器定义 ... [详细]
  • 小编给大家分享一下Golang端口复用测试的实现方法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有 ... [详细]
  • 成功安装Sabayon Linux在thinkpad X60上的经验分享
    本文分享了作者在国庆期间在thinkpad X60上成功安装Sabayon Linux的经验。通过修改CHOST和执行emerge命令,作者顺利完成了安装过程。Sabayon Linux是一个基于Gentoo Linux的发行版,可以将电脑快速转变为一个功能强大的系统。除了作为一个live DVD使用外,Sabayon Linux还可以被安装在硬盘上,方便用户使用。 ... [详细]
  • position属性absolute与relative的区别和用法详解
    本文详细解读了CSS中的position属性absolute和relative的区别和用法。通过解释绝对定位和相对定位的含义,以及配合TOP、RIGHT、BOTTOM、LEFT进行定位的方式,说明了它们的特性和能够实现的效果。同时指出了在网页居中时使用Absolute可能会出错的原因,即以浏览器左上角为原始点进行定位,不会随着分辨率的变化而变化位置。最后总结了一些使用这两个属性的技巧。 ... [详细]
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • C++语言入门:数组的基本知识和应用领域
    本文介绍了C++语言的基本知识和应用领域,包括C++语言与Python语言的区别、C++语言的结构化特点、关键字和控制语句的使用、运算符的种类和表达式的灵活性、各种数据类型的运算以及指针概念的引入。同时,还探讨了C++语言在代码效率方面的优势和与汇编语言的比较。对于想要学习C++语言的初学者来说,本文提供了一个简洁而全面的入门指南。 ... [详细]
author-avatar
等着日落看日出222
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有