在查看了几个资源和问题之后,我仍然遇到检测传入短信的问题。
以下代码显示了基础知识:
广播接收器类,显示吐司onReceive
public class IncomingSms extends BroadcastReceiver {
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "something received", Toast.LENGTH_SHORT).show();
}
}
注册接收者和权限的简单清单
<application
<receiver
android:name=".IncomingSms"
android:permission="android.permission.BROADCAST_SMS"
android:exported="true">
<intent-filter android:priority="2147483647" >
<action android:name="android.provider.Telephony.SMS_RECEIVED" />
</intent-filter>
</receiver>
</application>
<uses-permission android:name="android.permission.RECEIVE_SMS" />
<uses-permission android:name="android.permission.READ_SMS" />
<uses-permission android:name="android.permission.SEND_SMS" />
上面的代码声明并注册接收器,并具有适当的权限。此外,优先级设置为MAX_INT或2147483647。
我的设备是Nexus 6P,安装了默认的Messenger应用程序(我也尝试过环聊)。该应用程序仍然不显示我的祝酒词。尝试使用较旧的三星设备后,正确打印了祝酒词。
优先问题
我在6P上安装了一个名为 Manifest Viewer 的应用程序,它允许我查看设备上安装的应用程序的manifest.xml。我检查了Messenger和Hangouts的清单,为SMS标签的接收者,并发现他们两个也指定优先级2147483647.似乎这两个信使应用程序最大优先级,并且一旦他们消费该消息,他们不允许其他应用程序进行干预。请注意,这些是谷歌应用程序的库存,而不是第三方。
此时,我很困惑:
非常感谢
答案 0 :(得分:25)
好的问题已经解决了。 问题不在于优先事项,而是我的手机是Nexus 6P(a.k.a.API 23)。
仅在manifest.xml中提供权限是不够的,我不得不为运行时权限请求添加代码。见Android documentation for runtime permissions
将此代码添加到您的MainActiviy:
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.RECEIVE_SMS},
MY_PERMISSIONS_REQUEST_SMS_RECEIVE);
在MainActivity类的顶部定义它:
private int MY_PERMISSIONS_REQUEST_SMS_RECEIVE = 10;
并添加此覆盖:
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == MY_PERMISSIONS_REQUEST_SMS_RECEIVE) {
// YES!!
Log.i("TAG", "MY_PERMISSIONS_REQUEST_SMS_RECEIVE --> YES");
}
}
答案 1 :(得分:2)
请确保您的实施是这样的。
SMS Receiver类
public class SmsReceiver extends BroadcastReceiver {
private static final String SMS_RECEIVED = "android.provider.Telephony.SMS_RECEIVED";
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(SMS_RECEIVED)) {
Bundle bundle = intent.getExtras();
if (bundle != null) {
// get sms objects
Object[] pdus = (Object[]) bundle.get("pdus");
if (pdus.length == 0) {
return;
}
// large message might be broken into many
SmsMessage[] messages = new SmsMessage[pdus.length];
StringBuilder sb = new StringBuilder();
for (int i = 0; i < pdus.length; i++) {
messages[i] = SmsMessage.createFromPdu((byte[]) pdus[i]);
sb.append(messages[i].getMessageBody());
}
String sender = messages[0].getOriginatingAddress();
String message = sb.toString();
Toast.makeText(context, message, Toast.LENGTH_SHORT).show();
// prevent any other broadcast receivers from receiving broadcast
// abortBroadcast();
}
}
}
}
<强>的Manifest.xml 强>
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.smsreceiver"
android:versionCode="1"
android:versionName="1.0">
<uses-sdk android:minSdkVersion="4" />
<uses-permission android:name="android.permission.RECEIVE_SMS" />
<application android:icon="@drawable/icon" android:label="@string/app_name">
<activity
// your activity
</activity>
<receiver android:name="com.example.smsreceiver.SmsReceiver" android:enabled="true">
<intent-filter android:priority="2147483647">
<action android:name="android.provider.Telephony.SMS_RECEIVED" />
</intent-filter>
</receiver>
</application>
</manifest>
确保使用自己定义的包。这里定义的包是虚拟的。