通过SSL的JavaMail IMAP速度非常慢-批量提取多条消息

我目前正在尝试使用JavaMail从IMAP服务器(Gmail和其他服务器)获取电子邮件。基本上,我的代码有效:我确实可以获取标头,正文内容等。我的问题如下:在IMAP服务器(无SSL)上工作时,基本上需要1-2毫秒来处理一条消息。当我使用IMAPS服务器(因此使用SSL,例如Gmail)时,我达到了大约250m

/消息。我仅测量处理消息时的时间(不考虑连接,握手等)。

我知道由于这是SSL,因此数据已加密。但是,解密时间不应该那么重要,对吗?

我尝试设置更高的ServerCacheSize值和更高的connectionpoolsize,但是严重用尽了想法。有人遇到这个问题吗?解决了一个可能的希望?

我担心JavaMail API每次从IMAPS服务器获取邮件时都会使用不同的连接(涉及握手的开销……)。如果是这样,是否有一种方法可以覆盖此行为?

这是我从Main()类调用的代码(尽管很标准):

 public static int connectTest(String SSL, String user, String pwd, String host) throws IOException,

ProtocolException,

GeneralSecurityException {

Properties props = System.getProperties();

props.setProperty("mail.store.protocol", SSL);

props.setProperty("mail.imaps.ssl.trust", host);

props.setProperty("mail.imaps.connectionpoolsize", "10");

try {

Session session = Session.getDefaultInstance(props, null);

// session.setDebug(true);

Store store = session.getStore(SSL);

store.connect(host, user, pwd);

Folder inbox = store.getFolder("INBOX");

inbox.open(Folder.READ_ONLY);

int numMess = inbox.getMessageCount();

Message[] messages = inbox.getMessages();

for (Message m : messages) {

m.getAllHeaders();

m.getContent();

}

inbox.close(false);

store.close();

return numMess;

} catch (MessagingException e) {

e.printStackTrace();

System.exit(2);

}

return 0;

}

提前致谢。

回答:

经过大量工作和JavaMail员工的帮助,这种“缓慢”的来源来自API中的FETCH行为。确实,正如pjaol所说,每次需要消息的信息(标头或消息内容)时,我们都会返回服务器。

如果FetchProfile允许我们批量获取许多消息的标头信息或标志,则无法直接获取多个消息的内容。

幸运的是,我们可以编写自己的IMAP命令来避免这种“局限性”(这样做是为了避免出现内存不足错误:在一个命令中提取内存中的每个邮件可能会很繁重)。

这是我的代码:

import com.sun.mail.iap.Argument;

import com.sun.mail.iap.ProtocolException;

import com.sun.mail.iap.Response;

import com.sun.mail.imap.IMAPFolder;

import com.sun.mail.imap.protocol.BODY;

import com.sun.mail.imap.protocol.FetchResponse;

import com.sun.mail.imap.protocol.IMAPProtocol;

import com.sun.mail.imap.protocol.UID;

public class CustomProtocolCommand implements IMAPFolder.ProtocolCommand {

/** Index on server of first mail to fetch **/

int start;

/** Index on server of last mail to fetch **/

int end;

public CustomProtocolCommand(int start, int end) {

this.start = start;

this.end = end;

}

@Override

public Object doCommand(IMAPProtocol protocol) throws ProtocolException {

Argument args = new Argument();

args.writeString(Integer.toString(start) + ":" + Integer.toString(end));

args.writeString("BODY[]");

Response[] r = protocol.command("FETCH", args);

Response response = r[r.length - 1];

if (response.isOK()) {

Properties props = new Properties();

props.setProperty("mail.store.protocol", "imap");

props.setProperty("mail.mime.base64.ignoreerrors", "true");

props.setProperty("mail.imap.partialfetch", "false");

props.setProperty("mail.imaps.partialfetch", "false");

Session session = Session.getInstance(props, null);

FetchResponse fetch;

BODY body;

MimeMessage mm;

ByteArrayInputStream is = null;

// last response is only result summary: not contents

for (int i = 0; i < r.length - 1; i++) {

if (r[i] instanceof IMAPResponse) {

fetch = (FetchResponse) r[i];

body = (BODY) fetch.getItem(0);

is = body.getByteArrayInputStream();

try {

mm = new MimeMessage(session, is);

Contents.getContents(mm, i);

} catch (MessagingException e) {

e.printStackTrace();

}

}

}

}

// dispatch remaining untagged responses

protocol.notifyResponseHandlers(r);

protocol.handleResult(response);

return "" + (r.length - 1);

}

}

getContents(MimeMessage mm,int i)函数是一个经典函数,该函数以递归方式将消息的内容打印到文件中(网上有许多示例)。

为了避免出现内存不足错误,我只需设置maxDocs和maxSize限制(这是任意完成的,可能可以改进!),其用法如下:

public int efficientGetContents(IMAPFolder inbox, Message[] messages)

throws MessagingException {

FetchProfile fp = new FetchProfile();

fp.add(FetchProfile.Item.FLAGS);

fp.add(FetchProfile.Item.ENVELOPE);

inbox.fetch(messages, fp);

int index = 0;

int nbMessages = messages.length;

final int maxDoc = 5000;

final long maxSize = 100000000; // 100Mo

// Message numbers limit to fetch

int start;

int end;

while (index < nbMessages) {

start = messages[index].getMessageNumber();

int docs = 0;

int totalSize = 0;

boolean noskip = true; // There are no jumps in the message numbers

// list

boolean notend = true;

// Until we reach one of the limits

while (docs < maxDoc && totalSize < maxSize && noskip && notend) {

docs++;

totalSize += messages[index].getSize();

index++;

if (notend = (index < nbMessages)) {

noskip = (messages[index - 1].getMessageNumber() + 1 == messages[index]

.getMessageNumber());

}

}

end = messages[index - 1].getMessageNumber();

inbox.doCommand(new CustomProtocolCommand(start, end));

System.out.println("Fetching contents for " + start + ":" + end);

System.out.println("Size fetched = " + (totalSize / 1000000)

+ " Mo");

}

return nbMessages;

}

在此不要使用我所使用的消息号,它是不稳定的(如果从服务器中删除了消息,则这些更改)。更好的方法是使用UID!然后,将命令从FETCH更改为UID

FETCH。

希望这会有所帮助!

以上是 通过SSL的JavaMail IMAP速度非常慢-批量提取多条消息 的全部内容, 来源链接: utcz.com/qa/425436.html

回到顶部