設(shè)計(jì)思路
創(chuàng)新互聯(lián)是一家專業(yè)提供橫峰企業(yè)網(wǎng)站建設(shè),專注與網(wǎng)站制作、網(wǎng)站設(shè)計(jì)、HTML5、小程序制作等業(yè)務(wù)。10年已為橫峰眾多企業(yè)、政府機(jī)構(gòu)等服務(wù)。創(chuàng)新互聯(lián)專業(yè)網(wǎng)站制作公司優(yōu)惠進(jìn)行中。
如果在開(kāi)發(fā)當(dāng)中需要做push接入,而產(chǎn)品又要求不允許用第三方sdk(百度push,友盟push),而且又沒(méi)有網(wǎng)絡(luò)編程的經(jīng)驗(yàn),這個(gè)時(shí)候怎么辦?這里就給大家分享下用http請(qǐng)求解決這個(gè)問(wèn)題。
大體的設(shè)計(jì)思路是,寫(xiě)一個(gè)service(為了保證長(zhǎng)時(shí)間運(yùn)行不被kill,就將其定義到另外的進(jìn)程當(dāng)中去),在這個(gè)service里面啟動(dòng)鬧鐘,每隔一段時(shí)間(這個(gè)時(shí)間可以自己定義)去請(qǐng)求服務(wù)器,如果有新的push消息,就通知給用戶。
具體實(shí)現(xiàn)
貌似很簡(jiǎn)單定義一個(gè)鬧鐘不斷輪循請(qǐng)求服務(wù)器一句話,卻在實(shí)際開(kāi)發(fā)中要考慮很多問(wèn)題,下面簡(jiǎn)單給大家列出來(lái)。
1)鬧鐘時(shí)間校準(zhǔn)
2)每天push只能在固定的或者某個(gè)定義的時(shí)間內(nèi)推送
3)push類型的擴(kuò)展,新添加一種類型的push
什么都不說(shuō),類圖直接貼上
大概分為三級(jí)結(jié)構(gòu)
第一級(jí):MoboPushAlarmManager,主要工作是管理鬧鐘,初始化鬧鐘,對(duì)鬧鐘時(shí)間進(jìn)行校準(zhǔn),取消鬧鐘。
第二級(jí):MobogeniePushServiceNew,主要工作是對(duì)鬧鐘設(shè)置的pendingintent的動(dòng)作(startservice)進(jìn)行處理。
第三級(jí):MoboPushNotifyHelper,主要工作是實(shí)例化notification對(duì)象,發(fā)送出通知,并且取消通知。MoboMessageSelector,主要工作是從n多個(gè)push中根據(jù)時(shí)間選擇出可用的push。MoboPushRequest,主要是請(qǐng)求服務(wù)器,從服務(wù)器列表中獲取push隊(duì)列。
其余的PushActionCreator,MoboPushMessage等等都屬于第三級(jí)或者第一第二級(jí)的工具類了。
開(kāi)始來(lái)貼代碼了(具體的源碼還沒(méi)抽出來(lái),過(guò)段時(shí)間再貼上)
MoboPushAlarmManager來(lái)初始化鬧鐘,當(dāng)鬧鐘響了就會(huì)發(fā)送一個(gè)intent給MobogeniePushServiceNew
public void initPullAlarm(Context context, boolean boot) { Bundle bundle = new Bundle(); bundle.putInt(START_SERVICE_TYPE, TYPE_REQUEST); PendingIntent pendingIntent = getPendingIntent(context, bundle, REQ_PULL); //循環(huán)時(shí)間 long repeat_time = HOUR_MILLIS * 2; Log.i(TAG, "repeat_time is " + repeat_time); // 計(jì)算下一次執(zhí)行時(shí)間 long triggerAtMillis = boot ? 10000 : 0; Log.i(TAG, "initPullAlarm,and next pull time is after: " + triggerAtMillis); // 這個(gè)行為會(huì)覆蓋之前的Alarm,主要根據(jù)PendingIntent來(lái)辨別不同的鬧鐘 getAlarmManager(context).setRepeating(AlarmManager.RTC, System.currentTimeMillis() + triggerAtMillis, repeat_time, pendingIntent); }
MobogeniePushServiceNew接收到鬧鐘之后,就會(huì)解析intent,并且進(jìn)行相應(yīng)的操作(請(qǐng)求網(wǎng)絡(luò),發(fā)送通知)
@Override public void onStart(Intent intent, int startId) { super.onStart(intent, startId); LogUtil.p("pushservice onStart"); if (intent == null) { LogUtil.d("mobopush", "intent == null)"); return; } // 解析打開(kāi)service的意圖 parsePushServiceIntent(intent); } private void parsePushServiceIntent(Intent intent) { Log.i(TAG, "parsePushServiceIntent"); if (intent == null) { return; } Bundle bundle = intent.getExtras(); if (bundle == null) { // 不明渠道調(diào)起service,默認(rèn)處理為準(zhǔn)備獲取新消息,重設(shè)鬧鐘 PushAlarmManager.getInstance().initPullAlarm(this, false); return; } int type = bundle.getInt(PushAlarmManager.START_SERVICE_TYPE); if (type == PushAlarmManager.TYPE_STARTSERVICE) { //判斷鬧鐘是否過(guò)期,如果過(guò)期則重設(shè) } else if (type == PushAlarmManager.TYPE_REQUEST) { // 預(yù)設(shè)的網(wǎng)絡(luò)請(qǐng)求 mREQ_RESULT = REQ_RESULT.ING; MoboPushRequest.getInstance().pullPushMessages(this, this, MoboPushRequest.REQUEST_NORMAL); } } //請(qǐng)求網(wǎng)絡(luò)回調(diào)的數(shù)據(jù)處理 @Override public void onMessageLoaded(int actionCode, int requestCode, MessageResponse response) { //將網(wǎng)絡(luò)請(qǐng)求回來(lái)的結(jié)果利用MessageSelector選擇器,選擇出有用的消息 getMessageSelector().assignMessageFromNet(MobogeniePushServiceNew.this, new MessageGetListener() { @Override public void showMessages(Listmsgs) { if (msgs.size() > 0) { for (MoboPushMessage msg : msgs) { notifyMessageAndDelete(msg, false); } } } @Override public void prepareMessages(List msgs) { if (msgs == null || msgs.size() == 0) { return; } MoboPushMessageDBUtils.insertBatch(MobogeniePushServiceNew.this, msgs); initShowMessageAlarm(msgs); } }, response.messages); }
MoboPushRequest去拉取新的消息
沒(méi)有寫(xiě)網(wǎng)絡(luò)請(qǐng)求的部分,大家明白意思就行
public void pullPushMessages(Context context, final IMessageGetListener l, final int requestCode) { boolean pushFlag = true; if (!pushFlag) { return; } final Context appcontext = context.getApplicationContext(); //這里進(jìn)行http請(qǐng)求,得到j(luò)son數(shù)據(jù) String json = ""; if (!TextUtils.isEmpty(json)) { JSONObject jsonObj = null; String str = null; try { jsonObj = new JSONObject(json); if (jsonObj.optInt("code") == 100) { int interval = jsonObj.optInt("interval"); MessageResponse response = new MessageResponse(); response.interval = interval * 1000; JSONArray jsonArray = jsonObj.optJSONArray("list"); if (jsonArray == null || jsonArray.length() == 0) { } int aLength = jsonArray.length(); response.initMessageArray(aLength); response.resCode = 100; for (int i = 0; i < aLength; i++) { JSONObject jsonInArray = jsonArray.getJSONObject(i); str = jsonInArray.toString(); MoboPushMessage pushMessage = new MoboPushMessage(); pushMessage.parseJson(str); response.messages[i] = pushMessage; if (pushMessage != null ) { } } } else { } } catch (JSONException e) { } } Object object=null; int actionCode=0; if (l == null) { return; } if (actionCode==0 && object != null && object instanceof MessageResponse) { MessageResponse response = (MessageResponse) object; l.onMessageLoaded(actionCode, requestCode, response); } else { l.onMessageLoaded(actionCode, requestCode, null); } }
剩下的就是處理具體的push消息了
private void notifyMessageAndDelete(MoboPushMessage message, boolean delete) { if (message == null) { return; } //傳入message對(duì)象發(fā)送通知 getMoboPushNotifyHelper().showNotification(message); }
我們只有message對(duì)象怎樣利用這個(gè)message對(duì)象去發(fā)送不同類型的push呢,為了很好的擴(kuò)展,直接上代碼
public void showNotification(MoboPushMessage message) { final Intent targetIntent = obtainNotificationClick(message, false); setNotification(message, targetIntent); } public void setNotification(final MoboPushMessage message, final Intent intent) { //根據(jù)MoboPushMessage和Intent來(lái)show通知 }
重點(diǎn)就在這句話了,怎樣根據(jù)message檢測(cè)出需要的push呢
final Intent targetIntent = obtainNotificationClick(message, false);
我們可以看見(jiàn)getTargetIntent(mContext, message)這個(gè)方法主要作用
public Intent obtainNotificationClick(MoboPushMessage message, boolean flag) { if (message == null) { return null; } PushActionCreator actionCreator = new PushActionCreator(); return actionCreator.getTargetIntent(mContext, message); }
核心在這里
public Intent getTargetIntent(Context c,MoboPushMessage pushMessage){ if(pushMessage==null) { return null; } int type = pushMessage.type; Intent resultIntent = null; for(IPushIntentAction action:mPushActions){ if(action.isTypeForAction(type)){ try{ resultIntent = action.createTargetIntent(c,pushMessage); }catch(Exception e){ } break; } } return resultIntent; }
將不同類型的push對(duì)象添加到這個(gè)集合里面去
ArrayListmPushActions;
也就是所有的push數(shù)據(jù)都要實(shí)現(xiàn)這個(gè)公共的接口
public interface IPushIntentAction { public Intent createTargetIntent(Context context,MoboPushMessage message); public int getActionKey(); public boolean isTypeForAction(int type); public String getNextPage(); }
其中createTargetIntent這個(gè)根據(jù)不同的類型new出各自的intent,isTypeForAction比對(duì)類型
舉個(gè)例子大家就知道了。
public class PushActionDefault implements IPushIntentAction { @Override public Intent createTargetIntent(Context c, MoboPushMessage message) { Intent intent = new Intent(); intent.setAction("android.intent.action.MAIN"); intent.addCategory("android.intent.category.LAUNCHER"); return intent; } @Override public int getActionKey() { return MoboPushMessage.TYPE_OPEN_APP; } @Override public String getNextPage() { return null; } @Override public boolean isTypeForAction(int type) { return false; } }
基本上把一條線給貫穿了,其中
1)鬧鐘時(shí)間校準(zhǔn)
解決方法可以自由的寫(xiě)在初始化鬧鐘里面,如果發(fā)現(xiàn)時(shí)間有錯(cuò)誤,可以再初始化一下鬧鐘,這樣就可以覆蓋以前的鬧鐘
2)每天push只能在固定的或者某個(gè)定義的時(shí)間內(nèi)推送
我們?cè)贛oboMessageSelector選擇器里面專門對(duì)時(shí)間進(jìn)行了篩選
3)push類型的擴(kuò)展,新添加一種類型的push
我們要添加新的類型的push的時(shí)候,只需要實(shí)現(xiàn)IPushIntentAction 這個(gè)接口,然后在將對(duì)象添加到ArrayList mPushActions這個(gè)集合里面,供選擇就OK了
最近比較忙,之后一定把整理好的demo分享給大家。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。