如何从 PHP 过渡到 Golang?(php转go需要多久)
myzbx 2025-06-04 00:03 9 浏览
我是 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三目运算符不见了。
由于自己是小白,有些写得不对的或者可优化的欢迎大家指出。
相关推荐
- 如何用5分钟开发一个 Webpack Loader?
-
嗨,我是勾勾。今天分享的内容是如何开发一个简单的WebpackLoader,希望通过这个过程能够让你Get到WebpackLoader的工作原理与机制。Loader作为Webpack...
- 前端——CORS跨域请求的限制与解决
-
node中设置允许跨域如果需要设置多个域允许跨域,可以根据req请求的地址进行写入不同的header;consthttp=require('http')http.cre...
- 5分钟看懂的WebAssembly入门指南(webassembly开发)
-
子肃阿里开发者2023-06-2009:01发表于浙江阿里妹导读本文是一篇WebAssembly的入门文章,从理论介绍到实战方面有全面的讲述。历史进程由于javascript的动态类型特性...
- 刚刚发布!Claude 4连续工作7小时,比Cursor、Copilot还猛?
-
你见过不吃不喝、连续工作7小时的“程序员”吗?Anthropic最新发布的Claude4,不只是AI,更像是你团队里的CTO。一、什么是Claude4?别急,这不是你熟悉的GPT“亲戚”202...
- JS对象判空的几种方式,你真的会了吗?
-
前言:为什么空对象检测如此重要?在开发中我们经常会遇到这样的场景:if(isEmpty(userInfo)){//跳转登录页}四种主流检测方案对比方案一:Object.keys()基础版fun...
- 密码被破译,行踪被美军全程掌握,日本海军军神命丧太平洋
-
【军武次位面】FriedrichLau一.突袭1941年12月7日,伴随着日军偷袭美军位于珍珠港的基地,美国也终于卷入了这场绵延全球的战火之中。为了报复日军这一行动,美军随后打出了一套组合拳,除了在太...
- 提示词技术详解(2)——零样本提示词
-
一、零样本提示(Zero-Shot)是一种会起到作用的办法。首先让模型重写提示词,然后把重写后的提示词再发给模型,以期提升回答效果。论文给出的提示词如下,仅供参考。给定一位用户的以下文字,提取其中不带...
- 这些流行饮料的中文名称,你会说吗?
-
[Photo/Pexels]Summerisinfullswing,andtheweatherishot!Tohelpyoucooldown,coldandrefre...
- 密码被破译多可怕?被美军全程盯梢,日本海军军神命丧太平洋
-
【军武次位面】FriedrichLau一.突袭1941年12月7日,伴随着日军偷袭美军位于珍珠港的基地,美国也终于卷入了这场绵延全球的战火之中。为了报复日军这一行动,美军随后打出了一套组合拳,除了在太...
- 一课译词:刀子嘴(刀子嘴是什么)
-
你身边一定有一些人,他们的言语总是那么尖锐、刺耳,但内心却又格外善良柔软,了解他们的人都知道,他们其实只是“刀子嘴,豆腐心”。“刀子嘴”,形容人说话十分刻薄(speaksarcasticallya...
- 捷克插画家柯薇塔·巴可维斯卡逝世,曾为《灰姑娘》绘制插图
-
柯瑞塔·巴可维斯卡。(图源:捷克共和国文化部)据捷克多家媒体消息,当地时间2月6日,捷克插画家柯薇塔·巴可维斯卡逝世,享年94岁。该消息经由她的儿子斯特潘·格里格(StěpánGrygar)证实。柯...
- 网络“匿名提问箱”成年轻人社交新宠 为何这么火?
-
网络“匿名提问箱”成为年轻人社交新宠“来自陌生人的关心”为什么这么火?“年度歌单里排名第一的是哪首歌?”“未来十年你的人生规划?”“有没有被甩过?”最近,这种别人能够匿名向自己提问的“提问箱”越来越得...
- 美国要开始搞6G了?专家:关键技术仍在摸索
-
2月21日,美国总统特朗普发推特“我希望5G乃至6G早日在美国落地”。日前,美国联邦通信委员会朝着特朗普的指示迈出了第一步,决定开放95千兆赫到3太赫兹频段,供6G实验使用。纽约大学教授泰德·拉帕波特...
- 常见的连续型随机变量(1)(连续型随机变量的定义与性质)
-
1.均匀分布在概率论和统计学中,均匀分布也叫矩形分布,它是对称概率分布,在相同长度间隔的分布概率是等可能的。均匀分布由两个参数a和b定义,它们是数轴上的最小值和最大值,通常缩写为U(a,b)。统计...
- 身高表上的-2SD、-1SD、中位数.....都是啥?和百分位有关系吗?
-
上周日晚,小编正气呼呼地和娃上演“作业拉锯战”时,“叮”的一声,一条微信发了过来。无独有偶,第二天又有朋友发来门诊记录,不知道SD什么意思。从家长应用的角度来看,无需太纠结,根据个人习惯选择即可。从生...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 简介 (30)
- HTML 响应式设计 (31)
- HTML URL 编码 (32)
- HTML Web 服务器 (31)
- HTML 表单属性 (32)
- HTML 音频 (31)
- HTML5 支持 (33)
- HTML API (36)
- HTML 总结 (32)
- HTML 全局属性 (32)
- HTML 事件 (31)
- HTML 画布 (32)
- HTTP 方法 (30)
- 键盘快捷键 (30)
- CSS 语法 (35)
- CSS 选择器 (30)
- CSS 轮廓宽度 (31)
- CSS 谷歌字体 (33)
- CSS 链接 (31)
- CSS 定位 (31)
- CSS 图片库 (32)
- CSS 图像精灵 (31)
- SVG 文本 (32)
- 时钟启动 (33)
- HTML 游戏 (34)