0

MEDIUMBLOBを使用して画像をDBに保存しています。サーブレットを介して画像を読み込もうとすると、それらの画像を見ることができます。ただし、画像サイズが大きい場合(1 MB以上)、ブラウザに半分または3/4の画像が表示されます。

同じ画像をダウンロードして公開Webコンテンツに入れると、完全に機能します。この問題を克服する方法はありますか?サーブレットまたはMySQLに変数を設定する必要がありますか?

(JSFで生成された)HTMLコードは次のとおりです。

<img src="DisplayImage?mainID=drawing" />

イメージサーブレットは次のことを行います。

String imgLen = rs1.getString(1);
int len = imgLen.length();
byte[] rb = new byte[len];
InputStream readImg = rs1.getBinaryStream(1);
InputStream inputStream = readImg;
int index = readImg.read(rb, 0, len);
response.reset();
response.setHeader("Content-Length", String.valueOf(len));
response.setHeader("Content-disposition", "inline;filename=/file.png");
response.setContentType("image/png");
response.getOutputStream().write(rb, 0, len);
response.getOutputStream().flush();

編集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>

それらを削除できますか?なぜそれらを正確に追加したのか、私は再収集しません...

編集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>
4

1 に答える 1

3

画像の長さの測定方法が間違っています。基本的に、画像バイトを人間が読める文字に変換し(理解できない順序で、ただしそれは別として)、文字数を計算します。これは正しくないです。代わりに、未加工の未変換バイトの量を計算する必要があります。1文字は必ずしも1バイトだけで表されるわけではありませんが、複数のバイトで表すことができます。

問題を解決するには2つの方法があります。

  1. ResultSet#getBytes()代わりに使用してください。

    byte[] content = resultSet.getBytes("content");
    // ...
    response.setContentLength(content.length);
    response.getOutputStream().write(content);
    

    byte基本的にすべてがbyte[]Javaのメモリの1バイトを蓄積するため、これはメモリを大量に消費することに注意してください。

  2. クエリでもBLOBの長さを選択します。その方法は、使用するDBによって異なります。LENGTH()MySQLでは、このための関数を使用できます。

    SELECT content, LENGTH(content) AS contentLength FROM image WHERE id = ?
    

    次に、これを次のように処理します。

    InputStream content = resultSet.getBinaryStream("content");
    int contentLength = resultSet.getInt("contentLength");
    // ...
    response.setContentLength(contentLength);
    SomeUtil.streamByBuffer(content, response.getOutputStream());
    

    (この場合、プロセスはbyte[]画像の全長にわたって実行されません)


更新:結局のところ、画像に対するリクエストはMyFacesExtensionsFilter、応答を適切にフラッシュせずに応答を熱心にバッファリングしているように見えましたchain.doFilter()。このフィルターは、FacesServletが呼び出されたときにのみ呼び出されるマッピングルールに従います。しかし、これは起こるべきではありませんでした。画像リクエストは、facesサーブレットではなく、画像サーブレットのみを呼び出す必要があります。

マッピングルールに従って、イメージサーブレットがにマッピングされている間にFacesServletが呼び出されます。現在のリクエストURLに関連しているため、最終的には、最初に、次にサーブレットを呼び出すことになります。これは間違っています。/faces/*/DisplayImage<img src>/faces/DisplayImageFacesServletDisplayImage

それに応じてJSFコードを変更して、最終的にドメイン相対になり、サーブレット<img src>のみを呼び出すようにする必要があります。/DisplayImage

<img src="/contextpath/DisplayImage?mainID=drawing" />

これは、の先頭のスラッシュを使用するだけで実現できます<h:graphicImage>。コンテキストパスを自動追加します。

<h:graphicImage value="/DisplayImage?mainID=drawing" />
于 2012-10-21T16:57:23.970 に答える