سوالات متداول
سوالات عمومی
چگونه بارگذاری مجدد خودکار را در حین توسعه فعال کنم؟
از Air برای بارگذاری مجدد خودکار در حین توسعه استفاده کنید. Air فایلهای شما را نظارت میکند و هنگام تشخیص تغییرات، برنامه شما را بازسازی/راهاندازی مجدد میکند.
نصب:
go install github.com/air-verse/air@latestراهاندازی:
یک فایل پیکربندی .air.toml در ریشه پروژه خود ایجاد کنید:
air initسپس air را در پوشه پروژه خود به جای go run اجرا کنید:
airAir فایلهای .go شما را نظارت کرده و به طور خودکار برنامه Gin شما را هنگام تغییرات بازسازی/راهاندازی مجدد میکند. برای گزینههای پیکربندی مستندات Air را ببینید.
چگونه CORS را در Gin مدیریت کنم؟
از میانافزار رسمی gin-contrib/cors استفاده کنید:
package main
import ( "time"
"github.com/gin-contrib/cors" "github.com/gin-gonic/gin")
func main() { r := gin.Default()
// Default CORS configuration r.Use(cors.Default())
// Or customize CORS settings r.Use(cors.New(cors.Config{ AllowOrigins: []string{"https://example.com"}, AllowMethods: []string{"GET", "POST", "PUT", "DELETE"}, AllowHeaders: []string{"Origin", "Content-Type", "Authorization"}, ExposeHeaders: []string{"Content-Length"}, AllowCredentials: true, MaxAge: 12 * time.Hour, }))
r.GET("/ping", func(c *gin.Context) { c.JSON(200, gin.H{"message": "pong"}) })
r.Run()}برای مرور کامل امنیتی، بهترین روشهای امنیتی را ببینید.
چگونه فایلهای استاتیک را ارائه دهم؟
از Static() یا StaticFS() برای ارائه فایلهای استاتیک استفاده کنید:
func main() { r := gin.Default()
// Serve files from ./assets directory at /assets/* r.Static("/assets", "./assets")
// Serve a single file r.StaticFile("/favicon.ico", "./resources/favicon.ico")
// Serve from embedded filesystem (Go 1.16+) r.StaticFS("/public", http.FS(embedFS))
r.Run()}برای جزئیات بیشتر ارائه داده از فایل را ببینید.
چگونه آپلود فایل را مدیریت کنم؟
از FormFile() برای فایلهای تکی یا MultipartForm() برای فایلهای متعدد استفاده کنید:
// Single file uploadr.POST("/upload", func(c *gin.Context) { file, _ := c.FormFile("file") c.SaveUploadedFile(file, "./uploads/"+file.Filename) c.String(200, "File %s uploaded successfully", file.Filename)})
// Multiple files uploadr.POST("/upload-multiple", func(c *gin.Context) { form, _ := c.MultipartForm() files := form.File["files"]
for _, file := range files { c.SaveUploadedFile(file, "./uploads/"+file.Filename) } c.String(200, "%d files uploaded", len(files))})برای جزئیات بیشتر مستندات آپلود فایل را ببینید.
چگونه احراز هویت JWT را پیادهسازی کنم؟
از gin-contrib/jwt استفاده کنید یا میانافزار سفارشی پیادهسازی کنید. در اینجا یک مثال ساده آمده است:
package main
import ( "net/http" "time"
"github.com/gin-gonic/gin" "github.com/golang-jwt/jwt/v5")
var jwtSecret = []byte("your-secret-key")
type Claims struct { Username string `json:"username"` jwt.RegisteredClaims}
func AuthMiddleware() gin.HandlerFunc { return func(c *gin.Context) { tokenString := c.GetHeader("Authorization") if tokenString == "" { c.JSON(http.StatusUnauthorized, gin.H{"error": "Missing authorization token"}) c.Abort() return }
// Remove "Bearer " prefix if present if len(tokenString) > 7 && tokenString[:7] == "Bearer " { tokenString = tokenString[7:] }
token, err := jwt.ParseWithClaims(tokenString, &Claims{}, func(token *jwt.Token) (interface{}, error) { return jwtSecret, nil })
if err != nil || !token.Valid { c.JSON(http.StatusUnauthorized, gin.H{"error": "Invalid token"}) c.Abort() return }
if claims, ok := token.Claims.(*Claims); ok { c.Set("username", claims.Username) c.Next() } }}برای احراز هویت مبتنی بر نشست، مدیریت نشست را ببینید.
چگونه لاگگذاری درخواستها را تنظیم کنم؟
Gin شامل یک میانافزار لاگر پیشفرض از طریق gin.Default() است. برای لاگگذاری ساختاریافته JSON در تولید، لاگگذاری ساختاریافته را ببینید.
برای سفارشیسازی لاگ پایه:
r := gin.New()r.Use(gin.LoggerWithConfig(gin.LoggerConfig{ SkipPaths: []string{"/healthz"},}))r.Use(gin.Recovery())بخش لاگگذاری را برای تمام گزینهها شامل فرمتهای سفارشی، خروجی فایل و رد شدن از رشتههای پرسوجو ببینید.
چگونه خاموشی آرام را مدیریت کنم؟
راهاندازی مجدد یا توقف آرام را برای راهنمای کامل با مثالهای کد ببینید.
چرا به جای “405 Method Not Allowed” پیام “404 Not Found” دریافت میکنم؟
به طور پیشفرض، Gin برای مسیرهایی که از متد HTTP درخواستی پشتیبانی نمیکنند 404 برمیگرداند. HandleMethodNotAllowed = true را تنظیم کنید تا به جای آن 405 برگرداند:
r := gin.Default()r.HandleMethodNotAllowed = true
r.GET("/ping", func(c *gin.Context) { c.JSON(200, gin.H{"message": "pong"})})
r.Run()$ curl -X POST localhost:8080/ping
HTTP/1.1 405 Method Not AllowedAllow: GETچگونه پارامترهای پرسوجو و دادههای POST را با هم متصل کنم؟
از ShouldBind() استفاده کنید که به طور خودکار اتصال را بر اساس نوع محتوا انتخاب میکند:
type User struct { Name string `form:"name" json:"name"` Email string `form:"email" json:"email"` Page int `form:"page"`}
r.POST("/user", func(c *gin.Context) { var user User if err := c.ShouldBind(&user); err != nil { c.JSON(400, gin.H{"error": err.Error()}) return } c.JSON(200, user)})بخش اتصال داده را برای تمام گزینههای اتصال ببینید.
چگونه دادههای درخواست را اعتبارسنجی کنم؟
Gin از go-playground/validator برای اعتبارسنجی استفاده میکند. تگهای اعتبارسنجی را به structهای خود اضافه کنید:
type User struct { Name string `json:"name" binding:"required,min=3,max=50"` Email string `json:"email" binding:"required,email"` Age int `json:"age" binding:"gte=0,lte=130"`}
r.POST("/user", func(c *gin.Context) { var user User if err := c.ShouldBindJSON(&user); err != nil { c.JSON(400, gin.H{"error": err.Error()}) return } c.JSON(200, gin.H{"message": "User is valid"})})اتصال و اعتبارسنجی را برای اعتبارسنجهای سفارشی و استفاده پیشرفته ببینید.
چگونه Gin را در حالت تولید اجرا کنم؟
متغیر محیطی GIN_MODE را به release تنظیم کنید:
export GIN_MODE=release# orGIN_MODE=release ./your-appیا به صورت برنامهای تنظیم کنید:
gin.SetMode(gin.ReleaseMode)حالت release لاگگذاری اشکالزدایی را غیرفعال کرده و عملکرد را بهبود میبخشد.
چگونه اتصالات پایگاه داده را با Gin مدیریت کنم؟
یکپارچهسازی پایگاه داده را برای راهنمای کامل شامل database/sql، GORM، connection pooling و الگوهای تزریق وابستگی ببینید.
چگونه handlerهای Gin را تست کنم؟
از net/http/httptest برای تست مسیرهای خود استفاده کنید:
func TestPingRoute(t *testing.T) { router := gin.Default() router.GET("/ping", func(c *gin.Context) { c.JSON(200, gin.H{"message": "pong"}) })
w := httptest.NewRecorder() req, _ := http.NewRequest("GET", "/ping", nil) router.ServeHTTP(w, req)
assert.Equal(t, 200, w.Code) assert.Contains(t, w.Body.String(), "pong")}مستندات تست را برای مثالهای بیشتر ببینید.
سوالات عملکرد
چگونه Gin را برای ترافیک بالا بهینه کنم؟
- از حالت Release استفاده کنید:
GIN_MODE=releaseرا تنظیم کنید - میانافزارهای غیرضروری را غیرفعال کنید: فقط از آنچه نیاز دارید استفاده کنید
- از
gin.New()به جایgin.Default()برای کنترل دستی میانافزار استفاده کنید - Connection pooling: pool اتصالات پایگاه داده را پیکربندی کنید (ببینید یکپارچهسازی پایگاه داده)
- کشگذاری: کشگذاری را برای دادههایی که مکرراً دسترسی میشوند پیادهسازی کنید
- توزیع بار: از پروکسی معکوس (nginx، HAProxy) استفاده کنید
- پروفایلینگ: از pprof در Go برای شناسایی گلوگاهها استفاده کنید
- نظارت: متریکها و نظارت را برای ردیابی عملکرد تنظیم کنید
آیا Gin آماده تولید است؟
بله. Gin توسط بسیاری از شرکتها در تولید استفاده میشود و در مقیاس آزموده شده است. کاربران را برای مثالهایی از پروژههایی که Gin را در تولید استفاده میکنند ببینید.
عیبیابی
چرا پارامترهای مسیر من کار نمیکنند؟
مطمئن شوید پارامترهای مسیر از سینتکس : استفاده میکنند و به درستی استخراج میشوند:
// Correctr.GET("/user/:id", func(c *gin.Context) { id := c.Param("id") c.String(200, "User ID: %s", id)})
// Not: /user/{id} or /user/<id>برای جزئیات پارامترها در مسیر را ببینید.
چرا میانافزار من اجرا نمیشود؟
میانافزار باید قبل از مسیرها یا گروههای مسیر ثبت شود:
// Correct orderr := gin.New()r.Use(MyMiddleware()) // Register middleware firstr.GET("/ping", handler) // Then routes
// For route groupsauth := r.Group("/admin")auth.Use(AuthMiddleware()) // Middleware for this group{ auth.GET("/dashboard", handler)}برای جزئیات استفاده از میانافزار را ببینید.
چرا اتصال درخواست ناموفق است؟
دلایل رایج:
- تگهای اتصال گمشده: تگهای
json:"field"یاform:"field"اضافه کنید - عدم تطابق Content-Type: مطمئن شوید کلاینت هدر Content-Type صحیح ارسال میکند
- خطاهای اعتبارسنجی: تگهای اعتبارسنجی و الزامات را بررسی کنید
- فیلدهای unexported: فقط فیلدهای struct با حرف بزرگ (exported) متصل میشوند
type User struct { Name string `json:"name" binding:"required"` // ✓ Correct Email string `json:"email"` // ✓ Correct age int `json:"age"` // ✗ Won't bind (unexported)}برای جزئیات اتصال و اعتبارسنجی را ببینید.