gf/README.MD
2018-04-14 12:41:27 +08:00

18 KiB
Raw Blame History

介绍

gf(Go Frame)是一款为Web服务及网络服务开发设计的模块化、低耦合、轻量级、高性能的Go语言开发框架。 实现了通用的HTTP/TCP/UDP Server并提供了Web服务开发的系列核心组件 包括MVC、Cookie、Session、模板引擎、路由控制、配置管理、数据校验、数据库操作(ORM)等等, 并且提供了数十个实用开发模块包括缓存模块、日志模块、JSON模块、命令行模块、二进制模块、环境变量模块、并发安全容器、Goroutine池等等。

gf是开源的免费的基于MIT协议进行分发开源项目地址(gitee与github仓库保持实时同步) Gitee( https://gitee.com/johng/gf )Github( https://github.com/johng-cn/gf )

使用中有任何问题/建议欢迎加入技术QQ群交流116707870。如有优秀的gf框架使用案例欢迎联系作者将地址展示到项目库中您的牛逼将被世人所瞻仰。

特点

  1. 纯Go语言实现开源、免费、跨平台、高性能
  2. 充分利用Go语言特点采用模块化、低耦合设计
  3. 为Web服务及网络服务开发设计轻量级简便易用
  4. 精心设计的核心组件及数十个丰富实用的开发模块库;
  5. 功能强大的HttpServer及服务注册功能提高Web开发效率
  6. 丰富详尽的框架文档及专业的技术支持/讨论群,易于使用及维护;
  7. 更多特点请查阅框架手册( http://gf.johng.cn )

安装

go get -u gitee.com/johng/gf

使用

  1. 哈喽世界!

    package main
    
    import "gitee.com/johng/gf/g/net/ghttp"
    
    func main() {
        s := ghttp.GetServer()
        s.BindHandler("/", func(r *ghttp.Request){
            r.Response.Write("哈喽世界!")
        })
        s.Run()
    }
    
  2. 域名 & 多域名支持

    1. 自定义域名支持:
      package main
      
      import "gitee.com/johng/gf/g/net/ghttp"
      
      func main() {
          s := ghttp.GetServer()
          s.Domain("127.0.0.1").BindHandler("/", func(r *ghttp.Request) {
              r.Response.Write("127.0.0.1")   
          })
          s.Domain("localhost").BindHandler("/", func(r *ghttp.Request) {
              r.Response.Write("localhost")
          })
          s.Run()
      }
      
    2. 多域名绑定支持:
      package main
      
      import "gitee.com/johng/gf/g/net/ghttp"
      
      func main() {
          s := ghttp.GetServer()
          s.Domain("localhost").BindHandler("/", func(r *ghttp.Request) {
              r.Response.Write("localhost")
          })
          s.Domain("localhost1localhost2localhost3").BindHandler("/", func(r *ghttp.Request) {
              r.Response.Write("localhostx")
          })
          s.Run()
      }
      
  3. 服务注册

    gf框架提供了非常强大的服务注册方式这也是相对于其他框架如gin/beego/httprouter最突出的特点之一。

    gf框架提供了三种服务注册方式控制器注册、执行对象注册、回调函数注册具体介绍请查看官方文档(http://gf.johng.cn/494368)。服务注册方式比较:

    注册方式 使用难度 安全系数 执行性能 内存消耗
    控制器注册
    执行对象注册
    回调函数注册
    1. 控制器注册

      package main
      
      import (
          "gitee.com/johng/gf/g/net/ghttp"
          "gitee.com/johng/gf/g/frame/gmvc"
      )
      
      // 定义业务相关的控制器对象
      type ControllerUser struct {
          gmvc.Controller
      }
      // 定义操作逻辑 - 展示姓名
      func (c *ControllerUser) Name() {
          c.Response.Write("John")
      }
      
      // 定义操作逻辑 - 展示年龄
      func (c *ControllerUser) Age() {
          c.Response.Write("18")
      }
      
      func main() {
          // 绑定控制器到指定URI所有控制器的公开方法将会映射到指定URI末尾
          // 例如该方法执行后,查看效果可访问:
          // http://127.0.0.1:8199/user/name
          // http://127.0.0.1:8199/user/age
          s := ghttp.GetServer()
          s.BindController("/user", &ControllerUser{})
          s.SetPort(8199)
          s.Run()
      }
      
    2. RESTful控制器注册

      支持的HTTP Method: GET,PUT,POST,DELETE,PATCH,HEAD,CONNECT,OPTIONS,TRACE

      package main
      
      import (
          "gitee.com/johng/gf/g/net/ghttp"
          "gitee.com/johng/gf/g/frame/gmvc"
      )
      
      // 测试控制器
      type ControllerRest struct {
          gmvc.Controller
      }   
      
      // RESTFul - GET
      func (c *ControllerRest) Get() {
          c.Response.Write("RESTful HTTP Method GET")
      }
      
      // RESTFul - DELETE
      func (c *ControllerRest) Delete() {
          c.Response.Write("RESTful HTTP Method DELETE")
      }
      
      // 该方法无法映射,将会无法访问到
      func (c *ControllerRest) Hello() {
          c.Response.Write("Hello")
      }
      
      func main() {
          // 控制器公开方法中与HTTP Method方法同名的方法将会自动绑定映射
          s := ghttp.GetServer()
          s.BindControllerRest("/user.rest", &ControllerRest{})
          s.SetPort(8199)
          s.Run()
      }
      
    3. 其他注册方式

      由于服务注册方式功能相当丰富,其他注册方式请查看官方开发文档 - 服务注册章节(http://gf.johng.cn/494368)。

  4. 路由控制

    gf框架提供了自建的非常强大的路由控制功能支持流行的命名匹配规则及模糊匹配规则并提供了优秀的优先级管理机制。

    package main
    
    import "gitee.com/johng/gf/g/net/ghttp"
    
    func main() {
        s := ghttp.GetServer()
        s.BindHandler("/:name", func(r *ghttp.Request){
            r.Response.Write(r.Get("name"))
        })
        s.BindHandler("/:name/:action", func(r *ghttp.Request){
            r.Response.Write(r.Get("name"))
            r.Response.Write(r.Get("action"))
        })
        s.BindHandler("/:name/:any", func(r *ghttp.Request){
            r.Response.Write(r.Get("name"))
            r.Response.Write(r.Get("any"))
        })
        s.SetPort(8199)
        s.Run()
    }
    
    1. 命名匹配规则

      使用:name方式进行匹配(name为自定义的匹配名称)匹配成功后对应匹配参数会被解析为GET参数并传递给注册的服务使用。

      匹配示例1

      rule: /user/:user
      
      /user/john/profile        no match
      /user/                    no match
      /user/john                match
      /user/you                 match
      

      匹配示例2

      rule: /:name/action
      
      /john/name                no match
      /john/action              match
      /smith/info               no match
      /smith/info/age           no match
      /smith/action             match
      

      匹配示例3

      rule: /:name/:action
      
      /john/name                match
      /john/info                match
      /smith/info               match
      /smith/info/age           no match
      /smith/action/del         no match
      
    2. 模糊匹配规则

      使用*any方式进行匹配(any为自定义的匹配名称)一般常用语末尾匹配将会匹配URI随后所有的参数并将匹配参数解析为GET参数并传递给注册的服务使用。

      匹配示例1

      rule: /src/*path
      
      /src/                     match
      /src/somefile.go          match
      /src/subdir/somefile.go   match
      /user/                    no match
      /user/john                no match
      

      匹配示例2

      rule: /src/*path/:action
      
      /src/                     no match
      /src/somefile.go          no match
      /src/somefile.go/del      match
      /src/subdir/file.go/del   match
      

      匹配示例3

      rule: /src/*path/show
      
      /src/                     no match
      /src/somefile.go          no match
      /src/somefile.go/del      no match
      /src/somefile.go/show     match
      /src/subdir/file.go/show  match
      
    3. 路由优先级控制

      优先级控制最主要的是两点因素1、层级越深的规则优先级越高2、命名匹配比模糊匹配优先级高。 我们来看示例(左边的规则优先级比右边高):

       /user/name               >            /user/:action
       /:name/info              >            /:name/:action
       /:name/:action           >            /:name/*action
       /src/path/del            >            /src/path
       /src/path/del            >            /src/path/:action
       /src/path/*any           >            /src/path
       
  5. 数据库ORM

    1. 方法操作

      1. 获取ORM单例对象

        // 获取默认配置的数据库对象(配置名称为"default")
        db, err := gdb.Instance()
        // 获取配置分组名称为"user-center"的数据库对象
        db, err := gdb.Instance("user-center")
        
      2. 数据写入

        r, err := db.Insert("user", gdb.Map {
            "name": "john",
        })
        
      3. 数据查询(列表)

        list, err := db.GetAll("select * from user limit 2")
        
      4. 数据查询(单条)

        one, err := db.GetOne("select * from user limit 2")
        // 或者
        one, err := db.GetOne("select * from user where uid=1000")
        
      5. 数据保存

        r, err := db.Save("user", gdb.Map {
            "uid"  :  1,
            "name" : "john",
        })
        
      6. 批量操作

        // BatchInsert/BatchReplace/BatchSave 同理
        _, err := db.BatchInsert("user", gdb.List {
            {"name": "john_1"},
            {"name": "john_2"},
            {"name": "john_3"},
            {"name": "john_4"},
        }, 10)
        
      7. 数据更新/删除

        // db.Update/db.Delete 同理
        r, err := db.Update("user", gdb.Map {"name": "john"}, "uid=?", 10000)
        r, err := db.Update("user", "name='john'", "uid=10000")
        r, err := db.Update("user", "name=?", "uid=?", "john", 10000)
        

        注意参数域支持并建议使用预处理模式进行输入避免SQL注入风险。

    2. 链式操作

      1. 链式查询

        // 查询多条记录并使用Limit分页
        r, err := db.Table("user u").LeftJoin("user_detail ud", "u.uid=ud.uid").Fields("u.*, ud.site").Where("u.uid > ?", 1).Limit(0, 10).Select()
        // 查询符合条件的单条记录(第一条)
        r, err := db.Table("user u").LeftJoin("user_detail ud", "u.uid=ud.uid").Fields("u.*,ud.site").Where("u.uid=?", 1).One()
        // 查询字段值
        r, err := db.Table("user u").LeftJoin("user_detail ud", "u.uid=ud.uid").Fields("ud.site").Where("u.uid=?", 1).Value()
        // 分组及排序
        r, err := db.Table("user u").LeftJoin("user_detail ud", "u.uid=ud.uid").Fields("u.*,ud.city").GroupBy("city").OrderBy("register_time asc").Select()
        
      2. 链式更新/删除

        // 更新
        r, err := db.Table("user").Data(gdb.Map{"name" : "john2"}).Where("name=?", "john").Update()
        r, err := db.Table("user").Data("name='john3'").Where("name=?", "john2").Update()
        // 删除
        r, err := db.Table("user").Where("uid=?", 10).Delete()
        
      3. 链式写入/保存

        r, err := db.Table("user").Data(gdb.Map{"name": "john"}).Insert()
        r, err := db.Table("user").Data(gdb.Map{"uid": 10000, "name": "john"}).Replace()
        r, err := db.Table("user").Data(gdb.Map{"uid": 10001, "name": "john"}).Save()
        
      4. 链式批量写入

        r, err := db.Table("user").Data(gdb.List{
            {"name": "john_1"},
            {"name": "john_2"},
            {"name": "john_3"},
            {"name": "john_4"},
        }).Insert()
        

        可以指定批量操作中分批写入数据库的每批次写入条数数量:

        r, err := db.Table("user").Data(gdb.List{
            {"name": "john_1"},
            {"name": "john_2"},
            {"name": "john_3"},
            {"name": "john_4"},
        }).Batch(2).Insert()
        
      5. 链式批量保存

        r, err := db.Table("user").Data(gdb.List{
            {"uid":10000, "name": "john_1"},
            {"uid":10001, "name": "john_2"},
            {"uid":10002, "name": "john_3"},
            {"uid":10003, "name": "john_4"},
        }).Save()
        
    3. 事务操作

      开启事务操作可以通过执行db.Begin方法,该方法返回事务的操作对象,类型为*gdb.Tx,通过该对象执行后续的数据库操作,并可通过tx.Commit提交修改,或者通过tx.Rollback回滚修改。

      1. 开启事务操作
        if tx, err := db.Begin(); err == nil {
            fmt.Println("开启事务操作")
        }
        
        事务操作对象可以执行所有db对象的方法具体请参考API文档
      2. 事务回滚操作
        if tx, err := db.Begin(); err == nil {
            r, err := tx.Save("user", gdb.Map{
                "uid"  :  1,
                "name" : "john",
            })
            tx.Rollback()
            fmt.Println(r, err)
        }
        
      3. 事务提交操作
        if tx, err := db.Begin(); err == nil {
            r, err := tx.Save("user", gdb.Map{
                "uid"  :  1,
                "name" : "john",
            })
            tx.Commit()
            fmt.Println(r, err)
        }
        
      4. 事务链式操作 事务操作对象仍然可以通过tx.Table或者tx.From方法返回一个链式操作的对象,该对象与db.Table或者db.From方法返回值相同,只不过数据库操作在事务上执行,可提交或回滚。
        if tx, err := db.Begin(); err == nil {
            r, err := tx.Table("user").Data(gdb.Map{"uid":1, "name": "john_1"}).Save()
            tx.Commit()
            fmt.Println(r, err)
        }
        
        其他链式操作请参考上述链式操作章节。

文档

贡献

gf是开源的、免费的软件这意味着任何人都可以为其开发和进步贡献力量。gf的项目源代码目前同时托管在 Gitee 和 Github 平台上,您可以选择您喜欢的平台来 fork 项目和合并你的贡献两个平台的仓库将会保持即时的同步。我们非常欢迎有更多的朋友加入到gf框架的开发中来您为gf所做出的任何贡献都将会被记录到gf的史册中。