百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

如何从 PHP 过渡到 Golang?(php转golang)

wptr33 2025-05-08 06:55 25 浏览

我是 PHP 开发者,转 Go 两个月了吧,记录一下使用Golang怎么一步步开发新项目。

本着有坑填坑,有错改错的宗旨,从零开始,开始学习。因为我司没有专门的Golang大牛,所以我也只能一步步自己去摸索,我也表示很无奈。项目只需要提供api,所以狠狠地百度了一下,决定选用Gin框架,数据库选用的是MySql所以orm框架使用的Gorm,Redis使用的是Go-redis。

1.初始化框架:

go mod init gin

go get -u github.com/gin-gonic/gin

2.安装Gorm

go get -u github.com/jinzhu/gorm

3.安装Go-redis

go get -u github.com/go-redis/redis

4.本着mvc思想,创建项目基本目录:

config 配置文件

controllers 控制器

core 核心文件

logics 逻辑处理封装

models sql处理

routers 路由

以上是初始化开发项目的整个流程,第一次开发拼装框架有点麻烦,但是以后就顺手多了。现在我们来封装一下Mysql连接和Redis连接:

Mysql连接:

var db *gorm.DB

func Connection() (*gorm.DB) {
   db, err := gorm.Open("mysql",conf.DbUser + ":" + conf.DbPassword + "@(" + conf.DbHost + ":" + conf.DbPort + ")/" + conf.DbName + "?charset=utf8mb4&parseTime=True&loc=Local")
   // gorm1禁用表名加s方法
   db.SingularTable(true)
   db.LogMode(true)
   db.SetLogger(tools.Logger())
   db.DB().SetMaxIdleConns(10)
   db.DB().SetMaxOpenConns(100)
   if err !=  nil {
      panic(err)
   }

   return db
}

Redis连接:

var RedisClient *redis.Client

func init(){
   NewClient()
}

func NewClient() *redis.Client {
   if RedisClient != nil {
      return RedisClient
   }
   RedisClient = redis.NewClient(&redis.Options{
      Addr: conf.RedisAddr,
      Password: conf.RedisPassword,
      DB:   conf.RedisDb,
   })
   _, err := RedisClient.Ping().Result()
   if err != nil {
      //logs.Error("redis connection failed: ", err.Error())
   }
   return RedisClient
}

func RedisSet(key string, value interface{}, expire int) error {
   if expire > 0 {
      err := RedisClient.Do("SET", key, value, "EX", expire).Err()
      if err != nil {
         //logs.Error("RedisSet Error! key:", key, "Details:", err.Error())
         return err
      }
   } else {
      err := RedisClient.Do("SET", key, value).Err()
      if err != nil {
         //logs.Error("RedisSet Error! key:", key, "Details:", err.Error())
         return err
      }
   }

   return nil
}

func RedisKeyExists(key string) (bool, error) {
   ok, err := RedisClient.Do("EXISTS", key).Bool()
   return ok, err
}

func RedisGet(key string) (string, error) {
   value, err := RedisClient.Do("GET", key).String()
   if err != nil {
      return "", nil
   }
   return value, nil
}

func RedisDel(key string) error {
   err := RedisClient.Do("DEL", key).Err()
   if err != nil {
      //logs.Error("RedisDel Error! key:", key, "Details:", err.Error())
   }
   return err
}

下面我展示一下api的demo:

controllers:

// 获取列表
func GetBannerList(c*gin.Context){
   maps := make(map[string]interface{})
   // 页码
   page,_ := strconv.Atoi(c.DefaultQuery("page","1"))
   types,_ := strconv.Atoi(c.DefaultQuery("type","1"))
   size := 15
   maps["type"] = types
   total := models.GetBannerWhereCount(maps)
   pages := math.Ceil(float64(total) / float64(size))
   data := models.GetBannerWhereList(page,size,maps)
   var ViewBanners []models.ViewBanner
   var viewBanner models.ViewBanner
   for _,value := range data{
      Pid, _ := strconv.Atoi(value.Pid)
      arr := models.GetRowAdminUser(Pid,"")
      if len(arr) == 0{
         continue
      }
      viewBanner.ID = value.ID
      viewBanner.Img = value.Img
      viewBanner.Type = value.Type
      viewBanner.Abstract = value.Abstract
      viewBanner.UpdateTime = value.UpdateTime
      viewBanner.Status = value.Status
      viewBanner.PUserName = arr[0].Account
      ViewBanners = append(ViewBanners,viewBanner)
   }
   c.JSON(http.StatusOK,gin.H{
      "code" : 200,
      "message" : "SUCCESS!",
      "data" : ViewBanners,
      "page" : page,
      "pages" : pages,
   })
}

models:

type Banner struct {
   ID                      int `json:"id"`
   Type               int `json:"type"`
   Img                 string `json:"img"`
   Abstract            string `json:"abstract"`
   Status               int `json:"status"`
   Pid                   string `json:"pid"`
   CreateTime            string `json:"create_time"`
   UpdateTime            string `json:"update_time"`
}

type ViewBanner struct {
   ID                      int `json:"id"`
   Type               int `json:"type"`
   Img                   string `json:"img"`
   Abstract             string `json:"abstract"`
   PUserName                string `json:"p_user_name"`
   UpdateTime            string `json:"update_time"`
   Status               int `json:"status"`
}

// 获取数量
func GetBannerWhereCount(maps interface{}) (count int) {
   var db = core.Connection()
   db = db.Model(Banner{}).Where(maps).Count(&count)
   return
}

// 获取列表
func GetBannerWhereList(Page int,PageSize int,maps interface{}) (banner []Banner) {
   var db = core.Connection()
   db = db.Model(Banner{}).Where(maps).Offset((Page - 1) * PageSize).Limit(PageSize).Find(&banner)
   return
}

routers:

r.GET("/banner/getlist",Admin.GetBannerList)

直到开发这个程度的时候,我们总结一下Golang与PHP的差异:

1.弱类型语言(PHP7有类型限制,array很强大)上述两个方面带来的好处就是降低编程门槛,可以用较少的代码实现我们想要的功能。问题就是如果不遵循一定的编程规范,代码比较飘逸,可维护降低,另外由于要维护较为灵活的结构内存占用也会较大。

golang有较为严谨的语言风格检测,迫使我们统一风格,同时强类型使得我们编程的时候更加注重数据结构的设计,对于系统设计我的理解是有帮助的,当然也不那么易用。

2.Golang无第三方扩展,某些项目不提供PHP的扩展,在这方面可以选择自己封装(有一定维护成本)或者选择其他同类型的开源项目,问题不是太大。

3.Golang很多地方扩展采用多返回参数的形式返回error,时刻提醒开发者要关心异常,做好异常处理。

4.Golang三目运算符不见了。


由于自己是小白,有些写得不对的或者可优化的欢迎大家指出。

相关推荐

oracle数据导入导出_oracle数据导入导出工具

关于oracle的数据导入导出,这个功能的使用场景,一般是换服务环境,把原先的oracle数据导入到另外一台oracle数据库,或者导出备份使用。只不过oracle的导入导出命令不好记忆,稍稍有点复杂...

继续学习Python中的while true/break语句

上次讲到if语句的用法,大家在微信公众号问了小编很多问题,那么小编在这几种解决一下,1.else和elif是子模块,不能单独使用2.一个if语句中可以包括很多个elif语句,但结尾只能有一个else解...

python continue和break的区别_python中break语句和continue语句的区别

python中循环语句经常会使用continue和break,那么这2者的区别是?continue是跳出本次循环,进行下一次循环;break是跳出整个循环;例如:...

简单学Python——关键字6——break和continue

Python退出循环,有break语句和continue语句两种实现方式。break语句和continue语句的区别:break语句作用是终止循环。continue语句作用是跳出本轮循环,继续下一次循...

2-1,0基础学Python之 break退出循环、 continue继续循环 多重循

用for循环或者while循环时,如果要在循环体内直接退出循环,可以使用break语句。比如计算1至100的整数和,我们用while来实现:sum=0x=1whileTrue...

Python 中 break 和 continue 傻傻分不清

大家好啊,我是大田。今天分享一下break和continue在代码中的执行效果是什么,进一步区分出二者的区别。一、continue例1:当小明3岁时不打印年龄,其余年龄正常循环打印。可以看...

python中的流程控制语句:continue、break 和 return使用方法

Python中,continue、break和return是控制流程的关键语句,用于在循环或函数中提前退出或跳过某些操作。它们的用途和区别如下:1.continue(跳过当前循环的剩余部分,进...

L017:continue和break - 教程文案

continue和break在Python中,continue和break是用于控制循环(如for和while)执行流程的关键字,它们的作用如下:1.continue:跳过当前迭代,...

作为前端开发者,你都经历过怎样的面试?

已经裸辞1个月了,最近开始投简历找工作,遇到各种各样的面试,今天分享一下。其实在职的时候也做过面试官,面试官时,感觉自己问的问题很难区分候选人的能力,最好的办法就是看看候选人的github上的代码仓库...

面试被问 const 是否不可变?这样回答才显功底

作为前端开发者,我在学习ES6特性时,总被const的"善变"搞得一头雾水——为什么用const声明的数组还能push元素?为什么基本类型赋值就会报错?直到翻遍MDN文档、对着内存图反...

2023金九银十必看前端面试题!2w字精品!

导文2023金九银十必看前端面试题!金九银十黄金期来了想要跳槽的小伙伴快来看啊CSS1.请解释CSS的盒模型是什么,并描述其组成部分。答案:CSS的盒模型是用于布局和定位元素的概念。它由内容区域...

前端面试总结_前端面试题整理

记得当时大二的时候,看到实验室的学长学姐忙于各种春招,有些收获了大厂offer,有些还在苦苦面试,其实那时候的心里还蛮忐忑的,不知道自己大三的时候会是什么样的一个水平,所以从19年的寒假放完,大二下学...

由浅入深,66条JavaScript面试知识点(七)

作者:JakeZhang转发链接:https://juejin.im/post/5ef8377f6fb9a07e693a6061目录由浅入深,66条JavaScript面试知识点(一)由浅入深,66...

2024前端面试真题之—VUE篇_前端面试题vue2020及答案

添加图片注释,不超过140字(可选)1.vue的生命周期有哪些及每个生命周期做了什么?beforeCreate是newVue()之后触发的第一个钩子,在当前阶段data、methods、com...

今年最常见的前端面试题,你会做几道?

在面试或招聘前端开发人员时,期望、现实和需求之间总是存在着巨大差距。面试其实是一个交流想法的地方,挑战人们的思考方式,并客观地分析给定的问题。可以通过面试了解人们如何做出决策,了解一个人对技术和解决问...