Java 赢得图像';如果尺寸较大,则不能装载

Java 赢得图像';如果尺寸较大,则不能装载,java,mysql,image,jsf,servlets,Java,Mysql,Image,Jsf,Servlets,我正在使用MEDIUMBLOB将图像存储在DB中。当我尝试通过servlet加载图像时,我能够看到这些图像。但是,如果图像大小较大(1 MB或更大),我可以在浏览器上看到一半或3/4的图像 当我下载同一张图片并将其放到公共网络内容中时,它工作得非常完美。你知道如何克服这个问题吗?我需要在servlet或MySQL中设置任何变量吗 (JSF生成的)HTML代码如下所示: <img src="DisplayImage?mainID=drawing" /> InputStream con

我正在使用MEDIUMBLOB将图像存储在DB中。当我尝试通过servlet加载图像时,我能够看到这些图像。但是,如果图像大小较大(1 MB或更大),我可以在浏览器上看到一半或3/4的图像

当我下载同一张图片并将其放到公共网络内容中时,它工作得非常完美。你知道如何克服这个问题吗?我需要在servlet或MySQL中设置任何变量吗

(JSF生成的)HTML代码如下所示:

<img src="DisplayImage?mainID=drawing" />
InputStream content = resultSet.getBinaryStream("content");
int contentLength = resultSet.getInt("contentLength");
// ...
response.setContentLength(contentLength);
SomeUtil.streamByBuffer(content, response.getOutputStream());
编辑1 当我使用下面的代码并将文件保存到本地磁盘时,我看到了完整的图像

String imgLen = rs1.getString(1);
int len = imgLen.length();
rb = new byte[len];
inputStream = rs1.getBinaryStream(1);
while ((read = inputStream.read(rb)) != -1) {
    out.write(rb, 0, read);
}
out.flush();
out.close();
编辑2 当我保存观看了一半的图像时,我注意到那些图像的大小是100KB。我的1MB图像显示大小为100KB。所有图像都会发生这种情况:(

我认为这是最大的暗示什么是错的,但我不明白什么是错的

编辑3 如果我从
web.xml
中删除以下内容,我就能够查看这些图像

<filter>
    <filter-name>MyFacesExtensionsFilter</filter-name>
    <filter-class>org.apache.myfaces.webapp.filter.ExtensionsFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>MyFacesExtensionsFilter</filter-name>
    <servlet-name>Faces Servlet</servlet-name>
</filter-mapping>

MyFaceXtensionsFilter
org.apache.myfaces.webapp.filter.ExtensionsFilter
MyFaceXtensionsFilter
Facesservlet
我可以删除这些吗?我不重新收集为什么我添加了这些

编辑4 我的web.xml文件是

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="3.0" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd">
    <context-param>
        <param-name>javax.faces.PROJECT_STAGE</param-name>
        <param-value>Development</param-value>
    </context-param>
    <servlet>
        <servlet-name>Faces Servlet</servlet-name>
        <servlet-class>javax.faces.webapp.FacesServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>Faces Servlet</servlet-name>
        <url-pattern>/faces/*</url-pattern>
    </servlet-mapping>
    <session-config>
        <session-timeout>
            60
        </session-timeout>
    </session-config>
    <welcome-file-list>
        <welcome-file>faces/index.xhtml</welcome-file>
    </welcome-file-list>
    <filter>
        <filter-name>restrict</filter-name>
        <filter-class>com.sac.filter.MyFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>restrict</filter-name>
        <url-pattern>*.xhtml</url-pattern>
    </filter-mapping>
    <filter>
        <filter-name>MyFacesExtensionsFilter</filter-name>
        <filter-class>org.apache.myfaces.webapp.filter.ExtensionsFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>MyFacesExtensionsFilter</filter-name>
        <servlet-name>Faces Servlet</servlet-name>
    </filter-mapping>

    <servlet>
        <servlet-name>DisplayImage</servlet-name>
        <servlet-class>com.sac.databean.DisplayImage</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>DisplayImage</servlet-name>
        <url-pattern>/DisplayImage</url-pattern>
    </servlet-mapping>
    <servlet>
        <servlet-name>SaveMyImage</servlet-name>
        <servlet-class>com.sac.databean.SaveMyImage</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>SaveMyImage</servlet-name>
        <url-pattern>/SaveMyImage</url-pattern>
    </servlet-mapping>
</web-app>

javax.faces.PROJECT_阶段
发展
Facesservlet
javax.faces.webapp.FacesServlet
1.
Facesservlet
/面孔/*
60
faces/index.xhtml
限制
com.sac.filter.MyFilter
限制
*.xhtml
MyFaceXtensionsFilter
org.apache.myfaces.webapp.filter.ExtensionsFilter
MyFaceXtensionsFilter
Facesservlet
显示图像
com.sac.databean.DisplayImage
显示图像
/显示图像
SaveMyImage
com.sac.databean.SaveMyImage
SaveMyImage
/SaveMyImage

您测量图像长度的方式是错误的。您基本上是在将图像字节转换为人类可读的字符(以不可理解的顺序,但除此之外)然后计算字符数。这不对。您需要计算原始字节和未转换字节数。一个字符不一定只由一个字节表示,但可以由多个字节表示

有两种方法可以解决您的问题:

  • 改用
    ResultSet#getBytes()

    byte[] content = resultSet.getBytes("content");
    // ...
    response.setContentLength(content.length);
    response.getOutputStream().write(content);
    
    请注意,这是内存占用,因为
    byte[]
    的每个
    byte
    基本上累积了Java内存的一个字节

  • 在查询中也选择BLOB长度。如何选择取决于使用的数据库。在MySQL中,您可以使用
    length()
    函数进行此操作

    SELECT content, LENGTH(content) AS contentLength FROM image WHERE id = ?
    
    然后,您可以按如下方式处理:

    <img src="DisplayImage?mainID=drawing" />
    
    InputStream content = resultSet.getBinaryStream("content");
    int contentLength = resultSet.getInt("contentLength");
    // ...
    response.setContentLength(contentLength);
    SomeUtil.streamByBuffer(content, response.getOutputStream());
    
    (其中该处理不是通过全图像长度的
    字节[]
    进行的)


  • 更新:毕竟,图像上的请求似乎调用了
    MyFacesExtensionsFilter
    ,当
    chain.doFilter()时,它显然过度地缓冲响应,而没有正确地刷新响应
    返回。此筛选器仅在调用
    FacesServlet
    时根据映射规则调用。但这不应该发生。映像请求应该只调用映像servlet,而不调用FacesServlet

    根据映射规则,
    FacesServlet
    /faces/*
    上被调用,而图像servlet映射在
    /DisplayImage
    上。此时的
    与当前请求URL相关,因此它最终将成为
    /faces/DisplayImage
    ,它将首先调用
    FacesServletDisplayImage
    servlet。这是错误的

    您应该相应地更改JSF代码,使
    最终成为与域相关的,以便它只调用
    /DisplayImage
    servlet

    <img src="/contextpath/DisplayImage?mainID=drawing" />
    
    
    
    您只需在
    中使用前导斜杠即可实现这一点。它将自动在上下文路径前面加上前缀

    <h:graphicImage value="/DisplayImage?mainID=drawing" />
    

    您测量图像长度的方式是错误的。您基本上是在将图像字节转换为人类可读的字符(以不可理解的顺序,但除此之外)然后计算字符数。这不对。您需要计算原始字节和未转换字节数。一个字符不一定只由一个字节表示,但可以由多个字节表示

    有两种方法可以解决您的问题:

  • 改用
    ResultSet#getBytes()

    byte[] content = resultSet.getBytes("content");
    // ...
    response.setContentLength(content.length);
    response.getOutputStream().write(content);
    
    请注意,这是内存占用,因为
    byte[]
    的每个
    byte
    基本上累积了Java内存的一个字节

  • 在查询中也选择BLOB长度。如何选择取决于使用的数据库。在MySQL中,您可以使用
    length()
    函数进行此操作

    SELECT content, LENGTH(content) AS contentLength FROM image WHERE id = ?
    
    然后,您可以按如下方式处理:

    <img src="DisplayImage?mainID=drawing" />
    
    InputStream content = resultSet.getBinaryStream("content");
    int contentLength = resultSet.getInt("contentLength");
    // ...
    response.setContentLength(contentLength);
    SomeUtil.streamByBuffer(content, response.getOutputStream());
    
    (其中该处理不是通过全图像长度的
    字节[]
    进行的)


  • 更新:毕竟,图像上的请求似乎调用了
    MyFacesExtensionsFilter
    ,当
    chain.doFilter()时,它显然过度地缓冲响应,而没有正确地刷新响应
    返回。此筛选器仅在调用
    FacesServlet
    时根据映射规则调用。但这不应该发生。映像请求应该只调用映像servlet,而不调用FacesServlet

    根据映射规则,在
    /faces/*
    上调用
    FacesServlet
    ,而在
    /DisplayImage
    上映射图像servlet。此时的
    与当前请求URL相关,因此