php小編魚仔將為您介紹如何在 Gorm 中使用 Raw() 進行 Preload()。Gorm是一個強大的Go語言ORM庫,提供了許多便捷的方法來進行數據庫操作。在某些情況下,我們可能需要使用原生的SQL語句進行查詢,同時還想使用Preload()方法來預加載相關的數據。通過使用Raw()方法,我們可以在Gorm中結合Preload()來實現這一需求。接下來,我們將詳細講解具體的操作步驟,幫助您更好地理解和應用這個技巧。
問題內容
之前在我的項目中,我需要做一些復雜的查詢,所以我使用了 Raw()
。查詢如下所示:
SELECT tbl1.id, tbl1.some_name, tbl5.some_status, tbl1.some_tbl2_id, tbl1.type_id, tbl1.created_at, tbl5.name FROM table1 tbl1 JOIN table2 tbl2 ON tbl1.some_tbl2_id = tbl2.id JOIN table3 tbl3 ON tbl3.edge_device_info_id = tbl2.id JOIN table4 tbl4 ON tbl4.id = tbl3.account_id LEFT JOIN table5 tbl5 ON tbl5.tbl1_id = tbl1.id WHERE tbl5.tbl1_id IS NULL OR tbl5.updated_at = ( SELECT MAX(updated_at) FROM table5 WHERE tbl1_id = tbl1.id ) ORDER BY tbl1.created_at ASC;
登錄后復制
我不太確定我是否可以使用 gorm
中的內容/方法完全做到這一點,所以我只是使用我更熟悉的簡單查詢。現在,我想獲取與 tbl1.type_id
關聯的記錄。我嘗試在 Raw()
之前添加 Preload()
但這似乎不起作用,因為檢查我用來存儲查詢結果的結構數組的內容似乎沒有填充 Type
。
更新:
環顧四周后,我找到了一種將上面的 Raw()
查詢轉換為 gorm
的方法鏈接的方法。它看起來像這樣:
Model(models.ActuatorDeviceInfo{}).Preload("ActuatorDeviceInfo.Type"). Select("actuator_device_infos.*, actuator_device_infos.id, at.*"). Joins("JOIN edge_device_infos edi ON actuator_device_infos.parent_edge_device_id = edi.id"). Joins("JOIN user_owned_edge_devices ae ON ae.edge_device_info_id = edi.id"). Joins("JOIN accounts acc ON acc.id = ae.account_id"). Joins("JOIN actuator_types at ON at.id = actuator_device_infos.type_id"). Joins("LEFT JOIN actuator_updates au ON au.actuator_device_info_id = actuator_device_infos.id"). Where("au.actuator_device_info_id IS NULL OR au.updated_at = (?)", helpers.GetDB().Model(&models.ActuatorUpdate{}).Select("MAX(updated_at)"). Where("au.actuator_device_info_id = actuator_device_infos.id")). Order("actuator_device_infos.created_at DESC"). Scan(&actuator_device_infos)
登錄后復制
它的工作方式就像之前的 Raw()
查詢一樣,但它仍然缺少一些東西,這就是獲取關聯表到 table1 的方法(它是方法鏈接上的actuator_device_infos,有點懶惰仍然清理新代碼)。即使我在查詢構建方法之前添加 Preload()
,它似乎也不會影響結果記錄。我需要使用 Model()
進行方法鏈接,并且缺少它會產生錯誤,因為 Model()
中的解析表將用于最終查詢的 FROM
部分,因此 gorm
應該有所了解我想要預加載的東西存在。我預加載的是 actuator_types
,它已經有一個 Joins()
,如上面的代碼所示。
解決方法
感謝@Trock的評論,我終于找到了最后一塊拼圖。看來我只需要在方法鏈的末尾使用 Find()
,這樣 Preload()
就可以工作。最終的代碼看起來就像我在問題上的代碼,但我用 Find()
替換了 Scan()
。我嘗試對 Raw()
代碼執行相同的操作,并且效果也很好。 gorm
應該提到該機制。