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,但没有注销消息通知。 除非我错了,这应该是你的实际包 好吧,我把它改成了包名,现在在 MyGcmMessageHandler 下它说“找不到符号方法 post() 这是一个不同的问题。你确定你有正确的 Handler 和 Runnable 导入吗? 【参考方案1】:

问题 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 推送通知实现的主要内容,如果未能解决你的问题,请参考以下文章

使用 GCM 在 IO 中实现推送通知

GCM 推送通知的问题

推送通知:从 GCM 更改为 Parse

如何使用 php 在 android studio 中使用 GCM 实现推送通知?

本地化应用程序的 Android GCM 推送通知

使用 Rhodes 和 GCM 推送通知