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

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

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

我是 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三目运算符不见了。


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

相关推荐

MySql系列-常用命令

本篇是对...

Record.ToTable 格式转换

本期案例对表格格式进行转换,前后转换效果如下:解题套路1.Record.ToTable解题思路:思路就是构造可以透视的样式。使用Record.ToTable对行记录进行转换,获得包含两列的表,首行可以...

Table.Group 按时期累计计算唯一值

本期案例是根据不同id进行汇总统计:组内,相同日期的为一组,统计“from”、“to”中的非重复个数;连续日期的,统计累计数。前后转换效果如下:解题套路1.Table.Group...

MySQL 9.1正式发布,有哪些值得关注的新特性?

MySQL创新版9.1.0于2024年10月15日正式发布。此外,MySQL8.0.40及8.4.3补丁版本也同时发布。8.4.3是目前MySQL的LTS长期支持版本,该版本中将不会增加新的功能与特性...

SQL基本语句练习(基础版)

最近在学习SQL基本语句的练习,在此分享一下笔者做过的练习以及个人的解决教程:首先是基本练习表格的搭建,具体内容如下表所示:...

SQL 从入门到精通:全面掌握数据库操作

学习SQL(StructuredQueryLanguage)是掌握数据库操作的关键步骤。SQL是一种用于管理和处理关系型数据库的标准语言,广泛应用于数据检索、插入、更新和删除等操作。以下是一些...

ClickHouse学习笔记四ClickHouse基础语法

前言这里我们介绍ClickHouse的基本语法,使用环境是腾讯云的ClickHouse。默认情况下,ClickHouse在进行集群纬度执行建表等DDL操作时需要手动添加ONCLUSTERX...

程序员总结的常用sql语句大全

多年经验程序员总结的我们一般需要使用的sql语句,赶快收藏起来,方便以后使用。以下是一些常用的SQL语句及其用法:一、数据定义语言(DDL)创建库CREATEDATABASE:创建一个新数据库。...

PQ03-分组求和

目标已知:销售清单求:每个销售员的销量合计方法数据准备...

好荐:一款数据库元数据管理平台工具

“元数据”的定义在不同的软件、项目、工程的定义范围都不太一样。本文这里指的是软件项目开发使用的数据库表结构信息。我今天介绍的这个开源项目叫Databasir,它是一个面向团队的关系型数据库模型文档管理...

MySQL 8.0 SQL优化黑科技,面试官都不一定知道!

前言提到SQL优化,大多数人想到的还是那些经典套路:建索引、避免全表扫描、优化JOIN顺序…这些确实是基础,但如果你还停留在MySQL5.7时代的优化思维,那就out了。MySQL8.0已经发布好...

MySQL数据库深度优化指南:从基础到架构层面的20个关键策略

一、核心性能优化原则数据最小化原则...

动物源性食品中兽药残留的检测——喹啉类药物残留

喹啉类药物(quinoxaline)是具有喹啉-N1,N4-二氧化物基本结构的一类化学合成的动物专用药,具有广谱抗菌、提高饲料转化率和促生长作用。1965年德国拜耳公司以邻硝基苯胺为原料合成喹乙醇(o...

适合普通开发者和产品经理的PHP应用模板开发AI的SaaS应用框架

简单到傻!Liang_SaaS适合普通开发者和产品经理的PHP应用模板开发AI的SaaS应用框架,利用Php开发AI的SaaS应用框架,是一个强大的内容管理仪表板模板,基于Bootstrap和...

Power Query 交错合并表格的方法

两张表格合并成一张表格,需要交错排列,表1取一行,表2取一行,这样排列在一起:前提是两张表的行数相同,内容排列顺序相同:我们来看两张表:表1:12列10行表2:11列10行行数相同列数不同,我们在数据...