<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
postgres提供了兩種不同的方式儲存二進位制,要麼是使用bytea型別直接儲存二進位制,要麼就是使用postgres的LargeObject功能;決定使用哪中方式更加適合你,就需要了解這兩種儲存方式有哪些限制
bytea型別在單列中雖然可以支援1GB的容量,但是還是不建議使用bytea去儲存比較大的物件,因為它會佔用大量的記憶體
下面通過一個例子來說明,假如現在要在一個表中儲存圖片名和該圖片的資料,建立表如下:
CREATE TABLE images (imgname text, img bytea);
在表中插入一張圖片:
File file = new File("myimage.gif"); FileInputStream fis = new FileInputStream(file); PreparedStatement ps = conn.prepareStatement("INSERT INTO images VALUES (?, ?)"); ps.setString(1, file.getName()); ps.setBinaryStream(2, fis, file.length()); ps.executeUpdate(); ps.close(); fis.close();
上面的setBinaryStream就會將圖片內容設定到img欄位上面,也可以使用setBytes()直接設定圖片的內容
接下來,從表中取出圖片,程式碼如下:
PreparedStatement ps = con.prepareStatement("SELECT img FROM images WHERE imgname = ?"); ps.setString(1, "myimage.gif"); ResultSet rs = ps.executeQuery(); if (rs != null) { while (rs.next()) { byte[] imgBytes = rs.getBytes(1); // use the data in some way here } rs.close(); } ps.close();
Large Object就可以儲存大檔案,儲存的方式是在單獨的一張表中儲存大檔案,然後通過oid在當前表中進行參照;下面通過一個例子來解釋:
CREATE TABLE imageslo (imgname text, imgoid oid);
首先是建立一張表,該表中第二個欄位型別為oid,之後就是通過該欄位參照大檔案物件;下面我們在表中插入一張圖片:
// All LargeObject API calls must be within a transaction block conn.setAutoCommit(false); // Get the Large Object Manager to perform operations with LargeObjectManager lobj = ((org.postgresql.PGConnection)conn).getLargeObjectAPI(); // Create a new large object int oid = lobj.create(LargeObjectManager.READ | LargeObjectManager.WRITE); // Open the large object for writing LargeObject obj = lobj.open(oid, LargeObjectManager.WRITE); // Now open the file File file = new File("myimage.gif"); FileInputStream fis = new FileInputStream(file); // Copy the data from the file to the large object byte buf[] = new byte[2048]; int s, tl = 0; while ((s = fis.read(buf, 0, 2048)) > 0) { obj.write(buf, 0, s); tl += s; } // Close the large object obj.close(); // Now insert the row into imageslo PreparedStatement ps = conn.prepareStatement("INSERT INTO imageslo VALUES (?, ?)"); ps.setString(1, file.getName()); ps.setInt(2, oid); ps.executeUpdate(); ps.close(); fis.close();
在程式碼中需要使用lobp.open開啟一個大檔案,然後將圖片的內容寫入這個物件當中;下面從大檔案物件中讀取這個圖片:
// All LargeObject API calls must be within a transaction block conn.setAutoCommit(false); // Get the Large Object Manager to perform operations with LargeObjectManager lobj = ((org.postgresql.PGConnection)conn).getLargeObjectAPI(); PreparedStatement ps = con.prepareStatement("SELECT imgoid FROM imageslo WHERE imgname = ?"); ps.setString(1, "myimage.gif"); ResultSet rs = ps.executeQuery(); if (rs != null) { while (rs.next()) { // Open the large object for reading int oid = rs.getInt(1); LargeObject obj = lobj.open(oid, LargeObjectManager.READ); // Read the data byte buf[] = new byte[obj.size()]; obj.read(buf, 0, obj.size()); // Do something with the data read here // Close the object obj.close(); } rs.close(); } ps.close();
需要注意的是,對於Large Object的操作都需要放在一個事務(Transaction)當中;如果要刪除大檔案所在行,在刪除這行之後,還需要再執行刪除大檔案的操作
注:使用Large Object會有安全問題,連線到資料庫的使用者,即便沒有包含大物件所在列的許可權,也可以操作這個大物件
看完上面的postgres對於圖片的儲存,再來看下如何使用beego orm儲存一張圖片;在beego orm中支援了go的所有基礎型別,但是不支援slice;所以,不能直接將[]byte對映到bytea欄位上面
好在beego orm提供了一個Fielder介面,可以自定義型別,介面定義如下:
// Fielder define field info type Fielder interface { String() string FieldType() int SetRaw(interface{}) error RawValue() interface{} }
所以,現在就需要定義一個位元組陣列的型別,然後實現這些介面就好了,程式碼如下:
type ByteArrayField []byte // set value func (e *ByteArrayField) SetRaw(value interface{}) error { if value == nil { return nil } switch d := value.(type) { case []byte: *e = d case string: *e = []byte(d) default: return fmt.Errorf("[ByteArrayField] unsupported type") } return nil } func (e *ByteArrayField) RawValue() interface{} { return *e } // specified type func (f *ByteArrayField) FieldType() int { return orm.TypeTextField } func (f *ByteArrayField) String() string { return string(*f) }
然後,我們就可以在struct中進行對映了,如下:
type ImageModel struct{ ImageName string `orm:"column(image_name)"` ImageData ByteArrayField `orm:"column(image_data);type(bytea)"` }
這樣就可以使用orm的介面操作imageModel,向資料庫插入圖片,或者從資料庫讀出圖片的內容了
以上就是詳解如何使用beego orm在postgres中儲存圖片的詳細內容,更多關於beego orm postgres儲存圖片的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45