极光推送以及极光验证码
极光推送
配置
主要特点
产品功能说明
客户端维持连接占用资源少、耗电低
SDK丰富的接口,可定制通知栏提示样式
服务器大容量、稳定
jpush-android-release-3.x.y.zip 集成压缩包内容
AndroidManifest.xml
客户端嵌入SDK参考的配置文件
libs/jcore-android.v1.x.y.jar
极光开发者服务的核心包。
libs/jpush-android_v3.x.y.jar
JPush SDK 开发包。
libs/(cpu-type)/libjcore1xy.so
各种CPU类型的native开发包。
res
集成SDK必须添加的资源文件
example
是一个完整的 Android 项目,通过这个演示了 JPush SDK 的基本用法,可以用来做参考。
Android SDK 版本
目前SDK只支持Android 2.3或以上版本的手机系统。富媒体信息流功能则需Android3.0或以上版本的系统。
在这里我用到的是手动集成步骤
首先要在极光官网的后台创建一个应用
在这里要设置一下推送信息,我们点击完成推送设置
将自己创建的应用的包名添加到推送设置的应用包名当中
解压缩 jpush-android-release-3.x.y.zip 集成压缩包。
复制 libs/jcore-android_v1.x.y.jar 到工程 libs/ 目录下。
复制 libs/jpush-android_v3.x.y.jar 到工程 libs/ 目录下。
复制 libs/(cpu-type)/libjcore1xy.so 到你的工程中存放对应cpu类型的目录下。
复制 res/ 中drawable-hdpi, layout, values文件夹中的资源文件到你的工程中 res/ 对应同名的目录下。
说明 1:若没有res/drawable-xxxx/jpush_notification_icon这个资源默认使用应用图标作为通知icon,在5.0以上系统将应用图标作为statusbar icon可能显示不正常,用户可定义没有阴影和渐变色的icon替换这个文件,文件名不要变。
说明 2:使用android studio的开发者,如果使用jniLibs文件夹导入so文件,则仅需将所有cpu类型的文件夹拷进去;如果将so文件添加在module的libs文件夹下,注意在module的gradle配置中添加一下配置:
在这里要提一下一些小bug,我们创建的应用可能没有libs文件夹,在这里我们要手动创建
点击Project在app目录下创建libs包,最后在gradle下配置一下下面的代码,即可以生成。而且在Android视图下,也可以看到有个jniLibs的文件夹,这样我们运行应用时就不会包找不到so文件的异常了
android {
......
sourceSets {
main {
jniLibs.srcDirs = ['libs']
......
}
......
}
......
}
配置 AndroidManifest.xml
根据 SDK 压缩包里的 AndroidManifest.xml 样例文件,来配置应用程序项目的 AndroidManifest.xml 。
主要步骤为:
复制备注为 "Required" 的部分
将标注为“您应用的包名”的部分,替换为当前应用程序的包名
将标注为“您应用的Appkey”的部分,替换为在Portal上注册该应用的的Key,例如:9fed5bcb7b9b87413678c407
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="您应用的包名"
android:versionCode="303"
android:versionName="3.0.3"
>
<uses-sdk android:minSdkVersion="9" android:targetSdkVersion="23" />
<!-- Required -->
<permission
android:name="您应用的包名.permission.JPUSH_MESSAGE"
android:protectionLevel="signature" />
<!-- Required -->
<uses-permission android:name="您应用的包名.permission.JPUSH_MESSAGE" />
<uses-permission android:name="android.permission.RECEIVE_USER_PRESENT" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.WRITE_SETTINGS" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<!-- Optional. Required for location feature -->
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" /> <!-- 用于开启 debug 版本的应用在6.0 系统上 层叠窗口权限 -->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<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.CHANGE_NETWORK_STATE" />
<uses-permission android:name="android.permission.GET_TASKS" />
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:name="Your Application Name">
<!-- Required SDK 核心功能-->
<!-- 可配置android:process参数将PushService放在其他进程中 -->
<service
android:name="cn.jpush.android.service.PushService"
android:enabled="true"
android:exported="false" >
<intent-filter>
<action android:name="cn.jpush.android.intent.REGISTER" />
<action android:name="cn.jpush.android.intent.REPORT" />
<action android:name="cn.jpush.android.intent.PushService" />
<action android:name="cn.jpush.android.intent.PUSH_TIME" />
</intent-filter>
</service>
<!-- since 1.8.0 option 可选项。用于同一设备中不同应用的JPush服务相互拉起的功能。 -->
<!-- 若不启用该功能可删除该组件,将不拉起其他应用也不能被其他应用拉起 -->
<service
android:name="cn.jpush.android.service.DaemonService"
android:enabled="true"
android:exported="true">
<intent-filter >
<action android:name="cn.jpush.android.intent.DaemonService" />
<category android:name="您应用的包名"/>
</intent-filter>
</service>
<!-- Required SDK核心功能-->
<receiver
android:name="cn.jpush.android.service.PushReceiver"
android:enabled="true" >
<intent-filter android:priority="1000">
<action android:name="cn.jpush.android.intent.NOTIFICATION_RECEIVED_PROXY" />
<category android:name="您应用的包名"/>
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.USER_PRESENT" />
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
</intent-filter>
<!-- Optional -->
<intent-filter>
<action android:name="android.intent.action.PACKAGE_ADDED" />
<action android:name="android.intent.action.PACKAGE_REMOVED" />
<data android:scheme="package" />
</intent-filter>
</receiver>
<!-- Required SDK核心功能-->
<activity
android:name="cn.jpush.android.ui.PushActivity"
android:configChanges="orientation|keyboardHidden"
android:theme="@android:style/Theme.NoTitleBar"
android:exported="false" >
<intent-filter>
<action android:name="cn.jpush.android.ui.PushActivity" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="您应用的包名" />
</intent-filter>
</activity>
<!-- SDK核心功能-->
<activity
android:name="cn.jpush.android.ui.PopWinActivity"
android:configChanges="orientation|keyboardHidden"
android:exported="false"
android:theme="@style/MyDialogStyle">
<intent-filter>
<category android:name="android.intent.category.DEFAULT" />
<category android:name="您应用的包名" />
</intent-filter>
</activity>
<!-- Required SDK核心功能-->
<service
android:name="cn.jpush.android.service.DownloadService"
android:enabled="true"
android:exported="false" >
</service>
<!-- Required SDK核心功能-->
<receiver android:name="cn.jpush.android.service.AlarmReceiver" />
<!-- Required since 3.0.7 -->
<!-- 新的tag/alias接口结果返回需要开发者配置一个自定的广播 -->
<!-- 该广播需要继承JPush提供的JPushMessageReceiver类, 并如下新增一个 Intent-Filter -->
<receiver
android:name="自定义 Receiver"
android:enabled="true" >
<intent-filter>
<action android:name="cn.jpush.android.intent.RECEIVE_MESSAGE" />
<category android:name="您应用的包名" />
</intent-filter>
</receiver>
<!-- User defined. 用户自定义的广播接收器-->
<receiver
android:name="您自己定义的Receiver"
android:enabled="true">
<intent-filter>
<!--Required 用户注册SDK的intent-->
<action android:name="cn.jpush.android.intent.REGISTRATION" />
<!--Required 用户接收SDK消息的intent-->
<action android:name="cn.jpush.android.intent.MESSAGE_RECEIVED" />
<!--Required 用户接收SDK通知栏信息的intent-->
<action android:name="cn.jpush.android.intent.NOTIFICATION_RECEIVED" />
<!--Required 用户打开自定义通知栏的intent-->
<action android:name="cn.jpush.android.intent.NOTIFICATION_OPENED" />
<!-- 接收网络变化 连接/断开 since 1.6.3 -->
<action android:name="cn.jpush.android.intent.CONNECTION" />
<category android:name="您应用的包名" />
</intent-filter>
</receiver>
<!-- Required. For publish channel feature -->
<!-- JPUSH_CHANNEL 是为了方便开发者统计APK分发渠道。-->
<!-- 例如: -->
<!-- 发到 Google Play 的APK可以设置为 google-play; -->
<!-- 发到其他市场的 APK 可以设置为 xxx-market。 -->
<!-- 目前这个渠道统计功能的报表还未开放。-->
<meta-data android:name="JPUSH_CHANNEL" android:value="developer-default"/>
<!-- Required. AppKey copied from Portal -->
<meta-data android:name="JPUSH_APPKEY" android:value="您应用的Appkey"/>
</application>
</manifest>
我们目前没有做到两个自定义receiver的配置信息,目前阶段我们可以给注释掉,这样manifest最后配置完成,就不错报错了
将所有信息配置完成后,在自定义类继承自Application下声明方法
在此之前,我们manifest中要声明一下application的name值
public class ExampleApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
JPushInterface.setDebugMode(true);
JPushInterface.init(this);
}
}
就算是配置完成了,这样我们在相对应的极光官网上面的控制台,即可以发送推送。
比如:
我们点开控制台
点击广播所有人,平台是安卓平台,编辑一下通知内容,点击发送通知后,如果发送成功,会弹出一个Dialog提示成功
我们来看一下模拟器
极光短信验证码Demo
因为我们已经配置过了Appkey,权限等信息,所以当前我们只需要下载极光短信的jar即可
https://docs.jiguang.cn/jsms/resources/
极光短信的API不多,只需要记住几个就行,下面我将通过一个Demo展示一下注册模块加上验证码的逻辑\
与极光推送一样,我们要在全局Application初始化
@Override
public void onCreate() {
super.onCreate();
context = getApplicationContext();
handler = new Handler();
//获取主线程的id
mainThreadId = Process.myTid();
//初始化JPush
JPushInterface.setDebugMode(true);
JPushInterface.init(this);
//初始化极光短信
SMSSDK.getInstance().initSdk(this);
SMSSDK.getInstance().setDebugMode(true);
SMSSDK.getInstance().setIntervalTime(60000);
}
上面的三行代码就是初始化极光短信。
setIntervalTime就是设置短信验证码的间隔时间。
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">
<include layout="@layout/title" />
<LinearLayout
android:id="@+id/ll_et_login"
android:layout_width="match_parent"
android:layout_height="210dp"
android:layout_marginTop="130dp"
android:orientation="vertical"
android:paddingLeft="10dp"
android:paddingRight="10dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:src="@mipmap/ic_phone" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:layout_marginLeft="3dp"
android:text="手机号"
android:textColor="#666" />
<android.support.v7.widget.AppCompatEditText
android:id="@+id/et_tel_register"
style="@style/et_lab"
android:hint="请输入手机号"
android:inputType="number" />
</LinearLayout>
<View style="@style/ViewLine" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="50dp"
android:layout_marginRight="10dp">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:src="@mipmap/ic_psd_locked" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:layout_marginLeft="3dp"
android:text="验证码"
android:textColor="#666" />
<android.support.v7.widget.AppCompatEditText
android:id="@+id/et_verification_register"
style="@style/et_lab"
android:layout_weight="1"
android:hint="请输入验证码"
android:inputType="number" />
<Button
android:id="@+id/btn_verification_register"
android:layout_width="wrap_content"
android:layout_height="40dp"
android:layout_gravity="center_vertical"
android:background="@drawable/selector_btn_login"
android:padding="10dp"
android:text="请输入验证码"
android:textColor="#fff" />
</LinearLayout>
<View style="@style/ViewLine" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:src="@mipmap/ic_psd_locked" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:layout_marginLeft="3dp"
android:text="密 码"
android:textColor="#666" />
<android.support.v7.widget.AppCompatEditText
android:id="@+id/et_psd_register"
style="@style/et_lab"
android:hint="密码为6至12个数字或字母" />
</LinearLayout>
<View style="@style/ViewLine" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:src="@mipmap/ic_psd_locked" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:layout_marginLeft="3dp"
android:text="密 码"
android:textColor="#666" />
<android.support.v7.widget.AppCompatEditText
android:id="@+id/et_psd_confirm_register"
style="@style/et_lab"
android:hint="请再次确认密码" />
</LinearLayout>
<View style="@style/ViewLine" />
</LinearLayout>
<RelativeLayout
android:id="@+id/rl_login_text"
android:layout_width="match_parent"
android:layout_height="30dp"
android:layout_below="@id/ll_et_login">
<RelativeLayout
android:id="@+id/rl_isCheck"
android:layout_width="wrap_content"
android:layout_height="30dp"
android:layout_alignParentLeft="true"
android:layout_marginLeft="20dp">
<CheckBox
android:id="@+id/cb_consent"
android:layout_width="30dp"
android:layout_height="30dp"
android:checked="true"
android:shadowColor="#4581c7" />
<TextView
android:id="@+id/tv_confirm_register"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_toRightOf="@id/cb_consent"
android:text="我已阅读并同意"
android:textColor="#999999"
android:textSize="14sp" />
</RelativeLayout>
<TextView
android:id="@+id/tv_protocol_register"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_toRightOf="@id/rl_isCheck"
android:text="@string/register"
android:textColor="#ff0000"
android:textSize="14sp" />
</RelativeLayout>
<Button
android:id="@+id/btn_register"
style="@style/ButtonLoginStyle"
android:layout_below="@id/rl_login_text"
android:layout_marginTop="50dp"
android:text="注册" />
</RelativeLayout>
这是注册模块的界面
这是RegisterActivity注册界面的逻辑,在这里,点击按钮我用了安卓的一个CountDownTimer类方法实现
package com.qryl.qryl.activity.login;
import android.app.ProgressDialog;
import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.AppCompatEditText;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.CheckBox;
import android.widget.EditText;
import android.widget.RelativeLayout;
import android.widget.TextView;
import android.widget.Toast;
import com.qryl.qryl.R;
import com.qryl.qryl.activity.MainActivity;
import com.qryl.qryl.util.VerificationCountDownTimer;
import java.io.IOException;
import cn.jpush.sms.SMSSDK;
import cn.jpush.sms.listener.SmscheckListener;
import cn.jpush.sms.listener.SmscodeListener;
import okhttp3.Call;
import okhttp3.Callback;
import okhttp3.MultipartBody;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.RequestBody;
import okhttp3.Response;
public class RegisterActivity extends AppCompatActivity {
private static final String TAG = "RegisterActivity";
private Button btnVerification;
private AppCompatEditText etVerification;
private AppCompatEditText etPsd;
private AppCompatEditText etPsdComfirm;
private RelativeLayout rlIsCheck;
private CheckBox checkBox;
private TextView tvProtocol;
private Button btnRegisiter;
private AppCompatEditText etTel;
private ProgressDialog progressDialog;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_register);
initView();
}
private void initView() {
progressDialog = new ProgressDialog(this);
btnVerification = (Button) findViewById(R.id.btn_verification_register);
etTel = (AppCompatEditText) findViewById(R.id.et_tel_register);
etVerification = (AppCompatEditText) findViewById(R.id.et_verification_register);
etPsd = (AppCompatEditText) findViewById(R.id.et_psd_register);
etPsdComfirm = (AppCompatEditText) findViewById(R.id.et_psd_confirm_register);
rlIsCheck = (RelativeLayout) findViewById(R.id.rl_isCheck);
checkBox = (CheckBox) findViewById(R.id.cb_consent);
tvProtocol = (TextView) findViewById(R.id.tv_protocol_register);
btnRegisiter = (Button) findViewById(R.id.btn_register);
//根据cb判断是否点击了对号
rlIsCheck.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
boolean checked = checkBox.isChecked();
//根据checkBox的状态修改样式
if (checked) {
checkBox.setChecked(false);
checkBox.setBackgroundResource(R.mipmap.ic_frame_false);
} else {
checkBox.setChecked(true);
checkBox.setBackgroundResource(R.mipmap.ic_frame_true);
}
}
});
//实现倒计时并发送请求并获取验证码的功能
receiverSMS();
//注册
btnRegisiter.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//点击注册的时候要确认所有的注册信息不为空
registerIfNotNull();
}
});
}
/**
* 点击注册的时候要确认所有的注册信息不为空
*/
private void registerIfNotNull() {
if (TextUtils.isEmpty(etTel.getText().toString())) {
Toast.makeText(this, "请输入正确的手机号!", Toast.LENGTH_SHORT).show();
} else {
if (TextUtils.isEmpty(etPsd.getText().toString())) {
Toast.makeText(this, "请输入注册密码!", Toast.LENGTH_SHORT).show();
} else {
if (!etPsd.getText().toString().equals(etPsdComfirm.getText().toString())) {
Toast.makeText(this, "确认密码与密码不符!", Toast.LENGTH_SHORT).show();
} else {
if (!checkBox.isChecked()) {
Toast.makeText(this, "请同意用户注册协议!", Toast.LENGTH_SHORT).show();
} else {
progressDialog.setMessage("正在注册...");
progressDialog.show();
checkSMS();
}
}
}
}
}
/**
* 验证码通过正确,就注册发送服务器
*/
private void checkSMS() {
SMSSDK.getInstance().checkSmsCode(etTel.getText().toString(), etVerification.getText().toString(), new SmscheckListener() {
@Override
public void checkCodeSuccess(String s) {
Log.i(TAG, "checkCodeSuccess: 验证成功:" + s);
postData();
}
@Override
public void checkCodeFail(int i, String s) {
Log.i(TAG, "checkCodeSuccess: 验证失败:" + s);
runOnUiThread(new Runnable() {
@Override
public void run() {
if (progressDialog.isShowing() && progressDialog != null) {
progressDialog.dismiss();
}
Toast.makeText(RegisterActivity.this, "验证码验证失败!", Toast.LENGTH_SHORT).show();
}
});
}
});
}
/**
* 向服务器发送注册信息
*/
private void postData() {
OkHttpClient client = new OkHttpClient();
MultipartBody.Builder builder = new MultipartBody.Builder().setType(MultipartBody.FORM);
builder.addFormDataPart("password", etPsdComfirm.getText().toString());
builder.addFormDataPart("mobile", etTel.getText().toString());
MultipartBody requestBody = builder.build();
Request requset = new Request.Builder().url("").post(requestBody).build();
client.newCall(requset).enqueue(new Callback() {
@Override
public void onFailure(Call call, IOException e) {
Log.i(TAG, "onFailure: 服务器连接失败");
runOnUiThread(new Runnable() {
@Override
public void run() {
if (progressDialog.isShowing() && progressDialog != null) {
progressDialog.dismiss();
}
Toast.makeText(RegisterActivity.this, "注册失败!", Toast.LENGTH_SHORT).show();
}
});
}
@Override
public void onResponse(Call call, Response response) throws IOException {
Log.i(TAG, "onResponse: 服务器连接成功" + response.body().string());
runOnUiThread(new Runnable() {
@Override
public void run() {
if (progressDialog.isShowing() && progressDialog != null) {
progressDialog.dismiss();
}
Toast.makeText(RegisterActivity.this, "注册成功!", Toast.LENGTH_SHORT).show();
Intent intent = new Intent(RegisterActivity.this, MainActivity.class);
startActivity(intent);
}
});
}
});
}
/**
* 实现倒计时并发送请求并获取验证码的功能
*/
private void receiverSMS() {
btnVerification.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//实现倒计时
countDownTimer();
//请求验证码
String tel = etTel.getText().toString();
SMSSDK.getInstance().getSmsCode(tel, "1", new SmscodeListener() {
@Override
public void getCodeSuccess(String s) {
//获取成功时
Log.i(TAG, "获取验证码成功! " + s);
}
@Override
public void getCodeFail(int i, String s) {
//获取失败时
Log.i(TAG, "获取验证码失败! " + s);
}
});
}
});
}
/**
* 倒计时功能
*/
private void countDownTimer() {
VerificationCountDownTimer timer = new VerificationCountDownTimer(60000, 1000) {
//倒计时过程
@Override
public void onTick(long millisUntilFinished) {
super.onTick(millisUntilFinished);
btnVerification.setClickable(false);//防止重新倒计时
btnVerification.setText(millisUntilFinished / 1000 + "s");
}
//计时完毕的方法
@Override
public void onFinish() {
//重新给Button设置文字
btnVerification.setText("重新获取验证码");
//设置可点击
btnVerification.setClickable(true);
}
};
timer.start();
}
}
CountTimer类
package com.qryl.qryl.util;
import android.os.CountDownTimer;
/**
* Created by hp on 2017/8/28.
*/
public class VerificationCountDownTimer extends CountDownTimer {
/**
* @param millisInFuture The number of millis in the future from the call
* to {@link #start()} until the countdown is done and {@link #onFinish()}
* is called.
* @param countDownInterval The interval along the way to receive
* {@link #onTick(long)} callbacks.
*/
public VerificationCountDownTimer(long millisInFuture, long countDownInterval) {
super(millisInFuture, countDownInterval);
}
@Override
public void onTick(long millisUntilFinished) {
}
@Override
public void onFinish() {
}
}
这样,当我们输入手机号注册发送验证码即会收到验证码
大家可以看一下验证码样式,这是一分钟过期的。可是我们平常注册,1分钟的时间是不是太短了呢,而且文字什么的也是极光推送也不是自己定义的
我们可以点击极光推送的控制台
打开新建模板,可以设置短信有效期和短信模板
网友评论