如何使用Struts 2 + Hibernate 3将图像提供给浏览器?
我正在使用Struts 2.1.2和Hibernate 3.2.6.GA开发Web应用程序。我有一个实体" User",已使用Hibernate将其映射到数据库中的表" USERS"。我想有一个与此实体相关的图像,我打算将其存储为数据库中的" BLOB"。我也想在网页上显示图像以及"用户"的其他属性。
我想到的解决方案是有一个表IMAGES(ID,IMAGE),其中IMAGE是BLOB列。 " USERS"将有一个名为" IMAGEID"的" FK"列,该列指向" IMAGES"表。然后,我将在"用户"实体上映射一个名为" imageId"的属性,该属性以Long形式映射到此" IMAGEID"。当使用JSP渲染页面时,我会将图像添加为<img src =" images.action?id = 1" />`等,并具有一个Action来读取图像并将内容流式传输到浏览器,标头设置为长时间缓存图像。
这会工作吗?有没有更好的方法来渲染存储在数据库中的图像?首先将这样的图像存储在数据库中是正确的方法吗?
解决方案
如果要直接显示用户图像及其属性,则可以考虑将图像数据直接嵌入HTML中。
使用特殊数据:URL方案,我们可以将任何mime数据嵌入HTML页面中,其格式如下:
data:<mimetype>;base64,<data>
需要替换为数据的MIME类型(例如,image / png),
是文件实际字节的base64编码的字符串。
请参阅RFC 2557.
例子:
<img src="">
Internet Explorer不支持该样式的图像嵌入。
是的,我们建议的解决方案将起作用。假设我们正在Java环境中工作,那么将图像存储在数据库中是最好的方法。如果我们在带有应用程序服务器的单服务器环境中运行,从技术上讲,它可以让我们以爆炸格式进行部署,则可以将映像存储在磁盘上,但这不是最佳实践。一种建议是使用servlet而不是JSP。为了获得良好的浏览器行为,我们希望浏览器认为它显示的文件类型与期望的文件类型匹配。尽管存在mime类型标头,但文件扩展名仍然非常重要。因此,我们需要一个看起来像这样的链接:
<a href="foo.jsp"><img src="imageservlet/123456789.png"></a>
其中123456789是数据库中映像的主键。servlet映射如下所示:
<servlet> <servlet-name>ImageServlet</servlet-name> <servlet-class>com.example.ImageServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>ImageServlet</servlet-name> <url-pattern>/imageservlet/*</url-pattern> </servlet-mapping>
然后,在servlet中,只需解析图像ID的请求URL,而不是使用查询字符串,因为查询字符串会使某些浏览器感到困惑。使用查询字符串不会完全破坏浏览器,但在缓存方面会出现奇怪的行为,某些浏览器可能会将内容报告为不安全。
我们建议的解决方案将完美运行。我做过同样的事情。
但是我们不需要一个servlet。
Struts2已具有流结果。
请参阅此Struts 2示例,其中准确描述了我们想要的内容。