我正在嘗試使用 Go、protobuf 和 SQL Server 作為我的資料庫來建立 API 請求。我的 HTTP 伺服器由curl 請求啟動,回傳代碼:5
我的main.go
程式碼
package main import ( "context" "database/sql" "encoding/json" "fmt" "log" "net" "net/http" "github.com/grpc-ecosystem/grpc-gateway/v2/runtime" pb "github.com/myorg/nb_nb_nb/proto" "google.golang.org/grpc" "google.golang.org/grpc/credentials" _ "google.golang.org/grpc/grpclog/glogger" _ "github.com/denisenkom/go-mssqldb" ) type appServer struct { pb.UnimplementedMyAppServiceServer db *sql.DB } //mssql const ( dbserver = "windows.net" port = 1433 user = "reader" password = "mypassword" database = "dbnm" ) const sqlQuery = ` my query ` func (s *appServer) ExecuteQuery(ctx context.Context, req *pb.QueryRequest) (*pb.QueryResponseList, error) { log.Printf("received: Start Date - %v, End Date - %v", req.GetReportStartDate(), req.GetReportEndDate()) rows, err := s.db.Query(sqlQuery, req.ReportStartDate, req.ReportEndDate) if err != nil { log.Printf("Error executing query: %v", err) return nil, fmt.Errorf("failed to execute query: %v", err) } defer rows.Close() var results []*pb.QueryResponse for rows.Next() { var result pb.QueryResponse err := rows.Scan( //fields ) if err != nil { log.Printf("Error scanning rows: %v", err) return nil, fmt.Errorf("failed to scan rows: %v", err) } results = append(results, &result) } return &pb.QueryResponseList{Responses: results}, nil } func connectDB() (*sql.DB, error) { connectionString := fmt.Sprintf("server=%s;user id=%s;password=%s;port=%d;database=%s", dbserver, user, password, port, database) db, err := sql.Open("mssql", connectionString) if err != nil { return nil, fmt.Errorf("failed to connect to the database: %v", err) } if err := db.Ping(); err != nil { return nil, fmt.Errorf("database connection failed: %v", err) } fmt.Println("Database connection successful") return db, nil } func main() { defer func() { if r := recover(); r != nil { log.Println("Recovered from panic:", r) } }() db, err := connectDB() if err != nil { log.Fatalf("Failed to connect to the database: %v", err) } defer db.Close() log.Println("Starting gRPC server...") lis, err := net.Listen("tcp", ":7777") if err != nil { log.Fatalf("TCP connection failed: %v", err) } else { log.Printf("gRPC Server Listening at %v", lis.Addr()) } size := 1024 * 1024 * 50 s := grpc.NewServer(grpc.MaxRecvMsgSize(size), grpc.MaxSendMsgSize(size)) pb.RegisterMyAppServiceServer(s, &appServer{db: db}) mux := runtime.NewServeMux() opts := []grpc.DialOption{grpc.WithTransportCredentials(credentials.NewClientTLSFromCert(nil, ""))} err = pb.RegisterMyAppServiceHandlerFromEndpoint(context.Background(), mux, ":7777", opts) if err != nil { log.Fatalf("Failed to register gRPC gateway: %v", err) } else { log.Printf("Registered gRPC gateway") } http.HandleFunc("/api/query", func(w http.ResponseWriter, r *http.Request) { startDate := r.URL.Query().Get("startDate") endDate := r.URL.Query().Get("endDate") req := &pb.QueryRequest{ ReportStartDate: startDate, ReportEndDate: endDate, } fmt.Print(req) resp, err := (&appServer{db: db}).ExecuteQuery(r.Context(), req) if err != nil { log.Printf("Error executing query: %v", err) http.Error(w, "Internal Server Error", http.StatusInternalServerError) return } w.Header().Set("Content-Type", "application/json") json.NewEncoder(w).Encode(resp.Responses) }) go func() { log.Println("Starting HTTP server...") err := http.ListenAndServe(":7778", mux) if err != nil { log.Fatalf("HTTP server failed: %v", err) } else { log.Printf("HTTP server started.") } }() if err := s.Serve(lis); err != nil { log.Fatalf("gRPC server failed: %v", err) } else { log.Printf("gRPC, HTTP, and API server started.") } }
go執行server/main.go
返回
Database connection successful 2023/12/24 21:48:23 Starting gRPC server... 2023/12/24 21:48:23 gRPC Server Listening at [::]:7777 2023/12/24 21:48:23 Registered gRPC gateway 2023/12/24 21:48:23 Starting HTTP server...
和curl請求(curl -X GET "http://localhost:7778/api/query?startDate=2023-10-01&endDate=2023-10-02"
)返回:
{"code":5, "message":"Not Found", "details":[]}
我的程式碼中缺少什麼?為什麼我無法回傳任何結果?我的防火牆已關閉。
正確答案
看起來路由「/api/query」已經定義在預設的ServeMux上,但是在監聽介面時使用了mux ServeMux。
以上是即使 HTTP 伺服器開始使用 Go 和 protobuf、SQL Server,請求也不會傳回結果的詳細內容。更多資訊請關注PHP中文網其他相關文章!

在Go中,使用互斥鎖和鎖是確保線程安全的關鍵。 1)使用sync.Mutex進行互斥訪問,2)使用sync.RWMutex處理讀寫操作,3)使用原子操作進行性能優化。掌握這些工具及其使用技巧對於編寫高效、可靠的並發程序至關重要。

如何優化並發Go代碼的性能?使用Go的內置工具如gotest、gobench和pprof進行基準測試和性能分析。 1)使用testing包編寫基準測試,評估並發函數的執行速度。 2)通過pprof工具進行性能分析,識別程序中的瓶頸。 3)調整垃圾收集設置以減少其對性能的影響。 4)優化通道操作和限制goroutine數量以提高效率。通過持續的基準測試和性能分析,可以有效提升並發Go代碼的性能。

避免並發Go程序中錯誤處理的常見陷阱的方法包括:1.確保錯誤傳播,2.處理超時,3.聚合錯誤,4.使用上下文管理,5.錯誤包裝,6.日誌記錄,7.測試。這些策略有助於有效處理並發環境中的錯誤。

IndimitInterfaceImplementationingingoembodiesducktybybyallowingTypestoSatoSatiSatiSatiSatiSatiSatsatSatiSatplicesWithouTexpliclIctDeclaration.1)itpromotesflemotesflexibility andmodularitybybyfocusingion.2)挑戰挑戰InclocteSincludeUpdatingMethodSignateSignatiSantTrackingImplections.3)工具li

在Go編程中,有效管理錯誤的方法包括:1)使用錯誤值而非異常,2)採用錯誤包裝技術,3)定義自定義錯誤類型,4)復用錯誤值以提高性能,5)謹慎使用panic和recover,6)確保錯誤消息清晰且一致,7)記錄錯誤處理策略,8)將錯誤視為一等公民,9)使用錯誤通道處理異步錯誤。這些做法和模式有助於編寫更健壯、可維護和高效的代碼。

在Go中實現並發可以通過使用goroutines和channels來實現。 1)使用goroutines來並行執行任務,如示例中同時享受音樂和觀察朋友。 2)通過channels在goroutines之間安全傳遞數據,如生產者和消費者模式。 3)避免過度使用goroutines和死鎖,合理設計系統以優化並發程序。

Gooffersmultipleapproachesforbuildingconcurrentdatastructures,includingmutexes,channels,andatomicoperations.1)Mutexesprovidesimplethreadsafetybutcancauseperformancebottlenecks.2)Channelsofferscalabilitybutmayblockiffullorempty.3)Atomicoperationsareef

go'serrorhandlingisexplicit,治療eRROSASRETRATERTHANEXCEPTIONS,與pythonandjava.1)go'sapphifeensuresererrawaresserrorawarenessbutcanleadtoverbosecode.2)pythonandjavauseexeexceptionseforforforforforcleanerCodebutmaymobisserrors.3)


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

SublimeText3 Linux新版
SublimeText3 Linux最新版

SAP NetWeaver Server Adapter for Eclipse
將Eclipse與SAP NetWeaver應用伺服器整合。

SublimeText3漢化版
中文版,非常好用

Dreamweaver Mac版
視覺化網頁開發工具

PhpStorm Mac 版本
最新(2018.2.1 )專業的PHP整合開發工具