Lên lịch cho chuông báo

Chuông báo (dựa trên lớp AlarmManager) cung cấp cho bạn cách thực hiện các thao tác theo thời gian bên ngoài thời gian hoạt động của ứng dụng. Ví dụ: bạn có thể sử dụng chuông báo để bắt đầu một hoạt động kéo dài, chẳng hạn như khởi động dịch vụ mỗi ngày một lần để tải thông tin dự báo thời tiết xuống.

Báo thức có các đặc điểm sau:

  • Chúng cho phép bạn kích hoạt Ý định vào những thời điểm và/hoặc khoảng thời gian đã đặt.

  • Bạn có thể sử dụng các đối tượng này cùng với broadcast receiver để lên lịch cho công việc hoặc WorkRequest nhằm thực hiện các thao tác khác.

  • Các API này hoạt động bên ngoài ứng dụng, vì vậy, bạn có thể sử dụng chúng để kích hoạt các sự kiện hoặc hành động ngay cả khi ứng dụng không chạy, và ngay cả khi thiết bị tự ở chế độ ngủ.

  • Các thành phần này giúp bạn giảm thiểu các yêu cầu về tài nguyên của ứng dụng. Bạn có thể lên lịch cho các hoạt động mà không cần dựa vào đồng hồ hẹn giờ hoặc chạy dịch vụ liên tục.

Đặt chuông báo không chính xác

Khi một ứng dụng đặt chuông báo không chính xác, hệ thống sẽ đưa ra chuông báo vào một thời điểm nào đó trong tương lai. Chuông báo không chính xác đưa ra một số đảm bảo về thời gian gửi chuông báo trong khi vẫn tuân thủ các hạn chế về tiết kiệm pin như Nghỉ ngơi.

Nhà phát triển có thể tận dụng các đảm bảo API sau đây để tuỳ chỉnh thời gian gửi thông báo không chính xác.

Gửi chuông báo sau một thời điểm cụ thể

Nếu ứng dụng của bạn gọi set(), setInexactRepeating() hoặc setAndAllowWhileIdle(), chuông báo sẽ không bao giờ đổ trước thời gian kích hoạt đã cung cấp.

Trên Android 12 (API cấp 31) trở lên, hệ thống gọi chuông báo trong vòng một giờ sau thời gian kích hoạt được cung cấp, trừ phi có bất kỳ hạn chế nào về tiết kiệm pin, chẳng hạn như trình tiết kiệm pin hoặc Chế độ nghỉ.

Đưa ra chuông báo trong một khoảng thời gian

Nếu ứng dụng của bạn gọi setWindow(), thì chuông báo sẽ không bao giờ tắt trước thời gian kích hoạt đã cung cấp. Trừ phi có bất kỳ hạn chế tiết kiệm pin nào có hiệu lực, chuông báo sẽ được gửi trong khoảng thời gian đã chỉ định, bắt đầu từ thời gian kích hoạt nhất định.

Nếu ứng dụng của bạn nhắm đến Android 12 trở lên, thì hệ thống có thể trì hoãn ít nhất 10 phút gọi chuông báo không chính xác theo khung thời gian. Vì lý do này, các giá trị tham số windowLengthMillis trong 600000 được cắt thành 600000.

Báo thức lặp lại gần như đều đặn

Nếu ứng dụng của bạn gọi setInexactRepeating(), hệ thống sẽ gọi nhiều chuông báo:

  1. Chuông báo đầu tiên sẽ kêu trong khoảng thời gian được chỉ định, bắt đầu từ thời gian kích hoạt nhất định.
  2. Các chuông báo tiếp theo thường đổ chuông sau khi hết khoảng thời gian đã chỉ định. Thời gian giữa hai lần gọi báo thức liên tiếp có thể khác nhau.

Đặt chuông báo chính xác

Hệ thống sẽ gọi thông báo chính xác tại một thời điểm chính xác trong tương lai.

Hầu hết các ứng dụng đều có thể lên lịch cho các tác vụ và sự kiện bằng cách sử dụng thông báo không chính xác để hoàn tất một số trường hợp sử dụng phổ biến. Nếu chức năng cốt lõi của ứng dụng phụ thuộc vào chuông báo được hẹn giờ chính xác (chẳng hạn như ứng dụng đồng hồ báo thức hoặc ứng dụng lịch), thì bạn có thể sử dụng chuông báo chính xác.

Các trường hợp sử dụng có thể không cần đến chuông báo chính xác

Danh sách sau đây cho thấy những quy trình công việc thường gặp có thể không cần đến chuông báo chính xác:

Lên lịch cho các hoạt động định thời gian trong suốt vòng đời của ứng dụng
Lớp Handler bao gồm một số phương thức hiệu quả để xử lý các thao tác định thời gian, chẳng hạn như thực hiện một số thao tác mỗi n giây trong khi ứng dụng đang hoạt động: postAtTime()postDelayed(). Xin lưu ý rằng các API này dựa trên thời gian hoạt động của hệ thống chứ không phải thời gian thực.
Thao tác ở chế độ nền đã lên lịch, chẳng hạn như cập nhật ứng dụng và tải nhật ký lên
WorkManager cung cấp cách lập lịch công việc định kỳ có giới hạn thời gian. Bạn có thể cung cấp một khoảng thời gian lặp lại và flexInterval (tối thiểu 15 phút) để xác định thời gian chạy chi tiết cho công việc.
Hành động do người dùng chỉ định sẽ xảy ra sau một thời gian cụ thể (ngay cả khi hệ thống ở trạng thái rảnh)
Sử dụng chuông báo không chính xác. Cụ thể, hãy gọi setAndAllowWhileIdle().
Hành động do người dùng chỉ định sẽ xảy ra sau một thời gian cụ thể
Sử dụng chuông báo không chính xác. Cụ thể, hãy gọi set().
Hành động do người dùng chỉ định có thể xảy ra trong một khoảng thời gian chỉ định
Sử dụng chuông báo không chính xác. Cụ thể, hãy gọi setWindow(). Xin lưu ý rằng nếu ứng dụng của bạn nhắm đến Android 12 trở lên, thì thời lượng nhỏ nhất được phép là 10 phút.

Cách đặt chuông báo chính xác

Ứng dụng của bạn có thể đặt chuông báo chính xác bằng một trong các phương thức sau. Các phương thức này được sắp xếp sao cho các phương thức ở gần cuối danh sách phục vụ nhiều tác vụ quan trọng về thời gian hơn nhưng đòi hỏi nhiều tài nguyên hệ thống hơn.

setExact()

Gọi chuông báo vào thời điểm gần như chính xác trong tương lai, miễn là các biện pháp tiết kiệm pin khác không có hiệu lực.

Hãy sử dụng phương thức này để đặt chuông báo chính xác, trừ phi công việc của ứng dụng cần thiết về thời gian đối với người dùng.

setExactAndAllowWhileIdle()

Gọi chuông báo vào thời điểm gần như chính xác trong tương lai, ngay cả khi các biện pháp tiết kiệm pin đang được áp dụng.

setAlarmClock()

Gọi chuông báo vào một thời điểm chính xác trong tương lai. Vì người dùng có thể thấy rõ những chuông báo này, nên hệ thống không bao giờ điều chỉnh thời gian gửi. Hệ thống xác định đây là những chuông báo quan trọng nhất và để lại chế độ tiết kiệm pin nếu cần để gửi chuông báo.

Mức tiêu thụ tài nguyên hệ thống

Khi hệ thống kích hoạt chuông báo chính xác mà ứng dụng của bạn đặt, thiết bị sẽ tiêu tốn rất nhiều tài nguyên, chẳng hạn như thời lượng pin, đặc biệt là khi đang ở chế độ tiết kiệm pin. Hơn nữa, hệ thống không thể dễ dàng phân lô các yêu cầu này để sử dụng tài nguyên hiệu quả hơn.

Bạn nên tạo thông báo không chính xác bất cứ khi nào có thể. Để thực hiện thao tác lâu hơn, hãy lên lịch bằng WorkManager hoặc JobScheduler từ BroadcastReceiver của chuông báo. Để thực hiện công việc trong khi thiết bị ở chế độ Nghỉ, hãy tạo chuông báo không chính xác bằng cách sử dụng setAndAllowWhileIdle() và bắt đầu một công việc qua chuông báo.

Khai báo quyền thích hợp về chuông báo chính xác

Nếu ứng dụng của bạn nhắm đến Android 12 trở lên, bạn phải có quyền truy cập đặc biệt cho ứng dụng "Chuông báo và lời nhắc". Để thực hiện việc này, hãy khai báo quyền SCHEDULE_EXACT_ALARM trong tệp kê khai của ứng dụng, như trong đoạn mã sau:

<manifest ...>
    <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/>
    <application ...>
        ...
    </application>
</manifest>

Nếu ứng dụng của bạn nhắm đến Android 13 (API cấp 33) trở lên, thì bạn có thể khai báo quyền SCHEDULE_EXACT_ALARM hoặc USE_EXACT_ALARM.

<manifest ...>
    <uses-permission android:name="android.permission.USE_EXACT_ALARM"/>
    <application ...>
        ...
    </application>
</manifest>

Mặc dù cả quyền SCHEDULE_EXACT_ALARMUSE_EXACT_ALARM đều báo hiệu những khả năng như nhau, nhưng chúng được cấp theo cách khác nhau và hỗ trợ các trường hợp sử dụng khác nhau. Ứng dụng của bạn nên sử dụng chuông báo chính xác và khai báo quyền SCHEDULE_EXACT_ALARM hoặc USE_EXACT_ALARM chỉ khi một hàm dành cho người dùng trong ứng dụng yêu cầu hành động được xác định thời gian chính xác.

USE_EXACT_ALARM

SCHEDULE_EXACT_ALARM

  • Do người dùng cấp
  • Nhiều trường hợp sử dụng hơn
  • Ứng dụng phải xác nhận rằng quyền này chưa bị thu hồi

Quyền SCHEDULE_EXACT_ALARM không được cấp trước cho các lượt cài đặt mới của ứng dụng nhắm đến Android 13 (API cấp 33) trở lên. Nếu người dùng chuyển dữ liệu ứng dụng sang một thiết bị chạy Android 14 thông qua tác vụ sao lưu và khôi phục, thì quyền SCHEDULE_EXACT_ALARM sẽ bị từ chối trên thiết bị mới. Tuy nhiên, nếu một ứng dụng hiện có đã có quyền này, thì ứng dụng đó sẽ được cấp quyền trước khi thiết bị nâng cấp lên Android 14.

Lưu ý: Nếu bạn đặt chuông báo chính xác bằng đối tượng OnAlarmListener, chẳng hạn như với API setExact, thì bạn không cần cấp quyền SCHEDULE_EXACT_ALARM.

Sử dụng quyền SCHEDULE_EXACT_ALARM

Không giống như USE_EXACT_ALARM, người dùng phải cấp quyền SCHEDULE_EXACT_ALARM. Cả người dùng và hệ thống đều có thể thu hồi quyền SCHEDULE_EXACT_ALARM.

Để kiểm tra xem quyền đã được cấp cho ứng dụng của bạn hay chưa, hãy gọi canScheduleExactAlarms() trước khi cố gắng đặt chuông báo chính xác. Khi quyền SCHEDULE_EXACT_ALARM bị thu hồi đối với ứng dụng của bạn, ứng dụng sẽ dừng và mọi chuông báo chính xác trong tương lai sẽ bị huỷ. Điều này cũng có nghĩa là giá trị do canScheduleExactAlarms() trả về vẫn hợp lệ trong toàn bộ vòng đời của ứng dụng.

Khi quyền SCHEDULE_EXACT_ALARMS được cấp cho ứng dụng của bạn, hệ thống sẽ gửi thông báo ACTION_SCHEDULE_EXACT_ALARM_PERMISSION_STATE_CHANGED cho ứng dụng đó. Ứng dụng của bạn nên triển khai một broadcast receiver có chức năng sau:

  1. Xác nhận rằng ứng dụng của bạn vẫn có quyền truy cập đặc biệt. Để thực hiện việc này, hãy gọi canScheduleExactAlarms(). Bước kiểm tra này bảo vệ ứng dụng của bạn khỏi trường hợp người dùng cấp quyền cho ứng dụng rồi thu hồi quyền gần như ngay sau đó.
  2. Lên lịch lại mọi chuông báo chính xác mà ứng dụng của bạn cần, dựa trên trạng thái hiện tại của chuông báo. Logic này sẽ tương tự như hành động của ứng dụng khi nhận được thông báo ACTION_BOOT_COMPLETED.

Yêu cầu người dùng cấp quyền cho SCHEDULE_EXACT_ALARM

Tuỳ chọn này có tên là &quot;Cho phép đặt chuông báo và lời nhắc&quot;
Hình 1. Trang truy cập đặc biệt cho ứng dụng "Chuông báo và lời nhắc" trong phần cài đặt hệ thống. Tại đây, người dùng có thể cho phép ứng dụng của bạn đặt chuông báo chính xác.

Nếu cần, bạn có thể đưa người dùng đến màn hình Chuông báo và lời nhắc trong phần cài đặt hệ thống, như trong Hình 1. Để thực hiện điều này, vui lòng hoàn thành các bước sau:

  1. Trong giao diện người dùng của ứng dụng, hãy giải thích cho người dùng lý do ứng dụng cần lên lịch chuông báo chính xác.
  2. Gọi một ý định bao gồm thao tác theo ý định ACTION_REQUEST_SCHEDULE_EXACT_ALARM.

Đặt báo thức lặp lại

Việc lặp lại chuông báo cho phép hệ thống thông báo cho ứng dụng của bạn theo lịch định kỳ.

Chuông báo có thiết kế không tốt có thể gây tiêu hao pin và làm máy chủ phải tải đáng kể. Vì lý do này, trên Android 4.4 (API cấp 19) trở lên, tất cả chuông báo lặp lại đều là chuông báo không chính xác.

Chuông báo lặp lại có các đặc điểm sau:

  • Một loại chuông báo. Để thảo luận thêm, hãy xem phần Chọn loại chuông báo.

  • Thời gian kích hoạt. Nếu thời gian kích hoạt mà bạn chỉ định là trong quá khứ, thì chuông báo sẽ kích hoạt ngay lập tức.

  • Khoảng thời gian báo thức. Ví dụ: 1 lần/ngày, mỗi giờ hoặc 5 phút một lần.

  • Một ý định đang chờ xử lý kích hoạt khi chuông báo được kích hoạt. Khi bạn đặt chuông báo thứ hai sử dụng cùng một ý định đang chờ xử lý, chuông báo đó sẽ thay thế chuông báo ban đầu.

Để huỷ PendingIntent(), hãy truyền FLAG_NO_CREATE đến PendingIntent.getService() để lấy một thực thể của ý định (nếu có), sau đó chuyển ý định đó đến AlarmManager.cancel()

Kotlin

val alarmManager =
    context.getSystemService(Context.ALARM_SERVICE) as? AlarmManager
val pendingIntent =
    PendingIntent.getService(context, requestId, intent,
                                PendingIntent.FLAG_NO_CREATE)
if (pendingIntent != null && alarmManager != null) {
  alarmManager.cancel(pendingIntent)
}

Java

AlarmManager alarmManager =
    (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
PendingIntent pendingIntent =
    PendingIntent.getService(context, requestId, intent,
                                PendingIntent.FLAG_NO_CREATE);
if (pendingIntent != null && alarmManager != null) {
  alarmManager.cancel(pendingIntent);
}

Chọn một loại chuông báo

Một trong những yếu tố đầu tiên cần cân nhắc khi sử dụng chuông báo lặp lại là loại của chuông báo.

Có 2 loại đồng hồ chung cho chuông báo: "đã trôi qua theo thời gian thực" và "đồng hồ thời gian thực" (RTC). Thời gian thực đã trôi qua sử dụng "thời gian kể từ khi khởi động hệ thống" làm tham chiếu, còn đồng hồ thời gian thực sử dụng thời gian UTC (đồng hồ tường). Điều này có nghĩa là thời gian thực đã trôi qua phù hợp để đặt chuông báo dựa trên khoảng thời gian đã trôi qua (ví dụ: chuông báo kích hoạt 30 giây một lần) vì không bị ảnh hưởng bởi múi giờ hoặc ngôn ngữ. Loại đồng hồ theo thời gian thực phù hợp hơn với các chuông báo phụ thuộc vào ngôn ngữ hiện tại.

Cả hai loại đều có phiên bản "đánh thức", cho biết sẽ đánh thức CPU của thiết bị nếu màn hình đang tắt. Điều này đảm bảo rằng chuông báo sẽ kích hoạt vào thời gian đã lên lịch. Điều này rất hữu ích nếu ứng dụng có phần phụ thuộc về thời gian. Ví dụ: nếu ứng dụng có cửa sổ giới hạn để thực hiện một thao tác cụ thể. Nếu bạn không sử dụng phiên bản đánh thức của loại chuông báo, thì tất cả chuông báo lặp lại sẽ kích hoạt khi thiết bị của bạn bật vào lần tiếp theo.

Nếu bạn chỉ cần chuông báo kích hoạt ở một khoảng thời gian cụ thể (ví dụ: nửa giờ một lần), hãy sử dụng một trong các loại thời gian thực đã trôi qua. Nhìn chung, đây là lựa chọn tốt hơn.

Nếu bạn cần chuông báo kích hoạt vào một thời điểm cụ thể trong ngày, hãy chọn một trong các loại đồng hồ thời gian thực dựa trên đồng hồ. Tuy nhiên, xin lưu ý rằng phương pháp này có thể có một số hạn chế. Ứng dụng có thể không dịch tốt sang các ngôn ngữ khác và nếu người dùng thay đổi chế độ cài đặt giờ của thiết bị thì hành vi không mong muốn trong ứng dụng có thể xảy ra. Việc sử dụng loại chuông báo theo đồng hồ theo thời gian thực cũng không điều chỉnh được theo tỷ lệ như đã thảo luận ở trên. Bạn nên sử dụng chuông báo "thời gian thực đã trôi qua" nếu có thể.

Dưới đây là danh sách các loại:

  • ELAPSED_REALTIME: Kích hoạt ý định đang chờ xử lý dựa trên khoảng thời gian kể từ khi thiết bị được khởi động nhưng không đánh thức thiết bị. Thời gian đã trôi qua bao gồm cả thời gian thiết bị ở chế độ ngủ.

  • ELAPSED_REALTIME_WAKEUP: Đánh thức thiết bị và kích hoạt ý định đang chờ xử lý sau khi đã hết khoảng thời gian được chỉ định kể từ khi khởi động thiết bị.

  • RTC: Kích hoạt ý định đang chờ xử lý tại thời điểm được chỉ định nhưng không đánh thức thiết bị.

  • RTC_WAKEUP: Đánh thức thiết bị để kích hoạt ý định đang chờ xử lý tại thời điểm đã chỉ định.

Ví dụ về chuông báo theo thời gian thực đã trôi qua

Dưới đây là một số ví dụ về cách sử dụng ELAPSED_REALTIME_WAKEUP

Đánh thức thiết bị để kích hoạt chuông báo sau 30 phút và sau đó 30 phút một lần:

Kotlin

// Hopefully your alarm will have a lower frequency than this!
alarmMgr?.setInexactRepeating(
        AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() + AlarmManager.INTERVAL_HALF_HOUR,
        AlarmManager.INTERVAL_HALF_HOUR,
        alarmIntent
)

Java

// Hopefully your alarm will have a lower frequency than this!
alarmMgr.setInexactRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() + AlarmManager.INTERVAL_HALF_HOUR,
        AlarmManager.INTERVAL_HALF_HOUR, alarmIntent);

Đánh thức thiết bị để kích hoạt chuông báo một lần (không lặp lại) sau một phút:

Kotlin

private var alarmMgr: AlarmManager? = null
private lateinit var alarmIntent: PendingIntent
...
alarmMgr = context.getSystemService(Context.ALARM_SERVICE) as AlarmManager
alarmIntent = Intent(context, AlarmReceiver::class.java).let { intent ->
    PendingIntent.getBroadcast(context, 0, intent, 0)
}

alarmMgr?.set(
        AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() + 60 * 1000,
        alarmIntent
)

Java

private AlarmManager alarmMgr;
private PendingIntent alarmIntent;
...
alarmMgr = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
Intent intent = new Intent(context, AlarmReceiver.class);
alarmIntent = PendingIntent.getBroadcast(context, 0, intent, 0);

alarmMgr.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() +
        60 * 1000, alarmIntent);

Ví dụ về chuông báo theo thời gian thực

Dưới đây là một số ví dụ về cách sử dụng RTC_WAKEUP.

Đánh thức thiết bị để kích hoạt chuông báo vào khoảng 2 giờ chiều và lặp lại mỗi ngày một lần:

Kotlin

// Set the alarm to start at approximately 2:00 p.m.
val calendar: Calendar = Calendar.getInstance().apply {
    timeInMillis = System.currentTimeMillis()
    set(Calendar.HOUR_OF_DAY, 14)
}

// With setInexactRepeating(), you have to use one of the AlarmManager interval
// constants--in this case, AlarmManager.INTERVAL_DAY.
alarmMgr?.setInexactRepeating(
        AlarmManager.RTC_WAKEUP,
        calendar.timeInMillis,
        AlarmManager.INTERVAL_DAY,
        alarmIntent
)

Java

// Set the alarm to start at approximately 2:00 p.m.
Calendar calendar = Calendar.getInstance();
calendar.setTimeInMillis(System.currentTimeMillis());
calendar.set(Calendar.HOUR_OF_DAY, 14);

// With setInexactRepeating(), you have to use one of the AlarmManager interval
// constants--in this case, AlarmManager.INTERVAL_DAY.
alarmMgr.setInexactRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
        AlarmManager.INTERVAL_DAY, alarmIntent);

Đánh thức thiết bị để kích hoạt chuông báo vào đúng 8:30 sáng và sau đó 20 phút một lần:

Kotlin

private var alarmMgr: AlarmManager? = null
private lateinit var alarmIntent: PendingIntent
...
alarmMgr = context.getSystemService(Context.ALARM_SERVICE) as AlarmManager
alarmIntent = Intent(context, AlarmReceiver::class.java).let { intent ->
    PendingIntent.getBroadcast(context, 0, intent, 0)
}

// Set the alarm to start at 8:30 a.m.
val calendar: Calendar = Calendar.getInstance().apply {
    timeInMillis = System.currentTimeMillis()
    set(Calendar.HOUR_OF_DAY, 8)
    set(Calendar.MINUTE, 30)
}

// setRepeating() lets you specify a precise custom interval--in this case,
// 20 minutes.
alarmMgr?.setRepeating(
        AlarmManager.RTC_WAKEUP,
        calendar.timeInMillis,
        1000 * 60 * 20,
        alarmIntent
)

Java

private AlarmManager alarmMgr;
private PendingIntent alarmIntent;
...
alarmMgr = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
Intent intent = new Intent(context, AlarmReceiver.class);
alarmIntent = PendingIntent.getBroadcast(context, 0, intent, 0);

// Set the alarm to start at 8:30 a.m.
Calendar calendar = Calendar.getInstance();
calendar.setTimeInMillis(System.currentTimeMillis());
calendar.set(Calendar.HOUR_OF_DAY, 8);
calendar.set(Calendar.MINUTE, 30);

// setRepeating() lets you specify a precise custom interval--in this case,
// 20 minutes.
alarmMgr.setRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
        1000 * 60 * 20, alarmIntent);

Quyết định độ chính xác của chuông báo

Như đã mô tả trước đó, việc chọn loại chuông báo thường là bước đầu tiên trong quá trình tạo chuông báo. Một điểm khác biệt nữa là độ chính xác của chuông báo. Đối với hầu hết ứng dụng, setInexactRepeating() là lựa chọn phù hợp. Khi bạn sử dụng phương thức này, Android sẽ đồng bộ hoá nhiều chuông báo lặp lại không chính xác và kích hoạt các chuông báo đó cùng lúc. Điều này sẽ làm giảm mức tiêu hao pin.

Tránh sử dụng chuông báo chính xác nếu có thể. Tuy nhiên, đối với ứng dụng hiếm có các yêu cầu cố định về thời gian, bạn có thể đặt thông báo chính xác bằng cách gọi setRepeating().

Với setInexactRepeating(), bạn không thể chỉ định khoảng thời gian tuỳ chỉnh như bạn có thể dùng setRepeating(). Bạn phải sử dụng một trong các hằng số khoảng thời gian, chẳng hạn như INTERVAL_FIFTEEN_MINUTES, INTERVAL_DAY, v.v. Hãy xem AlarmManager để biết danh sách đầy đủ.

Hủy chuông báo

Tuỳ thuộc vào ứng dụng của bạn, bạn có thể muốn thêm tính năng huỷ chuông báo. Để huỷ một chuông báo, hãy gọi cancel() trên Trình quản lý chuông báo, truyền vào PendingIntent mà bạn không muốn kích hoạt nữa. Ví dụ:

Kotlin

// If the alarm has been set, cancel it.
alarmMgr?.cancel(alarmIntent)

Java

// If the alarm has been set, cancel it.
if (alarmMgr!= null) {
    alarmMgr.cancel(alarmIntent);
}

Bắt đầu chuông báo khi thiết bị khởi động lại

Theo mặc định, tất cả chuông báo đều bị huỷ khi thiết bị tắt. Để ngăn điều này xảy ra, bạn có thể thiết kế ứng dụng để tự động khởi động lại chuông báo lặp lại nếu người dùng khởi động lại thiết bị. Điều này đảm bảo AlarmManager sẽ tiếp tục thực hiện tác vụ của mình mà không cần người dùng khởi động lại chuông báo theo cách thủ công.

Sau đây là các bước:

  1. Đặt quyền RECEIVE_BOOT_COMPLETED trong tệp kê khai của ứng dụng. Điều này cho phép ứng dụng của bạn nhận ACTION_BOOT_COMPLETED được truyền đi sau khi hệ thống khởi động xong (đây chỉ hiệu quả nếu người dùng đã khởi chạy ứng dụng ít nhất một lần):

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
  2. Triển khai BroadcastReceiver để nhận thông báo truyền tin:

    Kotlin

    class SampleBootReceiver : BroadcastReceiver() {
    
        override fun onReceive(context: Context, intent: Intent) {
            if (intent.action == "android.intent.action.BOOT_COMPLETED") {
                // Set the alarm here.
            }
        }
    }
    

    Java

    public class SampleBootReceiver extends BroadcastReceiver {
    
        @Override
        public void onReceive(Context context, Intent intent) {
            if (intent.getAction().equals("android.intent.action.BOOT_COMPLETED")) {
                // Set the alarm here.
            }
        }
    }
    
  3. Thêm receiver vào tệp kê khai của ứng dụng bằng bộ lọc ý định để lọc hành động ACTION_BOOT_COMPLETED:

    <receiver android:name=".SampleBootReceiver"
            android:enabled="false">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED"></action>
        </intent-filter>
    </receiver>

    Xin lưu ý rằng trong tệp kê khai, trình nhận khởi động được đặt thành android:enabled="false". Điều này có nghĩa là trình thu nhận sẽ không được gọi trừ phi ứng dụng bật lệnh đó một cách rõ ràng. Điều này ngăn việc gọi trình thu khởi động một cách không cần thiết. Bạn có thể bật dịch vụ nhận (ví dụ: nếu người dùng đặt chuông báo) như sau:

    Kotlin

    val receiver = ComponentName(context, SampleBootReceiver::class.java)
    
    context.packageManager.setComponentEnabledSetting(
            receiver,
            PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
            PackageManager.DONT_KILL_APP
    )
    

    Java

    ComponentName receiver = new ComponentName(context, SampleBootReceiver.class);
    PackageManager pm = context.getPackageManager();
    
    pm.setComponentEnabledSetting(receiver,
            PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
            PackageManager.DONT_KILL_APP);
    

    Sau khi bạn bật receiver theo cách này, receiver sẽ luôn bật, ngay cả khi người dùng khởi động lại thiết bị. Nói cách khác, việc cho phép trình thu nhận theo phương thức lập trình sẽ ghi đè chế độ cài đặt tệp kê khai, ngay cả trên các lần khởi động lại. Bộ nhận sẽ vẫn bật cho đến khi ứng dụng của bạn tắt. Bạn có thể tắt bộ nhận (ví dụ: nếu người dùng huỷ chuông báo) như sau:

    Kotlin

    val receiver = ComponentName(context, SampleBootReceiver::class.java)
    
    context.packageManager.setComponentEnabledSetting(
            receiver,
            PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
            PackageManager.DONT_KILL_APP
    )
    

    Java

    ComponentName receiver = new ComponentName(context, SampleBootReceiver.class);
    PackageManager pm = context.getPackageManager();
    
    pm.setComponentEnabledSetting(receiver,
            PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
            PackageManager.DONT_KILL_APP);
    

Gọi chuông báo khi thiết bị ở chế độ Nghỉ

Các thiết bị chạy Android 6.0 (API cấp 23) hỗ trợ chế độ Nghỉ ngơi, giúp kéo dài thời lượng pin của thiết bị. Chuông báo sẽ không kích hoạt khi thiết bị ở Chế độ nghỉ. Mọi chuông báo đã lên lịch đều bị trì hoãn cho đến khi thiết bị thoát khỏi chế độ Nghỉ. Nếu cần hoàn tất công việc ngay cả khi thiết bị ở trạng thái rảnh, bạn có thể sử dụng một số tuỳ chọn sau:

  • Đặt thông báo chính xác.

  • Sử dụng API WorkManager được xây dựng để thực hiện công việc ở chế độ nền. Bạn có thể cho biết rằng hệ thống cần đẩy nhanh công việc của bạn để công việc hoàn tất sớm nhất có thể. Để biết thêm thông tin, hãy xem bài viết Lên lịch cho các tác vụ bằng WorkManager

Các phương pháp hay nhất

Mọi lựa chọn bạn đưa ra khi thiết kế chuông báo lặp lại đều có thể dẫn đến những hậu quả trong cách ứng dụng sử dụng (hoặc lạm dụng) tài nguyên hệ thống. Ví dụ: hãy tưởng tượng một ứng dụng phổ biến đồng bộ hoá với máy chủ. Nếu hoạt động đồng bộ hoá dựa trên thời gian đồng hồ và mọi phiên bản của ứng dụng đồng bộ hoá lúc 11 giờ đêm, thì việc tải trên máy chủ có thể dẫn đến độ trễ cao hoặc thậm chí "từ chối dịch vụ". Hãy làm theo các phương pháp hay nhất sau đây khi sử dụng chuông báo:

  • Thêm độ ngẫu nhiên (biến động) vào mọi yêu cầu mạng kích hoạt do chuông báo lặp lại:

    • Thực hiện mọi thao tác cục bộ khi chuông báo kích hoạt. "Công việc cục bộ" có nghĩa là mọi hoạt động không ảnh hưởng đến máy chủ hoặc yêu cầu dữ liệu từ máy chủ.

    • Đồng thời, hãy lên lịch cho chuông báo chứa các yêu cầu mạng kích hoạt vào một khoảng thời gian ngẫu nhiên nào đó.

  • Duy trì tần suất báo thức ở mức tối thiểu.

  • Đừng đánh thức thiết bị khi không cần thiết (hành vi này được xác định bởi loại chuông báo, như mô tả trong phần Chọn loại chuông báo).

  • Đừng đặt thời gian kích hoạt của báo thức chính xác hơn mức cần thiết.

    Sử dụng setInexactRepeating() thay vì setRepeating(). Khi bạn sử dụng setInexactRepeating(), Android sẽ đồng bộ hoá các chuông báo lặp lại từ nhiều ứng dụng và kích hoạt các chuông báo đó cùng lúc. Điều này làm giảm tổng số lần hệ thống phải đánh thức thiết bị, từ đó giảm mức tiêu hao pin. Kể từ Android 4.4 (API cấp 19), tất cả chuông báo lặp lại đều là cảnh báo không chính xác. Xin lưu ý rằng mặc dù setInexactRepeating() là một phiên bản cải tiến so với setRepeating(), nhưng nó vẫn có thể làm máy chủ bị quá tải nếu mọi thực thể của một ứng dụng đều truy cập vào máy chủ cùng một lúc. Do đó, đối với các yêu cầu về mạng, hãy thêm một số ngẫu nhiên vào chuông báo như đã thảo luận trước đó.

  • Tránh đặt báo thức dựa vào giờ đồng hồ nếu có thể.

    Việc lặp lại chuông báo dựa trên thời gian kích hoạt chính xác sẽ không điều chỉnh được theo tỷ lệ. Sử dụng ELAPSED_REALTIME nếu có thể. Các loại chuông báo khác nhau được mô tả chi tiết hơn trong phần sau.