GCM 推送通知实现
Posted
技术标签:
【中文标题】GCM 推送通知实现【英文标题】:GCM Push Notification Implementation 【发布时间】:2015-09-29 02:02:00 【问题描述】:我似乎无法在我的应用程序中使用基本 GCM 包来实现推送通知。我已经尝试了几个教程,但到目前为止都无法让它工作。
我希望也许我能得到一些帮助,当我从 Ubuntu 服务器发送时,我没有收到任何通知。
主活动
import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.os.AsyncTask;
import android.os.Bundle;
import android.support.v7.app.ActionBarActivity;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import com.google.android.gms.gcm.GoogleCloudMessaging;
import com.google.android.gms.iid.InstanceID;
import java.io.IOException;
import java.net.HttpURLConnection;
import java.net.URL;
public class MainActivity extends ActionBarActivity
private String RegID = "";
private Context context = null;
private GoogleCloudMessaging gcm = null;
private InstanceID instanceID = null;
private BroadcastReceiver mHandleMesageReceiver;
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
context = getApplicationContext();
if(RegID.isEmpty())
register RegInBG = new register();
RegInBG.execute("e");
post task = new post();
task.execute(RegID);
mHandleMesageReceiver = new BroadcastReceiver()
@Override
public void onReceive(Context context, Intent intent)
;
/*Runnable task to be used to register
app in the GCM service for push
notifications, made by Devin Adams
*/
public class register extends AsyncTask<String, Void, String>
@Override
protected String doInBackground(String... params)
try
if (instanceID == null)
instanceID = InstanceID.getInstance(context);
if (gcm == null)
gcm = GoogleCloudMessaging.getInstance(context);
RegID = instanceID.getToken(getString(R.string.sender_id), gcm.INSTANCE_ID_SCOPE, null);
Log.e("***********", "************");
Log.e("***********", "************");
Log.e("RegID", RegID);
Log.e("***********", "************");
Log.e("***********", "************");
catch (IOException ex)
Log.e("***********", "************");
Log.e("***********", "************");
Log.e("Error", ex.getMessage());
Log.e("***********", "************");
Log.e("***********", "************");
return "sd";
public class post extends AsyncTask<String, Void, String>
@Override
protected String doInBackground(String... params)
try
String tempurl = getString(R.string.server_url);
Uri.Builder b = Uri.parse(tempurl).buildUpon();
String url = b.build().toString();
HttpURLConnection connection = (HttpURLConnection) ((new URL(url))).openConnection();
connection.setRequestMethod("POST");
connection.setDoInput(true);
connection.setDoOutput(true);
connection.connect();
connection.getOutputStream().write((params[0]).getBytes());
connection.disconnect();
catch (IOException ex)
Log.e("Error", ex.getMessage());
return "getg";
我的Gcm监听器
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.media.RingtoneManager;
import android.net.Uri;
import android.os.Bundle;
import android.support.v4.app.NotificationCompat;
import android.util.Log;
import com.google.android.gms.gcm.GcmListenerService;
public class MyGcmListener extends GcmListenerService
private static final String TAG = "International Studies";
@Override
public void onMessageReceived(String from, Bundle data)
String message = data.getString("message");
Log.d(TAG, "From: " + from);
Log.d(TAG, "Message: " + message);
Log.e("Message", "received ");
sendNotification(message);
private void sendNotification(String message)
Intent intent = new Intent(this, MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_ONE_SHOT);
Uri defaultSoundUri = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);
NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_launcher)
.setContentTitle("International Studies")
.setContentText(message)
.setAutoCancel(true)
.setSound(defaultSoundUri)
.setContentIntent(pendingIntent);
NotificationManager notificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
notificationManager.notify(0, notificationBuilder.build());
MyGcmMessageHandler
import android.app.IntentService;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.widget.Toast;
import com.google.android.gms.gcm.GoogleCloudMessaging;
import java.lang.Runnable;
import java.util.logging.Handler;
import java.util.logging.LogRecord;
public class MyGCMMessageHandler extends IntentService
String mes;
private Handler handler;
public MyGCMMessageHandler()
super("MyGCMMessageHandler");
public void onCreate()
super.onCreate();
this.handler = new Handler()
@Override
public void close()
@Override
public void flush()
@Override
public void publish(LogRecord record)
;
protected void onHandleIntent(Intent intent)
Bundle extras = intent.getExtras();
GoogleCloudMessaging gcm = GoogleCloudMessaging.getInstance(this);
String messageType = gcm.getMessageType(intent);
this.mes = extras.getString("title");
this.showToast();
Log.i("GCM", "Received: (" + messageType + ") " + mes);
MyGCMReceiver.completeWakefulIntent(intent);
public void showToast()
this.handler.post(new Runnable()
@Override
public void run()
Toast.makeText(MyGCMMessageHandler.this.getApplicationContext(), MyGCMMessageHandler.this.mes, Toast.LENGTH_LONG).show();
);
并且推送脚本是使用 python-gcm 运行的
from gcm import *
gcm = GCM("")
data = 'message': 'This is a test push', 'param2': 'value2'
reg_id = ''
gcm.plaintext_request(registration_id=reg_id, data=data)
GCM 在此应用程序中填充了 API 密钥和 reg_id 我手机的 reg_id。
谁能帮帮我?我不知道出了什么问题。
编辑:清单。
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="abdroid.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".MainActivity"
android:configChanges="keyboardHidden|orientation"
android:label="@string/app_name"
android:screenOrientation="portrait" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<receiver
android:name="com.google.android.gms.gcm.GcmReceiver"
android:exported="true"
android:permission="com.google.android.c2dm.permission.SEND" >
<intent-filter>
<action android:name="com.google.android.c2dm.intent.RECEIVE" />
<category android:name="edu.csusb.internationalstudies" />
</intent-filter>
</receiver>
<service android:name=".MyGcmListener" android:exported="false">
<intent-filter>
<action android:name="com.google.android.gms.iid.InstanceID"/>
</intent-filter>
</service>
</application>
【问题讨论】:
代码很多,但仍然缺少最重要的清单!请仅添加清单的相关部分,并查看是否可以稍微修剪一下代码。你正在做很多日志记录。他们告诉你什么? 我获得了 RegID,但没有注销消息通知。 除非我错了,这应该是你的实际包问题 1:您的 AndroidManifest.xml 设置错误。这条线应该改了。
<category android:name="gcm.play.android.samples.com.gcmquickstart" />
gcm.play.android.samples.com.gcmquickstart 应替换为您自己的包。
问题二: 你有错误的进口。
import java.util.logging.Handler;
不是你想要的。需要的是android.os.Handler
处理程序有两个主要用途:(1)调度消息和 在未来某个时间点执行的可运行文件; (2) 到 将要在与您自己的线程不同的线程上执行的操作排入队列。
【讨论】:
好的,已经解决了这两个问题,当我推送一个时仍然没有收到任何通知。 请接受这个答案并发布一个新问题(它解决的不是一个而是两个不同的问题!)。在该问题中包括相关日志消息,还包括服务器端代码的结果。 就像我说的,我没有从消息处理程序类中得到日志,所以不会有太多的日志,也没有来自服务器的反馈。 您现在需要查看发送 GCM 的服务器端脚本。以上是关于GCM 推送通知实现的主要内容,如果未能解决你的问题,请参考以下文章