- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
这是我在这里的第一个问题,如果我的问题中有什么不对,请提前道歉!
我目前在 MacBook 上的 Visual Studio 2013
中使用 Xamarin
,我创建了一个 PCL 项目,并创建了使推送通知工作所需的所有内容iOS
和 Android
。使用模拟器和真实设备,我可以通过服务器端程序实现这两者,我使用从客户端创建的设备 token 。几个小时内一切正常,但过了一会儿,两个 token 都过期了。我知道,因为这是我从服务器端给出的响应。
设备注册 ID 已过期:{e_bSR1SFBzg:APA91bHEQUEFvjn_mblu26MeKhAkcKRtXRLtaHOKYIsDTMDvJT3jmBDb4O6-hhu00ZhEcuAkpfaK-GKicbH1o5gtr3J9Sj9jyf-DVuikTd6DOBM0Tw70KFNVcLcL}
如果我从那个链接尝试 http://1-dot-sigma-freedom-752.appspot.com/gcmpusher.jsp
为了让它再次运行,我从设备和模拟器中删除了该应用程序,并在重新启动后创建了一个新的设备 token ,它运行了一段时间,但在几个小时后出现了同样的问题。我的疑问是关于
InstanceIDListenerService
可能没有正确配置的文件。下面是我的部分代码,所以如果有人可以给我任何建议,因为最近两天我一直面临这个问题,我不知道该怎么做。
我的 AndroidManifest.xml
文件:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="MY.PACKAGE.NAME" android:installLocation="auto" android:versionCode="1" android:versionName="1.0.0">
<uses-sdk android:minSdkVersion="14" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_LOCATION_EXTRA_COMMANDS" />
<uses-permission android:name="android.permission.ACCESS_MOCK_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />
<uses-permission android:name="MY.PACKAGE.NAME.permission.C2D_MESSAGE" />
<permission android:name="MY.PACKAGE.NAME.permission.C2D_MESSAGE" android:protectionLevel="signature" />
<!--<application android:label="TruckMe" android:icon="@drawable/ic_launcher">-->
<application android:label="TruckMe" android:icon="@drawable/ic_launcher">
<meta-data android:name="com.google.android.maps.v2.API_KEY" android:value="KEY_FOR-MAPS" />
<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" />
<action android:name="com.google.android.c2dm.intent.REGISTRATION" />
<category android:name="MY.PACKAGE.NAME" />
</intent-filter>
</receiver>
</application>
</manifest>
当然,我用那个词替换了我的真实包名称,Api_KEY 也用同样的东西替换。RegistrationIntentService.cs
文件:
using System;
using Android.App;
using Android.Content;
using Android.Util;
using Android.Gms.Gcm;
using Android.Gms.Gcm.Iid;
namespace TruckMe.Droid.Infrastructure.Notifications
{
[Service(Exported = false)]
public class RegistrationIntentService : IntentService
{
private const string SENDER_ID = "my_sender_num";
static object locker = new object();
public RegistrationIntentService() : base("RegistrationIntentService") { }
protected override void OnHandleIntent(Intent intent)
{
try
{
Log.Info("RegistrationIntentService", "Calling InstanceID.GetToken");
lock (locker)
{
// Initially this call goes out to the network to retrieve the token, subsequent calls are local.
var instanceID = InstanceID.GetInstance(this);
// See https://developers.google.com/cloud-messaging/android/start for details on this file.
var token = instanceID.GetToken(SENDER_ID, GoogleCloudMessaging.InstanceIdScope, null);
//Log.Info("RegistrationIntentService", "GCM Registration Token: " + token);
Console.WriteLine("RegistrationIntentService - GCM Token: " + token);
SendRegistrationToAppServer(token);
Subscribe(token);
//unSubscribe(token);
}
}
catch (Exception ex)
{
Log.Debug("RegistrationIntentService", "Failed to get a registration token. Ex: " + ex.Message);
return;
}
}
void SendRegistrationToAppServer(string token)
{
// Add custom implementation here as needed.
//Console.WriteLine("RegistrationIntentService - GCM Token: " + token);
// save the value to send it to our server to manage the notifications
App.UserPreferences.SetString("DeviceToken", token);
}
void Subscribe(string token)
{
var pubSub = GcmPubSub.GetInstance(this);
// subscribe to a topic group
pubSub.Subscribe(token, "/topics/global", null); // working
}
}
}
然后是 InstanceIDListenerService
文件:
using Android.App;
using Android.Content;
using Android.Gms.Gcm.Iid;
namespace TruckMe.Droid.Infrastructure.Notifications
{
[Service(Exported = false), IntentFilter(new[] { "com.google.android.gms.iid.InstanceID" })]
public class MyInstanceIdListenerService : InstanceIDListenerService
{
/**
* Called if InstanceID token is updated. This may occur if the security of the previous token had been compromised.
* This call is initiated by the InstanceID provider.
*/
public override void OnTokenRefresh()
{
// Fetch updated Instance ID token and notify our app's server of any changes (if applicable).
var intent = new Intent(this, typeof(RegistrationIntentService));
StartService(intent);
}
}
}
让我知道理解和分析问题的代码是否足够,提前致谢
[编辑]我也忘了添加调用注册服务的“MainActivity”文件:
using System;
using Android.App;
using Android.Content;
using Android.Content.PM;
using Android.Gms.Common;
using Android.Graphics.Drawables;
using Android.Views;
using Android.OS;
using Xamarin;
using TruckMe.Droid.Infrastructure;
using TruckMe.Droid.Infrastructure.Notifications;
namespace TruckMe.Droid
{
[Activity(Label = "TruckMe",
MainLauncher = false,
ConfigurationChanges = ConfigChanges.ScreenSize | ConfigChanges.Orientation,
WindowSoftInputMode = SoftInput.AdjustPan)]
public class MainActivity : global::Xamarin.Forms.Platform.Android.FormsApplicationActivity
{
protected override void OnCreate(Bundle bundle)
{
base.OnCreate(bundle);
if (IsPlayServicesAvailable())
{
var intent = new Intent(this, typeof(RegistrationIntentService));
StartService(intent);
}
global::Xamarin.Forms.Forms.Init(this, bundle);
FormsMaps.Init(this, bundle);
//Forms.SetTitleBarVisibility(AndroidTitleBarVisibility.Never);
LoadApplication(new App());
// below the code to remove the icon from android navigation bar
ActionBar.SetIcon(new ColorDrawable(Android.Graphics.Color.Transparent));
App.Init(new AndroidUserPreferences());
}
public bool IsPlayServicesAvailable()
{
int resultCode = GoogleApiAvailability.Instance.IsGooglePlayServicesAvailable(this);
if (resultCode != ConnectionResult.Success)
{
if (GoogleApiAvailability.Instance.IsUserResolvableError(resultCode))
Console.WriteLine(GoogleApiAvailability.Instance.GetErrorString(resultCode));
else
{
Console.WriteLine("Sorry, this device is not supported");
Finish();
}
return false;
}
else
{
Console.WriteLine("Google Play Services is available.");
return true;
}
}
}
}
最佳答案
我意识到让 token 过期不是时间问题,而是当您构建解决方案并传输应用程序时,它会进行某种卸载(保留旧数据),但如果应用程序已安装 GCM 不适用于“旧” token 。我清理了设备上的数据并在真实设备上下载了应用程序,在模拟器上做了同样的事情并发送了通知,两者都在工作。然后只需重新构建解决方案并仅在模拟器中下载;再次发送通知,但只有真实设备收到。
如果您正在调试并且不想每次都删除应用或数据,只需在下面调用注册服务的地方添加代码即可。
var instanceID = InstanceID.GetInstance(this);
// See https://developers.google.com/cloud-messaging/android/start for details on this file.
var token = instanceID.GetToken(SENDER_ID, GoogleCloudMessaging.InstanceIdScope, null);
#if DEBUG
instanceID.DeleteToken(token, GoogleCloudMessaging.InstanceIdScope);
instanceID.DeleteInstanceID();
#endif
token = instanceID.GetToken(SENDER_ID, GoogleCloudMessaging.InstanceIdScope, null);
这部分代码是从以下链接向我推荐的: fromXamarinForum
关于c# - 如何刷新一段时间后过期的 GCM 中的注册 token ?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36737850/
我有一个问题,想知道如何在已经使用常规登录/注册系统的网站上集成第 3 方登录(也进行静默注册)。 基本上当前登录还是比较规律的: 当用户进入网站 session 时类(class)决定他是否需要重新
我在学习 Hyperledger Fabric 并运行示例代码。 我仍在尝试正确了解事情的运作方式,尤其是在使用证书和加密 Material 的用户/管理员注册和注册中。 我想知道以下如何工作。 1)
我正在尝试使用从我的 Android 设备的 PassWallet 应用程序中保存的票证中获取的 token 向我的设备发送消息。 设备发送使用苹果钱包规范更新通行证所需的所有信息。但是,我正在使用
使用 Passport 本地示例,我可以登录工作。没有关于如何注册用户的文档。 我想为用户提供一个“电子邮件”和“密码”字段,他们可以使用它们来注册该网站。我怎样才能做到这一点?有什么原因没有记录下来
在之前的一些 WSO2IS 版本中,有一个默认的 self 注册功能。但是,我在 5.0 版本中找不到它。 阅读 WSO2IS 5.0 文档,我发现有 2 个用于此功能的 API: getUserId
我已将 Airship SDK 集成到 Android 应用程序中。在应用程序启动和飞艇起飞后,我在日志中得到以下调试信息: 07-27 12:46:31.916 XXX - UALib( 1545)
Delphi 中设计时包的可怕错误之一是以下错误,这意味着注册安装新组件到您的 Palette 上的包: Component TSomething can't be registered by pac
我发现的大部分内容都使用 php 或类似的东西。 我有一个 Angular 前端和 Node/express 服务器代码。还没有后端。我不确定如何继续用户注册。 最佳答案 在没有后端的情况下进行注册等
我正在使用 Drupal 6 开发一个网站。我正在使用我自己的主题,并且一切正常。现在我需要使用我的自定义主题在我的网站上创建一个自定义登录/注册表单。我尝试了很多方法,但一切都重定向到我的管理主题,
我在运行此代码时收到 Sip 异常。这是因为 manager.register(me,20,listener)。 下面是我的代码,所以请帮我更正这段代码。我正在使用 SipDemo 代码注册我的帐户。
在过去的几天里,我一直在尝试为基于 VUE 的 excel 制作任务 Pane 插件。 我已按照 link 的指南进行操作我试图为 onSelectionChange 注册一个事件处理程序。它已经有些
我需要在使用 django-registration 应用程序的登录表单中实现一个“记住我”按钮。任何ane可以帮助我向我展示这样做的方法吗? 谢谢 最佳答案 一种方法是更改 session 到期
我发现,如果您使用 Django 1.5 版本,则 django 注册模块会中断,因为在最新的 django 开发版本中,simple.py 类已被删除。 最佳答案 此问题现已修复: hg clone
我正在尝试将 facebook connect 实现到我的网站,但有几个问题。 1:是否可以使用用户当前的 Facebook 电子邮件/密码在我的网站上注册用户。 假设用户点击链接通过 faceboo
我使用 Facebook 注册来允许人们在我的网站上注册。有没有可能,在注册后,他在我的网站注册的成员(member)墙上会张贴? 最佳答案 这可能不是您正在寻找的答案,但我强烈建议您不要这样做。用户
I would like to use a slash (/) for a search during a vimscript, but I don't want to overwrite the "
我正在使用 jqgrid 并且有显示日期的列,但是来自服务器的日期以 json 格式出现,如下所示, "CommentedDate": "\/Date(1304324941000+0530)\/" 如
我希望用户可以直接登录主页,而不是在“../account/login/”页面上登录。我应该做什么才能使它成为可能?如何将主页上的输入字段与 allauth 连接?我不知道这样是否太复杂而无法以这种方
This question already has answers here: Understanding NSString comparison (7个答案) 5年前关闭。 我正在尝试制作注册表。有
我正在使用 Django 注册。它提供了处理 registration_form.html 的 View ,该 html 当前包含用户名、密码 1、密码 2 和电子邮件作为我的应用程序中的用户可输入字
我是一名优秀的程序员,十分优秀!