在Android应用程序上接收短信

13

我遵循了一个教程,在我的应用程序上接收短信并将其读取以将SMS正文传递给Toast。

这就是Receiver类。

public class SmsReciever extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent){
        Bundle bundle= intent.getExtras();
        SmsMessage[] msgs= null;
        String str="";
        if(bundle != null ){
            Object [] pdus=(Object[]) bundle.get("pdus");
            msgs=new SmsMessage[pdus.length];
            for(int i=0; i< msgs.length;i++){
                msgs[i]= SmsMessage.createFromPdu((byte[])pdus[i]);
                str+= msgs[i].getMessageBody();

            }
             Toast.makeText(context, str, Toast.LENGTH_LONG).show();
        }
        }

    }

清单文件

    <receiver android:name="com.msoft.masrooq.SmsReciever">
    <intent-filter>
    <action android:name="android.provider.telephony.SMS_RECIEVED"></action>
    </intent-filter>
    </receiver>
        <uses-permission android:name="android.permission.RECEIVE_SMS"/>
            <uses-permission android:name="android.permission.READ_SMS" />

应用程序可以正常启动,但无法响应接收到的短信,什么也不做。


我曾经遇到过类似的问题。虽然我认为我已经实现了所有必要的功能,但我并没有收到任何消息。我不知道这是否是解决某些问题的方法,但我在清单文件中的主活动标签上方声明了接收器,从那时起它就可以正常工作了。希望这能解决一些人的问题。 - zulu_papa
3个回答

33

这是我接收短信消息的实现方式。短信消息可能会被分为多条,注意一下如何处理它们。也请检查android:priority属性。

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();
            }
        }
    }
}

AndroidManifest.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
            android:name=".SmsLoggerActivity"
            android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </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>

注意事项: 如果你在XML中声明了接收器,则系统无论你的应用程序是否启动都可以使用你的接收器。 自Android 1.6版本以来,关于已接收短信消息的通知被作为有序广播传送,你可以使用<intent-filter>android:priority属性告诉系统将短信首先发送给你的应用程序(你也可以调用abortBroadcast()方法,这样其他应用程序就不会接收该短信,如原生短信应用)。不要忘记广播接收器有大约10秒的时间来执行其操作,否则在完成其工作之前可能会被过早地终止。


提醒一下,android.permission.SEND_SMS和android.permission.RECEIVE_SMS是不同的权限。在你的主活动中只需创建一个SmsReceiver()对象即可。 - Sydwell
如何在 MMS 中实现这个呢? - Arya
@biegleux 谷歌Play商店的管理员认为教程中提到的RECEIVE_SMS权限是危险的。因此,包含该权限的应用将被拒绝。然后开发者必须向谷歌Play管理员提交表格以获得批准。其他开发者已经提到这个过程很糟糕,反馈需要数周时间,并且会收到没有解释或通用反馈的明确拒绝。有什么避免的方法吗? - AJW

2

注意:在一些设备上,如果你的代码没有在intent filter中加入android:priority="100",它可能无法正常工作。

 <application
 <uses-permission android:name="android.permission.RECEIVE_SMS" />
 <uses-permission android:name="android.permission.READ_SMS" />
 ...
  <receiver
        android:name=".SMSReceiver">
        <intent-filter android:priority="100">
            <action android:name="android.provider.Telephony.SMS_RECEIVED" />
        </intent-filter>
  </receiver>
 ...
</application>

以下是Java代码:

public class SMSReceiver extends BroadcastReceiver {
public static final String ACTION ="android.provider.Telephony.SMS_RECEIVED";
private static final String SMS_SENDER="123456789";

@Override
public void onReceive(Context context, Intent intent) {
        if (intent != null && intent.getAction() != null &&
                ACTION.compareToIgnoreCase(intent.getAction()) == 0) {
            Object[] pduArray = (Object[]) intent.getExtras().get("pdus");
            SmsMessage[] messages = new SmsMessage[pduArray.length];
            for (int i = 0; i < pduArray.length; i++) {
                messages[i] = SmsMessage.createFromPdu((byte[]) pduArray[i]);
            }
            // SMS Sender, example: 123456789
            String sms_from = messages[0].getDisplayOriginatingAddress(); 

            //Lets check if SMS sender is 123456789
            if (sms_from.equalsIgnoreCase(SMS_SENDER)) {
                StringBuilder bodyText = new StringBuilder();

                // If SMS has several parts, lets combine it :)
                for (int i = 0; i < messages.length; i++) {
                    bodyText.append(messages[i].getMessageBody());
                }
                //SMS Body
                String body = bodyText.toString();
                // Lets get SMS Code
                String code  = body.replaceAll("[^0-9]", ""); 
            }
      }
 }

2
广播是区分大小写的。请使用 android.provider.Telephony.SMS_RECEIVED 而不是 android.provider.telephony.SMS_RECEIVED
此外,我也有一个类别设置,但我不确定它是否强制执行。
<intent-filter>
   <action android:name="android.provider.Telephony.SMS_RECEIVED" />
   <category android:name="android.intent.category.DEFAULT" />
</intent-filter>

Google Play商店的管理员认为教程中提到的RECEIVE_SMS权限是危险的。因此,包含该权限的应用将被拒绝。然后开发者必须向Google Play管理员提交表格以获得批准。其他开发者已经提到这个过程很糟糕,反馈需要数周时间,并且会收到没有解释或通用反馈的明确拒绝。有什么避免的想法吗? - AJW

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接