例子: Alarm service 定时发送短信到手机
主要是自己想做一个摘自杭州气象的详细的天气预报,普通的天气预报服务的内容太过简单。不够满足日常的需求。
新手开发,第一个安卓程序,很多东西都不熟悉。
这里先发一个定时发送短信的代码吧。
主要用到AlarmManager和BroardCast
public void startTimer() { Log.i("MSG", "start MainActivity!"); setTitle("Waiting... Alarm=5"); Intent intent = new Intent(MainActivity.this, SendMessageReciever.class); PendingIntent p_intent = PendingIntent.getBroadcast( MainActivity.this, 0, intent, 0); Calendar calendar = Calendar.getInstance(); calendar.setTimeInMillis(System.currentTimeMillis()); calendar.add(Calendar.SECOND, 5); // Schedule the alarm! AlarmManager am = (AlarmManager) getSystemService(ALARM_SERVICE); am.set(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), p_intent); }
AlarmManager的常用方法有三个:
(1)set(inttype,longstartTime,PendingIntentpi);
该方法用于设置一次性闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟执行时间,第三个参数表示闹钟响应动作。
(2)setRepeating(inttype,longstartTime,longintervalTime,PendingIntentpi);
该方法用于设置重复闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟首次执行时间,第三个参数表示闹钟两次执行的间隔时间,第三个参数表示闹钟响应动作。
(3)setInexactRepeating(inttype,longstartTime,longintervalTime,PendingIntentpi);
该方法也用于设置重复闹钟,与第二个方法相似,不过其两个闹钟执行的间隔时间不是固定的而已。
闹钟的类型,常用的有5个值:
AlarmManager.ELAPSED_REALTIME、AlarmManager.ELAPSED_REALTIME_WAKEUP、AlarmManager.RTC、AlarmManager.RTC_WAKEUP、AlarmManager.POWER_OFF_WAKEUP。
AlarmManager.ELAPSED_REALTIME表示闹钟在手机睡眠状态下不可用,该状态下闹钟使用相对时间(相对于系统启动开始),状态值为3;
AlarmManager.ELAPSED_REALTIME_WAKEUP表示闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟也使用相对时间,状态值为2;
AlarmManager.RTC表示闹钟在睡眠状态下不可用,该状态下闹钟使用绝对时间,即当前系统时间,状态值为1;
AlarmManager.RTC_WAKEUP表示闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟使用绝对时间,状态值为0;
AlarmManager.POWER_OFF_WAKEUP表示闹钟在手机关机状态下也能正常进行提示功能,所以是5个状态中用的最多的状态之一,该状态下闹钟也是用绝对时间,状态值为4;不过本状态好像受SDK版本影响,某些版本并不支持;
关于最后一个参数
PendingIntentpi:是闹钟的执行动作,比如发送一个广播、给出提示等等。PendingIntent是Intent的封装类。需要注意的是,如果是通过启动服务来实现闹钟提示的话,PendingIntent对象的获取就应该采用Pending.getService(Contextc,inti,Intentintent,intj)方法;如果是通过广播来实现闹钟提示的话,PendingIntent对象的获取就应该采用PendingIntent.getBroadcast(Contextc,inti,Intentintent,intj)方法;如果是采用Activity的方式来实现闹钟提示的话,PendingIntent对象的获取就应该采用PendingIntent.getActivity(Contextc,inti,Intentintent,intj)方法。如果这三种方法错用了的话,虽然不会报错,但是看不到闹钟提示效果。
此段闹钟方法的介绍摘自http://my.oschina.net/rotiwen/blog/114016。
回到例子的代码AlarmManager时间到了以后调用BroadCast:
import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.util.Log; public class SendMessageReciever extends BroadcastReceiver{ @Override public void onReceive(Context context, Intent intent) { // TODO Auto-generated method stub Log.i("MSG", "start SendMessageReciever!"); context.startService(new Intent(context,SendMessageService.class)); } }
Reciever里面又启动了一个service,让service去发送短信。
import java.util.List; import android.annotation.SuppressLint; import android.app.Activity; import android.app.PendingIntent; import android.app.Service; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.os.IBinder; import android.telephony.SmsManager; import android.util.Log; import android.widget.Toast; public class SendMessageService extends Service{ @Override public IBinder onBind(Intent intent) { // TODO Auto-generated method stub return null; } @SuppressLint("NewApi")@Override public void onCreate() { Log.i("MSG", "start SendmessageService!"); Toast.makeText(this, "Start sending!", Toast.LENGTH_LONG).show(); String SENT_SMS_ACTION = "SENT_SMS_ACTION"; Intent sentIntent = new Intent(SENT_SMS_ACTION); PendingIntent sentPI = PendingIntent.getBroadcast(this, 0, sentIntent, 0); // register the Broadcast Receivers this.registerReceiver(new BroadcastReceiver() { @Override public void onReceive(Context _context, Intent _intent) { switch (getResultCode()) { case Activity.RESULT_OK: Toast.makeText(_context, "Send success!", Toast.LENGTH_LONG).show(); break; case SmsManager.RESULT_ERROR_GENERIC_FAILURE: break; case SmsManager.RESULT_ERROR_RADIO_OFF: break; case SmsManager.RESULT_ERROR_NULL_PDU: break; } } }, new IntentFilter(SENT_SMS_ACTION)); String DELIVERED_SMS_ACTION = "DELIVERED_SMS_ACTION"; // create the deilverIntent parameter Intent deliverIntent = new Intent(DELIVERED_SMS_ACTION); PendingIntent deliverPI = PendingIntent.getBroadcast(this, 0, deliverIntent, 0); this.registerReceiver(new BroadcastReceiver() { @Override public void onReceive(Context _context, Intent _intent) { Toast.makeText(_context, "收信人已经成功接收", Toast.LENGTH_SHORT) .show(); } }, new IntentFilter(DELIVERED_SMS_ACTION)); String content = "这里的字数很多,可能要分几条才能收到!"; SmsManager smsManager = SmsManager.getDefault(); List<String> divideContents = smsManager.divideMessage(content); for (String text : divideContents) { smsManager.sendTextMessage("5556", null, text, sentPI, deliverPI); } } }
发短信主要是调用了SmsManager,这里发送到了以后中文是乱码,不知道是虚拟机的原因还是确实需要字符编码。回家在自己手机上去试试是否仍是乱码。SmsManager的用法还没有捉摸过,不敢在此胡言乱语。需要用的人自己研究一下。