快速掌握 GO 之 RabbitMQ 結合 gin+gorm 案例

更多個人筆記見:
注意點擊“繼續”,而不是“發現新項目”
github個人筆記倉庫 https://github.com/ZHLOVEYY/IT_note
gitee 個人筆記倉庫 https://gitee.com/harryhack/it_note
個人學習,學習過程中還會不斷補充~ (后續會更新在github上)

文章目錄

    • gin+gorm框架例子
        • 服務端生產者
        • 數據庫存儲
        • 客戶端消費者
        • 訪問測試

gin+gorm框架例子

post-platform/
├── main.go           # Gin 服務(生產者)
├── rabbitmq.go       # RabbitMQ 操作
├── models/
│   └── post.go       # 帖子模型
├── db/
│   └── db.go         # 數據庫連接和操作
├── consumer/
│   └── main.go       # 消費者(存儲到 MySQL)
├── go.mod
└── go.sum
服務端生產者
  • 定義 post.go

package modelstype Post struct {Title   string `json:"title"`Content string `json:"content"`
}

gin 框架:"go get github.com/gin-gonic/gin"

  • main.go:

package mainimport ("encoding/json""log""net/http""github.com/gin-gonic/gin""github.com/streadway/amqp"
)func failOnError(err error, msg string) {if err != nil {log.Fatalf("%s: %s", msg, err)}
}func main() {// 連接 RabbitMQconn, err := amqp.Dial("amqp://guest:guest@localhost:5672/")failOnError(err, "Failed to connect to RabbitMQ")defer conn.Close()ch, err := conn.Channel()failOnError(err, "Failed to open a channel")defer ch.Close()q, err := ch.QueueDeclare("post_queue", false, false, false, false, nil)failOnError(err, "Failed to declare a queue")// 初始化 Ginr := gin.Default()// 提交帖子接口r.POST("/posts", func(c *gin.Context) {var post struct {Title   string `json:"title" binding:"required"`Content string `json:"content" binding:"required"`}if err := c.ShouldBindJSON(&post); err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})return}// 序列化帖子為 JSONpostData, err := json.Marshal(post)if err != nil {c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to serialize post"})return}// 發送到 RabbitMQerr = ch.Publish("",     // 交換機q.Name, // 隊列名稱false,  // 強制false,  // 立即amqp.Publishing{ContentType: "application/json",Body:        postData,})if err != nil {c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to publish to RabbitMQ"})return}c.JSON(http.StatusOK, gin.H{"message": "Post submitted successfully"})})r.Run(":8081")
}
數據庫存儲

gorm 框架,需要 go get:

    "gorm.io/driver/mysql""gorm.io/gorm"
  • db.go
package dbimport ("log""test/model""gorm.io/driver/mysql""gorm.io/gorm"
)func InitDB() *gorm.DB {dsn := "root:password@tcp(localhost:3306)/dbname?charset=utf8mb4&parseTime=True&loc=Local"//根據情況填寫password 和 dbname(具體的數據庫和密碼),這里用的本地 sqldb, err := gorm.Open(mysql.Open(dsn), &gorm.Config{})if err != nil {log.Fatalf("Failed to connect to database: %v", err)}// 自動遷移,創建 posts 表err = db.AutoMigrate(&model.Post{})if err != nil {log.Fatalf("Failed to migrate database: %v", err)}return db
}
客戶端消費者
  • consumer.go
package mainimport ("encoding/json""log""test/db""github.com/streadway/amqp"
)func failOnError(err error, msg string) {if err != nil {log.Fatalf("%s: %s", msg, err)}
}type Post struct {Title   string `json:"title"`Content string `json:"content"`
}func main() {conn, err := amqp.Dial("amqp://guest:guest@localhost:5672/")failOnError(err, "Failed to connect to RabbitMQ")defer conn.Close()ch, err := conn.Channel()failOnError(err, "Failed to open a channel")defer ch.Close()q, err := ch.QueueDeclare("post_queue", false, false, false, false, nil)failOnError(err, "Failed to declare a queue")// 初始化數據庫db := db.InitDB()//消費消息msgs, err := ch.Consume(q.Name, "", true, false, false, false, nil)failOnError(err, "Failed to register a consumer")forever := make(chan bool)go func() {for d := range msgs {var posts Postif err := json.Unmarshal(d.Body, &posts); err != nil {log.Printf("Failed to unmarshal post: %v", err)continue}// 存儲到數據庫if err := db.Create(&posts).Error; err != nil {log.Printf("Failed to save post to database: %v", err)continue}log.Printf("Received post: Title=%s, Content=%s", posts.Title, posts.Content)// TODO: 存儲到數據庫(如 MySQL)}}()log.Printf(" [*] Waiting for posts. To exit press CTRL+C")<-forever // 等待程序退出,防止主線程退出,主動阻塞
}

gorm 中的 Create 是只要結構體的名字一樣就會找對應的表,所以結構體命名為 Post/Posts都可以,雖然和 model 中的不一樣,但是如果名字不一樣,Create 函數就“找不到”

訪問測試

分別終端運行程序后:
地址:http://localhost:8081/posts
發送內容:

{"title": "My First Post","content": "Hello, world!"
}

可以發現能正確送達,同時能存儲到數據庫中

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/85641.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/85641.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/85641.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

android FragmentManager 刪除所有Fragment 重建

在Android開發中&#xff0c;管理Fragment是一項常見任務&#xff0c;有時需要刪除所有Fragment并重新創建。這在某些場景下&#xff0c;例如用戶需要重置應用狀態或切換內容時&#xff0c;顯得尤為重要。本文將詳細介紹如何通過 FragmentManager刪除所有Fragment并重建。 一、…

ubuntu之開機自啟frpc

在 Ubuntu 系統中為 frpc 設置開機自啟&#xff08;以 frpc -c frpc.toml 命令為例&#xff09;&#xff0c;可以通過 systemd 服務實現。以下是詳細步驟&#xff1a; 創建 systemd 服務文件 sudo vim /etc/systemd/system/frpc.service 寫入以下內容&#xff08;根據你的路…

推薦一款PDF壓縮的工具

今天一位小伙伴找來&#xff0c;問我有沒有辦法將PDF變小的辦法。 詳細了解了一下使用場景&#xff1a; 小伙伴要在某系統上傳一個PDF文件&#xff0c;原文件是11.6MB&#xff0c;但是上傳時系統做了限制&#xff0c;只能上傳小于10MB的文件&#xff0c;如圖&#xff1a; 我聽…

JDK21深度解密 Day 11:云原生環境中的JDK21應用

【JDK21深度解密 Day 111】云原生環境中的JDK21應用 本文是《JDK21深度解密:從新特性到生產實踐的全棧指南》專欄的第11天內容,聚焦云原生環境中的JDK21應用。我們將深入探討如何在容器化、微服務、Serverless等云原生架構中充分發揮JDK21的技術優勢,提升Java應用的性能、穩…

Java-redis實現限時在線秒殺功能

1.使用redisson pom文件添加redisson <!--redisson--><dependency><groupId>org.redisson</groupId><artifactId>redisson-spring-boot-starter</artifactId><version>3.23.4</version></dependency> 2.mysql數據庫表設…

QT- QML Layout+anchors 布局+錨點實現窗口部件權重比例分配

布局管理 簡單比較兩種界面管理錨點布局實現比例布局布局管理實現比例布局循環依賴問題簡談 在日常打螺絲中&#xff0c;我們偶爾會需要實現界面各組件能按比例放置&#xff0c;自適應各種分辨率的需求。我用錨點和布局都實現過相關界面&#xff0c;記錄下來兩種方式實現的差異…

Java項目OOM排查

排查思路 Java項目出現OOM&#xff08;Out Of Memory&#xff0c;內存溢出&#xff09;問題時&#xff0c;排查思路如下&#xff1a; 確認OOM類型&#xff1a; Java Heap Space&#xff1a;堆內存溢出&#xff0c;通常是對象創建過多或內存泄漏。PermGen Space&#xff1a;永久…

vue+threeJs 生成云狀特效屏幕

嗨&#xff0c;我是小路。今天主要和大家分享的主題是“vuethreeJs 生成云狀特效屏幕”。 動態云狀特效示例圖 二、實例代碼 <!--創建一個動態數字屏幕--> <template><div class"pageBox"><div class"leftBox" ref"lef…

ABAP設計模式之---“高內聚,低耦合(High Cohesion Low Coupling)”

“高內聚、低耦合”是面向對象編程中非常重要的設計原則&#xff0c;它有助于提高代碼的可維護性、擴展性和復用性。 1. 初衷&#xff1a;為什么會有這個原則&#xff1f; 在軟件開發中&#xff0c;隨著業務需求的復雜化&#xff0c;代碼難免會變得越來越龐大。如果開發者將一…

Registry和docker有什么關系?

當遇到多個服務器需要同時傳docker鏡像的時候&#xff0c;一個一個的傳效率會非常慢且壓力完全在發送方的網絡帶寬&#xff1b;可以參考git hub&#xff0c;通常我們會用git push將代碼傳到git hub&#xff0c;如果誰需要代碼用git pull就可以拉到自己的機器上&#xff0c;dock…

linux命令 systemctl 和 supervisord 區別及用法解讀

目錄 基礎與背景服務管理范圍配置文件和管理方式監控與日志依賴管理適用場景常用命令對照表實際應用場景舉例優缺點對比小結參考鏈接 1. 基礎與背景 systemctl 和 supervisord 都是用于管理和控制服務&#xff08;進程&#xff09;的工具&#xff0c;但它們在設計、使用場景和…

(11)java+ selenium->元素定位之By_tag_name

1.簡介 繼續WebDriver關于元素定位,這篇介紹By ClassName。tagName是DOM結構的一部分,其中頁面上的每個元素都是通過輸入標簽,按鈕標簽或錨定標簽等標簽定義的。每個標簽都具有多個屬性,例如ID,名稱,值類等。就其他定位符而言在Selenium中,我們使用了標簽的這些屬性值來…

2021 RoboCom 世界機器人開發者大賽-高職組(復賽)解題報告 | 珂學家

前言 題解 2021 RoboCom 世界機器人開發者大賽-高職組&#xff08;復賽&#xff09;解題報告。 模擬題為主&#xff0c;包含進制轉換等等。 最后一題&#xff0c;是對向量/自定義類型&#xff0c;重定義小于操作符。 7-1 人工智能打招呼 分值: 15分 考察點: 分支判定&…

day42 簡單CNN

目錄 一、從圖像分類任務談起 二、CNN架構解剖實驗室 2.1 卷積層&#xff1a;空間特征的魔法師 2.2 歸一化層&#xff1a;加速收斂的隱形推手 2.3 激活函數&#xff1a;非線性的靈魂 三、工程實踐避坑指南 3.1 數據增強工程 3.2 調度器工程實戰 四、典型問題排查手冊 …

Gitee Wiki:以知識管理賦能 DevSecOps,推動關鍵領域軟件自主演進

關鍵領域軟件研發中的知識管理困境 傳統文檔管理模式問題顯著 關鍵領域軟件研發領域&#xff0c;傳統文檔管理模式問題顯著&#xff1a;文檔存儲無系統&#xff0c;查找困難&#xff0c;降低效率&#xff1b;更新不及時&#xff0c;與實際脫節&#xff0c;誤導開發&#xff1…

清理 pycharm 無效解釋器

1. 起因&#xff0c; 目的: 經常使用 pycharm 來調試深度學習項目&#xff0c;每次新建虛擬環境&#xff0c;都是顯示一堆不存在的名稱&#xff0c;刪也刪不掉。 總覺得很煩&#xff0c;是個痛點。決定深入研究一下。 2. 先看效果 效果是能行&#xff0c;而且清爽多了。 3. …

【ConvLSTM第二期】模擬視頻幀的時序建模(Python代碼實現)

目錄 1 準備工作&#xff1a;python庫包安裝1.1 安裝必要庫 案例說明&#xff1a;模擬視頻幀的時序建模ConvLSTM概述損失函數說明&#xff08;python全代碼&#xff09; 參考 ConvLSTM的原理說明可參見另一博客-【ConvLSTM第一期】ConvLSTM原理。 1 準備工作&#xff1a;pytho…

MySQL DDL操作全解析:從入門到精通,包含索引視圖分區表等全操作解析

目錄 一、DDL 基礎概述 1.1 DDL 定義與作用 1.2 DDL 語句分類 1.3 數據類型與存儲引擎 1.3.1 數據類型 1.3.2 存儲引擎差異 二、基礎 DDL 語句詳解 2.1 創建數據庫與表 2.1.1 創建數據庫 2.1.2 創建表 2.2 修改表結構 2.2.1 添加列 2.2.2 修改列屬性 2.2.3 刪除列…

設計模式——抽象工廠設計模式(創建型)

摘要 抽象工廠設計模式是一種創建型設計模式&#xff0c;旨在提供一個接口&#xff0c;用于創建一系列相關或依賴的對象&#xff0c;無需指定具體類。它通過抽象工廠、具體工廠、抽象產品和具體產品等組件構建&#xff0c;相比工廠方法模式&#xff0c;能創建一個產品族。該模…

Express教程【006】:使用Express寫接口

文章目錄 8、使用Express寫接口8.1 創建API路由模塊8.2 編寫GET接口8.3 編寫POST接口 8、使用Express寫接口 8.1 創建API路由模塊 1??新建routes/apiRouter.js路由模塊&#xff1a; /*** 路由模塊*/ // 1-導入express const express require(express); // 2-創建路由對象…