首先声明者只是一个个人开发的个人使用的小程序,本意是为了防止手机被盗而写的,由于程序需要在不为人知的情况下采集信息然后发送,所以程序没有界面。整个程序的大概思路是这样的
手机开始时:
1、获取手机联系人信息、通话记录、手机号码
2、检查手机gps状态:关闭状态则开启 ,然后获取手机的所在地
3、检查手机网络开关,关闭则开启
4、将采集的信息发送到指定邮箱中(本文以qq邮箱实现,由于代码太多这里就不贴出来了,有需要的联系我)
5、手机网络还原到初始状态、关闭gps(尚未实现,有实现的朋友可以一起研究一下)
添加权限
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.WRITE_SETTINGS" />
<uses-permission android:name="android.permission.READ_CONTACTS" />
<uses-permission android:name="android.permission.WRITE_SECURE_SETTINGS" />
<uses-permission android:name="android.permission.WRITE_APN_SETTINGS"/>
下面我们就按部就班来实现我们的想法:
1、获取联系人、电话号码、通话记录:
/**
* 获取本机号码
* @param context
* @return
*/
public static String getLocalNumber() {
TelephonyManager mTelephonyMgr;
mTelephonyMgr = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);
return mTelephonyMgr.getLine1Number();
}
/**
* 获取联系人
* @param context
* @return
*/
public static String getContact(){
ContentResolver cr = context.getContentResolver();
//取得电话本中开始一项的光标
Cursor cursor = cr.query(ContactsContract.Contacts.CONTENT_URI, null, null, null, null);
String string ="";
while (cursor.moveToNext())
{
// 取得联系人名字
int nameFieldColumnIndex = cursor.getColumnIndex(PhoneLookup.DISPLAY_NAME);
String name = cursor.getString(nameFieldColumnIndex);
string += ("联系人:"+name+", 电话号码:");
// 取得联系人ID
String contactId = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts._ID));
Cursor phone = cr.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = "
+ contactId, null, null);
// 取得电话号码(可能存在多个号码)
int i=0;
while (phone.moveToNext())
{
String strPhoneNumber = phone.getString(phone.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
if(i==0){
string += (strPhoneNumber);
}else{
string += ("," + strPhoneNumber);
}
i++;
}
string += "\n";
phone.close();
System.out.println("persion:"+string);
}
cursor.close();
return string;
}
/**
*获取通话记录
*/
public static List<Call> getCallList() {
List<Call> callList = new ArrayList<Call>();
int type;
Date date;
String time = "";
String telName = "";
String telNo = "";
ContentResolver cr = context.getContentResolver();
final Cursor cursor = cr.query(CallLog.Calls.CONTENT_URI,
new String[] { CallLog.Calls.NUMBER, CallLog.Calls.CACHED_NAME,
CallLog.Calls.TYPE, CallLog.Calls.DATE,
CallLog.Calls.DURATION }, null, null,
CallLog.Calls.DEFAULT_SORT_ORDER);
for (int i = 0; i < cursor.getCount(); i++) {
Call call = new Call();
cursor.moveToPosition(i);
telName = cursor.getString(1);
telNo = cursor.getString(0);
type = cursor.getInt(2);
SimpleDateFormat sfd = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
date = new Date(Long.parseLong(cursor.getString(3)));
time = sfd.format(date);
// call.setLongTime(formatDuring(Long.valueOf(cursor.getString(4))));
// String callDate = getdays(Long.parseLong(cursor.getString(3)));
if (telName != null) {
call.setName(telName);
} else {
call.setName("未知联系人");
}
call.setNumber(telNo);
call.setTime(time);
if (CallLog.Calls.INCOMING_TYPE == type) {
call.setType("接听");
} else if (CallLog.Calls.OUTGOING_TYPE == type) {
call.setType("拨出");
} else if (CallLog.Calls.MISSED_TYPE == type) {
call.setType("未接");
}
callList.add(call);
}
return callList;
}
private String formatDuring(long mss) {
long hours = mss / (60 * 60);
long minutes = (mss % (1000 * 60 * 60)) / 60;
long seconds = (mss % (1000 * 60));
return hours + ":" + minutes + ":" + seconds;
}
private String getdays(long callTime) {
String value = "";
long newTime = new Date().getTime();
long duration = (newTime - callTime) / (1000 * 60);
if (duration < 60) {
value = duration + "分钟前";
} else if (duration >= 60 && duration < DAY) {
value = (duration / 60) + "小时前";
} else if (duration >= DAY && duration < DAY * 2) {
value = "昨天";
} else if (duration >= DAY * 2 && duration < DAY * 3) {
value = "前天";
} else if (duration >= DAY * 7) {
SimpleDateFormat sdf = new SimpleDateFormat("M月dd日");
value = sdf.format(new Date(callTime));
} else {
value = (duration / DAY) + "天前";
}
return value;
}
/**
*实体类
*/
public class Call {
private String name;
private String number;
private String time;
private String type;
}
popuwindows使用时出现异常:
android.view.WindowManager$BadTokenException: Unable to add window -- token null is not valid; is your activity running?
解决方法:
应把pop.showAtLocation(parent, Gravity.TOP,0, 0)这一句移出oncreate方法,在控件渲染完毕后再使用
参考:http://multiarrow.iteye.com/blog/1158492
AlertDialog也会有类似问题 参见:
http://pangaoyuan.iteye.com/blog/1086374
在两个activity之间传递数据无非是用Bundle 和Intent.putExtra的重载方法。
不过Bundle貌似可以传递对象,前提是对象必须是可序列化的,序列化却会降低性能,传递的时候要把这个对象序列化,取对象的时候还要进行反序列化。
不知道大家知道不知道jsp,或者Servlet里面的Session,可以放入对象,也可以取对象。
按照这个原理,我写了个极简单的Session类用来保存对象,无论你跳转到哪个activity你都可以获取你之前放入的对象。
代码如下
再解释一下吧,这个类是个单例模式,要通过Session.getSession()拿到一个Session对象,得到Session后就可以放入对象了,别忘了用完后调用 remove方法或cleanUpSession方法,好让GC回收。
示例如下:
在一个Activity里使用如下代码放入对象
Session session = Session.getSession();
Date date = new Date();
session.put("date", date);
在另外一个Activity里拿到对象
Session session = Session.getSession();
Date date = (Date)session.get("date");
至于拿到的是不是同一个对象,大家自己比较一下就行了。
欢迎大家提出意见!