Android - ฟังข้อความ SMS ที่เข้ามา


155

ฉันพยายามสร้างแอพพลิเคชั่นสำหรับตรวจสอบข้อความ SMS ที่เข้ามาและเปิดโปรแกรมผ่านทาง SMS ขาเข้าและควรอ่านเนื้อหาจาก SMS

ขั้นตอนการทำงาน:

  • ส่ง SMS ไปยังอุปกรณ์ Android
  • แอพลิเคชันปฏิบัติการด้วยตนเอง
  • อ่านข้อมูล SMS

1
ฉันรู้ที่จะสร้างแอพเพื่อส่ง SMS แต่ที่นี่ฉันต้องสร้างแอพ SMS ที่รับข้อมูลจาก SMS และบันทึกลงในฐานข้อมูล SQLite ..... ฉันจะพัฒนาแอพดังกล่าวได้อย่างไร
iShader

@iShader ฉันหวังว่าคุณจะประสบความสำเร็จในการสร้างแอพแค่อยากรู้ว่าคุณจัดการซิงค์ msgs b / w กับอุปกรณ์และเซิร์ฟเวอร์อย่างไร
John x

คำตอบ:


265
public class SmsListener extends BroadcastReceiver{

    private SharedPreferences preferences;

    @Override
    public void onReceive(Context context, Intent intent) {
        // TODO Auto-generated method stub

        if(intent.getAction().equals("android.provider.Telephony.SMS_RECEIVED")){
            Bundle bundle = intent.getExtras();           //---get the SMS message passed in---
            SmsMessage[] msgs = null;
            String msg_from;
            if (bundle != null){
                //---retrieve the SMS message received---
                try{
                    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]);
                        msg_from = msgs[i].getOriginatingAddress();
                        String msgBody = msgs[i].getMessageBody();
                    }
                }catch(Exception e){
//                            Log.d("Exception caught",e.getMessage());
                }
            }
        }
    }
}

หมายเหตุ: ในไฟล์รายการเพิ่ม BroadcastReceiver-

<receiver android:name=".listener.SmsListener">
    <intent-filter>
        <action android:name="android.provider.Telephony.SMS_RECEIVED" />
    </intent-filter>
</receiver>

เพิ่มการอนุญาตนี้:

<uses-permission android:name="android.permission.RECEIVE_SMS" />

2
คุณช่วยอธิบายฉันหน่อยได้ไหมว่าทำไมคุณถึงใช้เครื่องรับสัญญาณรอง?
WindRider

2
@VineetShukla คุณช่วยอธิบายหน่อยได้ไหมว่า pdus คืออะไร ??
TheGrmasterGuy

11
ใช้ Intents.SMS_RECEIVED_ACTION แทนที่จะใช้รหัสฮาร์ด
Ahmad Kayyali

6
ความคิดเห็นข้างต้นไม่ถูกต้อง แอปใด ๆ ที่ยังคงสามารถรับการSMS_RECEIVEDถ่ายทอดได้ใน 4.4+ และตอนนี้การออกอากาศนั้นไม่สามารถยกเลิกได้มันมีความแน่นอนมากกว่าเวอร์ชั่นก่อนหน้า
Mike M.

3
@RuchirBaronia ข้อความหลายส่วน ข้อความ SMS เดียวมีขีด จำกัด อักขระ (ขึ้นอยู่กับชุดอักขระที่คุณใช้ แต่ขีด จำกัด ทั่วไปคือ 70, 140, 160 อักขระ) หากข้อความเกินขีด จำกัด นั้นสามารถแบ่งออกเป็นหลายข้อความส่วน อาเรย์นั้นคืออาเรย์ของส่วนต่าง ๆ ที่คุณต้องต่อกันเพื่อให้ได้ข้อความที่สมบูรณ์ ผู้รับของคุณจะได้รับข้อความที่สมบูรณ์เพียงครั้งเดียวเท่านั้นในแต่ละครั้ง มันอาจมีหลายส่วน
Mike M.

65

โปรดทราบว่าในบางอุปกรณ์รหัสของคุณจะไม่ทำงานโดยไม่ใช้Android: priority = "1000"ในการกรองแบบเจตนา:

<receiver android:name=".listener.SmsListener">
    <intent-filter android:priority="1000">
        <action android:name="android.provider.Telephony.SMS_RECEIVED" />
    </intent-filter>
</receiver>

และนี่คือการเพิ่มประสิทธิภาพบางส่วน:

public class SmsListener extends BroadcastReceiver{

    @Override
    public void onReceive(Context context, Intent intent) {
        if (Telephony.Sms.Intents.SMS_RECEIVED_ACTION.equals(intent.getAction())) {
            for (SmsMessage smsMessage : Telephony.Sms.Intents.getMessagesFromIntent(intent)) {
                String messageBody = smsMessage.getMessageBody();
            }
        }
    }
}

หมายเหตุ :
ค่าต้องเป็นจำนวนเต็มเช่น "100" ตัวเลขที่สูงกว่ามีลำดับความสำคัญสูงกว่า ค่าเริ่มต้นคือ 0 ค่าต้องมากกว่า -1000 และน้อยกว่า 1,000

นี่คือลิงค์


30
คำตอบนี้อาจจะดูหรูหรามากขึ้น แต่ต้องใช้ API 19 เพียงแค่เป็น FYI สำหรับคนอื่น ๆ
baekacaek

10
ตามนี้ , android:priorityไม่สามารถจะสูงกว่า1000(หรือน้อยกว่า-1000)
craned

2
มันไม่ทำงานบน Xiaomi Redmi Note 3 Pro กับ Android 5.1 ทุกคนกำลังให้บริการโซลูชันนี้ แต่ดูเหมือนจะไม่ได้ผลสำหรับฉัน
Sermilion

แทรกมาร์กอัป <receiver ... ไว้ในไฟล์รายการอยู่ที่ไหน
John Ward

3
@Sermilion คุณต้องอนุญาตการอ่าน SMS ด้วยตัวจัดการแอปพลิเคชั่นมือถือด้วยตนเอง
Sanjay Kushwah

6

@Mike M. และฉันพบปัญหากับคำตอบที่ยอมรับ (ดูความคิดเห็นของเรา):

โดยทั่วไปไม่มีจุดที่จะผ่านสำหรับลูปถ้าเราไม่ได้เชื่อมข้อความหลายส่วนในแต่ละครั้ง:

for (int i = 0; i < msgs.length; i++) {
    msgs[i] = SmsMessage.createFromPdu((byte[])pdus[i]);
    msg_from = msgs[i].getOriginatingAddress();
    String msgBody = msgs[i].getMessageBody();
}

โปรดสังเกตว่าเราเพิ่งตั้งค่าmsgBodyสตริงของส่วนที่เกี่ยวข้องของข้อความไม่ว่าเราจะอยู่ในดัชนีใดซึ่งทำให้จุดทั้งหมดของการวนลูปผ่านส่วนต่าง ๆ ของข้อความ SMS นั้นไร้ประโยชน์เนื่องจากมันจะถูกตั้งค่าเป็นอย่างมาก ค่าดัชนีสุดท้าย แต่เราควรใช้+=หรือตามที่ไมค์สังเกตStringBuilder:

ทั้งหมดนี้คือรหัสรับ SMS ของฉันที่มีลักษณะ:

if (myBundle != null) {
    Object[] pdus = (Object[]) myBundle.get("pdus"); // pdus is key for SMS in bundle

    //Object [] pdus now contains array of bytes
    messages = new SmsMessage[pdus.length];
    for (int i = 0; i < messages.length; i++) {
         messages[i] = SmsMessage.createFromPdu((byte[]) pdus[i]); //Returns one message, in array because multipart message due to sms max char
         Message += messages[i].getMessageBody(); // Using +=, because need to add multipart from before also
    }

    contactNumber = messages[0].getOriginatingAddress(); //This could also be inside the loop, but there is no need
}

เพียงวางคำตอบนี้ไว้ในกรณีที่คนอื่นมีความสับสนเหมือนกัน


4

นี่คือสิ่งที่ฉันใช้!

public class SMSListener extends BroadcastReceiver {

    // Get the object of SmsManager
    final SmsManager sms = SmsManager.getDefault();
String mobile,body;

    public void onReceive(Context context, Intent intent) {

        // Retrieves a map of extended data from the intent.
        final Bundle bundle = intent.getExtras();

        try {

            if (bundle != null) {

                final Object[] pdusObj = (Object[]) bundle.get("pdus");

                for (int i = 0; i < pdusObj.length; i++) {

                    SmsMessage currentMessage = SmsMessage.createFromPdu((byte[]) pdusObj[i]);
                    String phoneNumber = currentMessage.getDisplayOriginatingAddress();

                    String senderNum = phoneNumber;
                    String message = currentMessage.getDisplayMessageBody();
                     mobile=senderNum.replaceAll("\\s","");
                     body=message.replaceAll("\\s","+");


                    Log.i("SmsReceiver", "senderNum: "+ senderNum + "; message: " + body);


                    // Show Alert
                    int duration = Toast.LENGTH_LONG;
                    Toast toast = Toast.makeText(context,
                            "senderNum: "+ mobile+ ", message: " + message, duration);
                    toast.show();

                } // end for loop
            } // bundle is null

        } catch (Exception e) {
            Log.e("SmsReceiver", "Exception smsReceiver" +e);

        }
    }
}

2

ในกรณีที่คุณต้องการจัดการกับกิจกรรมที่เปิดคุณสามารถใช้ PendintIntent (ทำตามขั้นตอนด้านล่าง):

public class SMSReciver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        final Bundle bundle = intent.getExtras();
        try {
            if (bundle != null) {
                final Object[] pdusObj = (Object[]) bundle.get("pdus");
                for (int i = 0; i < pdusObj.length; i++) {
                    SmsMessage currentMessage = SmsMessage.createFromPdu((byte[]) pdusObj[i]);
                    String phoneNumber = currentMessage.getDisplayOriginatingAddress();
                    String senderNum = phoneNumber;
                    String message = currentMessage.getDisplayMessageBody();
                    try {
                        if (senderNum.contains("MOB_NUMBER")) {
                            Toast.makeText(context,"",Toast.LENGTH_SHORT).show();

                            Intent intentCall = new Intent(context, MainActivity.class);
                            intentCall.putExtra("message", currentMessage.getMessageBody());

                            PendingIntent pendingIntent= PendingIntent.getActivity(context, 0, intentCall, PendingIntent.FLAG_UPDATE_CURRENT);
                            pendingIntent.send();
                        }
                    } catch (Exception e) {
                    }
                }
            }
        } catch (Exception e) {
        }
    }
} 

ประจักษ์:

<activity android:name=".MainActivity"
            android:launchMode="singleTask"/>
<receiver android:name=".SMSReciver">
            <intent-filter android:priority="1000">
                <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
            </intent-filter>
        </receiver>

onNewIntent:

 @Override
         protected void onNewIntent(Intent intent) {
                super.onNewIntent(intent);
                Toast.makeText(this, "onNewIntent", Toast.LENGTH_SHORT).show();

                onSMSReceived(intent.getStringExtra("message"));

            }

สิทธิ์:

<uses-permission android:name="android.permission.RECEIVE_SMS" />
<uses-permission android:name="android.permission.READ_SMS" />
<uses-permission android:name="android.permission.SEND_SMS" />

ผู้ดูแลระบบ Google ของ Google Play Store พิจารณาสิทธิ์ RECEIVE_SMS (ในบทช่วยสอนที่คุณพูดถึง) ว่าเป็นอันตราย ดังนั้นแอปที่มีสิทธิ์จะถูกปฏิเสธ จากนั้นผู้พัฒนาจะต้องส่งแบบฟอร์มไปยังผู้ดูแลระบบ Google Play เพื่อขออนุมัติ ผู้พัฒนารายอื่นได้กล่าวถึงกระบวนการนี้ว่าแย่มากเมื่อได้รับผลตอบรับเป็นเวลาหลายสัปดาห์และได้รับคำปฏิเสธทันทีโดยไม่มีคำอธิบายหรือข้อเสนอแนะทั่วไป ความคิดใด ๆ เกี่ยวกับวิธีการหลีกเลี่ยง?
AJW

2

หากมีคนอ้างถึงวิธีการทำคุณสมบัติเดียวกัน (อ่าน OTP โดยใช้ SMS ที่ได้รับ) บน Xamarin Android เช่นฉัน:

  1. เพิ่มรหัสนี้ในไฟล์ AndroidManifest.xml ของคุณ:

    <receiver android:name=".listener.BroadcastReveiverOTP">
    <intent-filter>
        <action android:name="android.provider.Telephony.SMS_RECEIVED" />
    </intent-filter>
    </receiver>
    <uses-permission android:name="android.permission.RECEIVE_SMS" />
    <uses-permission android:name="android.permission.BROADCAST_SMS" />
    <uses-permission android:name="android.permission.READ_SMS" />
  2. จากนั้นสร้างคลาส BroadcastReveiver ในโครงการ Android ของคุณ

    [BroadcastReceiver(Enabled = true)] [IntentFilter(new[] { "android.provider.Telephony.SMS_RECEIVED" }, Priority = (int)IntentFilterPriority.HighPriority)] 
    public class BroadcastReveiverOTP : BroadcastReceiver {
            public static readonly string INTENT_ACTION = "android.provider.Telephony.SMS_RECEIVED";
    
            protected string message, address = string.Empty;
    
            public override void OnReceive(Context context, Intent intent)
            {
                if (intent.HasExtra("pdus"))
                {
                    var smsArray = (Java.Lang.Object[])intent.Extras.Get("pdus");
                    foreach (var item in smsArray)
                    {
                        var sms = SmsMessage.CreateFromPdu((byte[])item);
                        address = sms.OriginatingAddress;
                        if (address.Equals("NotifyDEMO"))
                        {
                            message = sms.MessageBody;
                            string[] pin = message.Split(' ');
                            if (!string.IsNullOrWhiteSpace(pin[0]))
                            { 
                                    // NOTE : Here I'm passing received OTP to Portable Project using MessagingCenter. So I can display the OTP in the relevant entry field.
                                    MessagingCenter.Send<object, string>(this,MessengerKeys.OnBroadcastReceived, pin[0]);
                            }
                            }
                    }
                }
            }
    }
  3. ลงทะเบียนคลาส BroadcastReceiver ในคลาส MainActivity ของคุณในโครงการ Android:

    public class MainActivity : global::Xamarin.Forms.Platform.Android.FormsAppCompatActivity {
    
            // Initialize your class
            private BroadcastReveiverOTP _receiver = new BroadcastReveiverOTP ();
    
            protected override void OnCreate(Bundle bundle) { 
                    base.OnCreate(bundle);
    
                    global::Xamarin.Forms.Forms.Init(this, bundle);
                    LoadApplication(new App());
    
                    // Register your receiver :  RegisterReceiver(_receiver, new IntentFilter("android.provider.Telephony.SMS_RECEIVED"));
    
            }
    }

มีข้อผิดพลาดของคอมไพเลอร์ว่า "android.permission.BROADCAST_SMS" อนุญาตเฉพาะกับแอประบบ
committedandroider

2

ขอบคุณ @Vineet Shukla (คำตอบที่ยอมรับ) และ @Ruchir Baronia (พบปัญหาในคำตอบที่ยอมรับ) ด้านล่างนี้เป็นKotlinเวอร์ชั่น:

เพิ่มการอนุญาต:

<uses-permission android:name="android.permission.RECEIVE_SMS" />

ลงทะเบียน BroadcastReceiver ใน AndroidManifest:

<receiver
    android:name=".receiver.SmsReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter android:priority="2332412">
        <action android:name="android.provider.Telephony.SMS_RECEIVED" />
    </intent-filter>
</receiver>

เพิ่มการใช้งานสำหรับ BroadcastReceiver:

class SmsReceiver : BroadcastReceiver() {
    private var mLastTimeReceived = System.currentTimeMillis()

    override fun onReceive(p0: Context?, intent: Intent?) {
        val currentTimeMillis = System.currentTimeMillis()
        if (currentTimeMillis - mLastTimeReceived > 200) {
            mLastTimeReceived = currentTimeMillis

            val pdus: Array<*>
            val msgs: Array<SmsMessage?>
            var msgFrom: String?
            var msgText: String?
            val strBuilder = StringBuilder()
            intent?.extras?.let {
                try {
                    pdus = it.get("pdus") as Array<*>
                    msgs = arrayOfNulls(pdus.size)
                    for (i in msgs.indices) {
                        msgs[i] = SmsMessage.createFromPdu(pdus[i] as ByteArray)
                        strBuilder.append(msgs[i]?.messageBody)
                    }

                    msgText = strBuilder.toString()
                    msgFrom = msgs[0]?.originatingAddress

                    if (!msgFrom.isNullOrBlank() && !msgText.isNullOrBlank()) {
                        //
                        // Do some thing here
                        //
                    }
                } catch (e: Exception) {
                }
            }
        }
    }
}

เหตุการณ์บางครั้งไฟไหม้สองครั้งดังนั้นฉันจึงเพิ่ม mLastTimeReceived = System.currentTimeMillis()


1

การใช้งานการออกอากาศบน Kotlin:

 private class SmsListener : BroadcastReceiver() {
    override fun onReceive(context: Context?, intent: Intent?) {
        Log.d(TAG, "SMS Received!")

        val txt = getTextFromSms(intent?.extras)
        Log.d(TAG, "message=" + txt)
    }

    private fun getTextFromSms(extras: Bundle?): String {
        val pdus = extras?.get("pdus") as Array<*>
        val format = extras.getString("format")
        var txt = ""
        for (pdu in pdus) {
            val smsmsg = getSmsMsg(pdu as ByteArray?, format)
            val submsg = smsmsg?.displayMessageBody
            submsg?.let { txt = "$txt$it" }
        }
        return txt
    }

    private fun getSmsMsg(pdu: ByteArray?, format: String?): SmsMessage? {
        return when {
            SDK_INT >= Build.VERSION_CODES.M -> SmsMessage.createFromPdu(pdu, format)
            else -> SmsMessage.createFromPdu(pdu)
        }
    }

    companion object {
        private val TAG = SmsListener::class.java.simpleName
    }
}

หมายเหตุ: ในไฟล์รายการเพิ่ม BroadcastReceiver-

<receiver android:name=".listener.SmsListener">
    <intent-filter>
        <action android:name="android.provider.Telephony.SMS_RECEIVED" />
    </intent-filter>
</receiver>

เพิ่มการอนุญาตนี้:

<uses-permission android:name="android.permission.RECEIVE_SMS" />

1

คำตอบที่ได้รับการยอมรับนั้นถูกต้องและทำงานบน Android รุ่นเก่ากว่าที่ Android OS ขออนุญาตเมื่อติดตั้งแอปอย่างไรก็ตาม Android เวอร์ชันใหม่จะไม่ทำงานทันทีเพราะ Android OS รุ่นใหม่ขอสิทธิ์ระหว่างรันไทม์เมื่อแอพต้องการคุณสมบัตินั้น . ดังนั้นเพื่อที่จะรับ SMS บน Android รุ่นใหม่โดยใช้เทคนิคที่กล่าวถึงในคำตอบที่ได้รับการยอมรับโปรแกรมเมอร์จะต้องใช้รหัสที่จะตรวจสอบและขอสิทธิ์จากผู้ใช้ในระหว่างรันไทม์ ในกรณีนี้สิทธิ์การตรวจสอบการทำงาน / รหัสสามารถนำไปใช้ใน onCreate () ของกิจกรรมแรกของแอป เพียงคัดลอกและวางสองวิธีต่อไปนี้ในกิจกรรมแรกของคุณและเรียกใช้วิธีการ checkForSmsReceivePermissions () ที่ส่วนท้ายของ onCreate ()

    void checkForSmsReceivePermissions(){
    // Check if App already has permissions for receiving SMS
    if(ContextCompat.checkSelfPermission(getBaseContext(), "android.permission.RECEIVE_SMS") == PackageManager.PERMISSION_GRANTED) {
        // App has permissions to listen incoming SMS messages
        Log.d("adnan", "checkForSmsReceivePermissions: Allowed");
    } else {
        // App don't have permissions to listen incoming SMS messages
        Log.d("adnan", "checkForSmsReceivePermissions: Denied");

        // Request permissions from user 
        ActivityCompat.requestPermissions(this, new String[] {Manifest.permission.RECEIVE_SMS}, 43391);
    }
}

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    if(requestCode == 43391){
        if(grantResults.length>0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){
            Log.d("adnan", "Sms Receive Permissions granted");
        } else {
            Log.d("adnan", "Sms Receive Permissions denied");
        }
    }
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.