0

0

如何构建每小时报时的Android语音时钟

霞舞

霞舞

发布时间:2025-12-27 20:51:04

|

114人浏览过

|

来源于php中文网

原创

如何构建每小时报时的Android语音时钟

本文详解如何在android中实现精准的每小时语音报时功能,修正原代码中因handler无限循环导致的秒级误触发问题,并提供基于alarmmanager的稳定解决方案。

在Android开发中,实现“整点语音报时”看似简单,但若使用Handler + postDelayed()轮询检测时间(如原代码中每1秒检查一次Minute == 00),不仅造成CPU与电量浪费,更会因逻辑漏洞(如未清除旧任务、未处理跨天/休眠场景)导致重复播报或漏报——正如提问者所遇:本想每小时说一次“X点整”,结果每秒都在执行判断,甚至可能在非整点时刻意外触发(例如Hour==3 && Minute==00在任意秒都成立,但postDelayed(this, 1000)让该判断永不停歇)。

✅ 正确思路:避免轮询,改用系统级定时调度
Handler适用于UI线程短周期任务,但整点触发属于精确、低频、需跨进程/休眠存活的场景,应优先选用 AlarmManager(兼容至API 19+)或 WorkManager(推荐用于Android 12+后台限制场景)。以下是基于 AlarmManager 的轻量级实现方案:

Pixelcut
Pixelcut

AI产品图片处理——背景移除替换、物体抹除和图片放大

下载

1. 配置权限与组件(AndroidManifest.xml)



 

2. 创建广播接收器(负责语音播报)

public class TimeSpeakReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // 获取前台Activity或使用NotificationChannel播报(避免后台TTS限制)
        TextToSpeech tts = new TextToSpeech(context, status -> {
            if (status == TextToSpeech.SUCCESS) {
                tts.setLanguage(Locale.getDefault());
                Calendar now = Calendar.getInstance();
                int hour = now.get(Calendar.HOUR_OF_DAY); // 24小时制
                String timeStr = String.format("The time is %d o'clock", 
                    hour == 0 ? 12 : (hour > 12 ? hour - 12 : hour));

                // 使用QUEUE_ADD避免打断上一次播报
                tts.speak(timeStr, TextToSpeech.QUEUE_FLUSH, null, null);

                // 延迟释放资源(防止TTS未完成即销毁)
                new Handler(Looper.getMainLooper()).postDelayed(() -> {
                    if (tts != null) {
                        tts.stop();
                        tts.shutdown();
                    }
                }, 3000);
            }
        });
    }
}

3. 在Activity中设置首次报警并注册Receiver

private void scheduleNextHourlyAlarm() {
    AlarmManager alarmManager = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
    Intent intent = new Intent(this, TimeSpeakReceiver.class);
    PendingIntent pendingIntent = PendingIntent.getBroadcast(
        this, 0, intent, PendingIntent.FLAG_IMMUTABLE | PendingIntent.FLAG_ONE_SHOT);

    // 计算下一个整点时间(例如现在14:25 → 设为15:00:00)
    Calendar nextHour = Calendar.getInstance();
    nextHour.set(Calendar.MINUTE, 0);
    nextHour.set(Calendar.SECOND, 0);
    nextHour.set(Calendar.MILLISECOND, 0);
    if (nextHour.get(Calendar.MINUTE) != 0 || nextHour.get(Calendar.SECOND) != 0) {
        nextHour.add(Calendar.HOUR_OF_DAY, 1);
    }

    // 设置精确报警(Android 6.0+建议用setExactAndAllowWhileIdle)
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        alarmManager.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP,
            nextHour.getTimeInMillis(), pendingIntent);
    } else {
        alarmManager.setExact(AlarmManager.RTC_WAKEUP,
            nextHour.getTimeInMillis(), pendingIntent);
    }
}

⚠️ 关键注意事项:

  • 不要在BroadcastReceiver中长期持有TTS实例:TTS初始化耗时且需上下文,应在onReceive()内创建并及时释放(如示例中3秒后shutdown);
  • 处理系统休眠:setExactAndAllowWhileIdle()确保设备休眠时仍能唤醒执行;
  • 避免重复注册:在onCreate()中调用scheduleNextHourlyAlarm()一次即可,无需循环;
  • 适配24小时制与12小时口语化:示例中将0点转为12,13-23转为1-11,符合英语报时习惯;
  • Android 12+后台限制:若需更高可靠性,可结合Foreground Service + Notification,但需用户授权。

通过此方案,语音报时将严格在每个整点(如9:00:00、10:00:00)精准触发一次,彻底规避轮询陷阱,同时兼顾系统兼容性与电池效率。

相关专题

更多
pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1844

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2080

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

915

2024.11.28

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

465

2023.08.10

android开发三大框架
android开发三大框架

android开发三大框架是XUtil框架、volley框架、ImageLoader框架。本专题为大家提供android开发三大框架相关的各种文章、以及下载和课程。

244

2023.08.14

android是什么系统
android是什么系统

Android是一种功能强大、灵活可定制、应用丰富、多任务处理能力强、兼容性好、网络连接能力强的操作系统。本专题为大家提供android相关的文章、下载、课程内容,供大家免费下载体验。

1708

2023.08.22

android权限限制怎么解开
android权限限制怎么解开

android权限限制可以使用Root权限、第三方权限管理应用程序、ADB命令和Xposed框架解开。详细介绍:1、Root权限,通过获取Root权限,用户可以解锁所有权限,并对系统进行自定义和修改;2、第三方权限管理应用程序,用户可以轻松地控制和管理应用程序的权限;3、ADB命令,用户可以在设备上执行各种操作,包括解锁权限;4、Xposed框架,用户可以在不修改系统文件的情况下修改应用程序的行为和权限。

1940

2023.09.19

android重启应用的方法有哪些
android重启应用的方法有哪些

android重启应用有通过Intent、PendingIntent、系统服务、Runtime等方法。本专题为大家提供Android相关的文章、下载、课程内容,供大家免费下载体验。

254

2023.10.18

ip地址修改教程大全
ip地址修改教程大全

本专题整合了ip地址修改教程大全,阅读下面的文章自行寻找合适的解决教程。

27

2025.12.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Excel 教程
Excel 教程

共162课时 | 9.7万人学习

Java 教程
Java 教程

共578课时 | 38.5万人学习

Uniapp从零开始实现新闻资讯应用
Uniapp从零开始实现新闻资讯应用

共64课时 | 6.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号