الميزات وواجهات برمجة التطبيقات

يقدّم Android 17 ميزات وواجهات برمجة تطبيقات جديدة ورائعة للمطوّرين. تلخّص الأقسام التالية هذه الميزات لمساعدتك في البدء باستخدام واجهات برمجة التطبيقات ذات الصلة.

للحصول على قائمة مفصّلة بواجهات برمجة التطبيقات الجديدة والمعدَّلة والمُزالة، يُرجى قراءة تقرير مقارنة واجهات برمجة التطبيقات. للحصول على تفاصيل حول واجهات برمجة التطبيقات الجديدة، يُرجى الانتقال إلى مرجع واجهة برمجة تطبيقات Android. يتم تمييز واجهات برمجة التطبيقات الجديدة لتسهيل رؤيتها.

عليك أيضًا مراجعة المجالات التي قد تؤثر فيها تغييرات النظام الأساسي في تطبيقاتك. لمزيد من المعلومات، يُرجى الاطّلاع على الصفحات التالية:

الوظيفة الأساسية

يضيف Android 17 الميزات الجديدة التالية المتعلقة بالوظيفة الأساسية لنظام Android.

عوامل التشغيل الجديدة في ProfilingManager

Android 17 向 ProfilingManager 添加了多个新的系统触发器,以 帮助您收集深入数据来调试性能问题。

新触发器包括:

如需了解如何设置系统触发器,请参阅有关 基于触发器的性能分析的文档以及有关如何检索和分析性能分析数据 的文档

应用异常的性能分析触发器

Android 17 引入了一项设备端异常检测服务,用于监控资源密集型行为和潜在的兼容性回归。此服务与ProfilingManager集成,可让您的应用接收由特定系统检测到的事件触发的性能分析工件。

使用 TRIGGER_TYPE_ANOMALY 触发器检测系统性能问题 例如 binder 调用过多和内存用量过高。当应用违反操作系统定义的内存限制时,异常触发器允许开发者接收特定于应用的堆转储,以帮助识别和修复内存问题。此外,对于 binder 垃圾内容过多,异常触发器会提供有关 binder 事务的堆栈抽样分析报告。

此 API 回调发生在系统强制执行任何操作之前。例如,它可以帮助开发者在应用因超出内存限制而被系统终止之前收集调试数据。

val profilingManager =
    applicationContext.getSystemService(ProfilingManager::class.java)
val triggers = ArrayList<ProfilingTrigger>()
triggers.add(ProfilingTrigger.Builder(ProfilingTrigger.TRIGGER_TYPE_ANOMALY))
val mainExecutor: Executor = Executors.newSingleThreadExecutor()
val resultCallback = Consumer<ProfilingResult> { profilingResult ->
    if (profilingResult.errorCode != ProfilingResult.ERROR_NONE) {
        // upload profile result to server for further analysis
        setupProfileUploadWorker(profilingResult.resultFilePath)
    }
    profilingManager.registerForAllProfilingResults(mainExecutor,
                                                    resultCallback)
    profilingManager.addProfilingTriggers(triggers)
}

واجهات برمجة التطبيقات JobDebugInfo

Android 17 引入了新的 JobDebugInfo API,可帮助开发者调试其 JobScheduler 作业,了解作业未运行的原因、运行时长以及其他汇总信息。

扩展后的 JobDebugInfo API 的第一个方法是 getPendingJobReasonStats(),该方法会返回一个映射,其中包含作业处于待执行状态的原因及其各自的累计待执行时长。此方法将 getPendingJobReasonsHistory()getPendingJobReasons() 方法联接在一起,可让您了解预定作业未按预期运行的原因,但通过在单个方法中同时提供时长和作业原因,简化了信息检索。

例如,对于指定的 jobId,该方法可能会返回 PENDING_JOB_REASON_CONSTRAINT_CHARGING 和 60000 毫秒的时长,表示作业因未满足充电约束而处于等待状态 60000 毫秒。

تقليل عمليات قفل التنشيط باستخدام ميزة متتبِّع المنبّهات التي يتم تفعيلها أثناء وضع غير مستخدَم من قِبل أي برنامج حاليًا

يقدّم نظام التشغيل Android 17 نوعًا جديدًا من AlarmManager.setExactAndAllowWhileIdle يقبل OnAlarmListener بدلاً من PendingIntent. تُعد الآلية الجديدة المستندة إلى معاودة الاتصال مثالية للتطبيقات التي تعتمد حاليًا على عمليات قفل التنشيط المستمرة لتنفيذ مهام دورية، مثل تطبيقات المراسلة التي تحافظ على اتصالات المقبس.

الخصوصية

يتضمّن Android 17 الميزات الجديدة التالية لتحسين خصوصية المستخدم.

توافُق النظام الأساسي مع Encrypted Client Hello ‏ (ECH)

يتيح الإصدار 17 من نظام التشغيل Android استخدام Encrypted Client Hello (ECH)، وهي ميزة مهمة لتحسين الخصوصية في ما يتعلّق باتصالات الشبكة. ‫ECH هي إضافة إلى الإصدار 1.3 من بروتوكول TLS تشفّر الإشارة إلى اسم الخادم (SNI) أثناء تأكيد اتصال TLS الأوّلي. يساعد هذا التشفير في حماية خصوصية المستخدمين من خلال صعوبة تحديد الوسطاء في الشبكة للنطاق المحدّد الذي يتصل به التطبيق.

تتضمّن المنصة الآن واجهات برمجة التطبيقات اللازمة لمكتبات إنشاء الشبكات من أجل تنفيذ ECH. ويشمل ذلك إمكانات جديدة في DnsResolver للاستعلام عن سجلّات نظام أسماء النطاقات (DNS) عبر HTTPS التي تتضمّن إعدادات ECH، وطُرقًا جديدة في SSLEngines وSSLSockets من Conscrypt لتفعيل ECH من خلال تمرير هذه الإعدادات عند الاتصال بنطاق. يمكن للمطوّرين ضبط الإعدادات المفضَّلة لـ ECH، مثل تفعيلها بشكل انتهازي أو فرض استخدامها، من خلال العنصر الجديد <domainEncryption> ضمن ملف إعدادات أمان الشبكة، والذي يمكن تطبيقه على مستوى العالم أو على أساس كل نطاق.

من المتوقّع أن تدمج مكتبات الشبكات الشائعة، مثل HttpEngine وWebView وOkHttp، واجهات برمجة التطبيقات هذه في التحديثات المستقبلية، ما يسهّل على التطبيقات استخدام ECH وتعزيز خصوصية المستخدمين.

لمزيد من المعلومات، يُرجى الاطّلاع على مستندات Encrypted Client Hello.

أداة اختيار جهات الاتصال في Android

Android 联系人选择工具是一个标准化的可浏览界面,供用户与您的应用分享联系人。该选择工具适用于搭载 Android 17(API 级别 37)或更高版本的设备,可提供一种可保护隐私的替代方案,以取代广泛的 READ_CONTACTS 权限。您的应用无需请求访问用户的整个地址簿,而是指定所需的数据字段(例如电话号码或电子邮件地址),然后用户选择要分享的特定联系人。这样,您的应用便只能读取所选数据,从而确保精细控制,同时提供一致的用户体验,并具有内置搜索、个人资料切换和多选功能,而无需构建或维护界面。

如需了解详情,请参阅联系人选择工具文档

الأمان

يضيف Android 17 الميزات الجديدة التالية لتحسين أمان الجهاز والتطبيق.

وضع "الحماية المتقدّمة على Android"‏ (AAPM)

يوفّر "وضع الحماية المتقدّمة" على Android مجموعة جديدة وفعّالة من ميزات الأمان لمستخدمي Android، ما يمثّل خطوة مهمة في حماية المستخدمين، لا سيما المعرّضين لخطر أكبر، من الهجمات المتطورة. تم تصميم "إدارة التطبيقات تلقائيًا" كميزة اختيارية، ويتم تفعيلها من خلال إعداد واحد يمكن للمستخدمين تفعيله في أي وقت لتطبيق مجموعة من إجراءات الحماية الأمنية.

تشمل هذه الإعدادات الأساسية حظر تثبيت التطبيقات من مصادر غير معروفة (التثبيت الجانبي) وتقييد إشارات بيانات USB وفرض عمليات الفحص التي تجريها خدمة "Google Play للحماية"، ما يقلّل بشكل كبير من مساحة سطح الاختراق على الجهاز. يمكن للمطوّرين الاستفادة من هذه الميزة باستخدام واجهة برمجة التطبيقات AdvancedProtectionManager لرصد حالة الوضع، ما يتيح للتطبيقات اعتماد وضع أمان محسّن تلقائيًا أو حظر الوظائف العالية الخطورة عندما يوافق المستخدم على تفعيل الوضع.

توقيع حزمة APK باستخدام تقنية PQC

Android 现在支持混合 APK 签名方案,以保护应用的签名身份免受利用量子计算的攻击的潜在威胁。此功能引入了一种新的 APK 签名方案,可让您将经典签名密钥(例如 RSA 或 EC)与新的后量子加密 (PQC) 算法 (ML-DSA) 配对。

这种混合方法可确保您的应用在未来免受量子攻击,同时与依赖于经典签名验证的旧版 Android 和设备保持完全的向后兼容性。

对开发者的影响

  • 使用 Play 应用签名的应用:如果您使用 Play 应用签名,可以等待 Google Play 为您提供使用 Google Play 生成的 PQC 密钥升级混合签名的选项,从而确保您的应用受到保护,而无需手动管理密钥。
  • 使用自行管理的密钥的应用:自行管理签名密钥的开发者可以利用更新后的 Android build 工具(例如 apksigner)轮换到混合身份,将 PQC 密钥与新的经典密钥相结合。(您必须创建新的经典密钥,无法重复使用旧密钥。)

إمكانية الاتصال

يضيف Android 17 الميزات التالية لتحسين إمكانية اتصال الجهاز والتطبيق.

شبكات الأقمار الصناعية المقيّدة

تتضمّن هذه السمة تحسينات تتيح للتطبيقات العمل بفعالية على شبكات الأقمار الصناعية ذات معدل نقل البيانات المنخفض.

تجربة المستخدم وواجهة مستخدم النظام

يتضمّن Android 17 التغييرات التالية لتحسين تجربة المستخدم.

مصدر صوت مخصّص لمستوى صوت "مساعد Google"

Android 17 为 Google 助理应用引入了专用的 Google 助理音量流, 以便使用 USAGE_ASSISTANT 进行播放。此项更改将 Google 助理音频与标准媒体流分离,让用户可以单独控制这两个音量。这样便可实现以下场景:将媒体播放静音,同时保持 Google 助理响应的可听性,反之亦然。

有权访问新的 MODE_ASSISTANT_CONVERSATION 音频模式的 Google 助理应用可以进一步提高音量控制的一致性。Google 助理应用可以使用此模式向系统提供有关活跃 Google 助理会话的提示,确保可以在活跃 USAGE_ASSISTANT 播放之外或使用连接的蓝牙外设控制 Google 助理流。

التسليم (Handoff)

切换是 Android 17 中新增的一项功能和 API,应用开发者可以将其集成到应用中,以便为用户提供跨设备连续性。它允许用户在一个 Android 设备上启动应用 activity,然后将其转移到另一个 Android 设备。Handoff 在用户设备的后台运行,并通过各种入口点(例如接收设备上的启动器和任务栏)显示用户附近其他设备上的可用活动。

应用可以指定 Handoff 来启动相同的原生 Android 应用(如果该应用已安装在接收设备上且可供使用)。在此应用到应用流程中,用户通过深层链接跳转到指定 activity。或者,应用到网站切换功能可以作为后备选项提供,也可以通过网址切换功能直接实现。

切换支持是按 activity 实现的。如需启用 Handoff,请针对 activity 调用 setHandoffEnabled() 方法。可能需要随切换传递其他数据,以便接收设备上重新创建的 activity 可以恢复适当的状态。实现 onHandoffActivityDataRequested() 回调以返回 HandoffActivityData 对象,该对象包含用于指定 Handoff 应如何处理并在接收设备上重新创建 activity 的详细信息。

تحديث مباشر: واجهة برمجة التطبيقات للألوان الدلالية

في نظام التشغيل Android 17، تطلق ميزة التحديث المباشر واجهات برمجة التطبيقات Semantic Coloring لدعم الألوان ذات المعنى العالمي.

تتيح الفئات التالية التلوين الدلالي:

ألعاب التلوين

  • الأخضر: يرتبط بالأمان. يجب استخدام هذا اللون في الحالات التي تشير إلى أنّك في وضع آمن.
  • اللون البرتقالي: يُستخدم للإشارة إلى الحذر ولتمييز المخاطر المادية. يجب استخدام هذا اللون في الحالات التي يحتاج فيها المستخدمون إلى الانتباه لضبط إعدادات حماية أفضل.
  • أحمر: يشير عادةً إلى الخطر أو التوقف. يجب أن يتم عرضها في الحالات التي تتطلّب جذب انتباه المستخدمين بشكل عاجل.
  • الأزرق: لون محايد للمحتوى الذي يقدّم معلومات ويجب أن يبرز عن المحتوى الآخر.

يوضّح المثال التالي كيفية تطبيق أنماط دلالية على النص في إشعار:

  val ssb = SpannableStringBuilder()
        .append("Colors: ")
        .append("NONE", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_UNSPECIFIED), 0)
        .append(", ")
        .append("INFO", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_INFO), 0)
        .append(", ")
        .append("SAFE", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_SAFE), 0)
        .append(", ")
        .append("CAUTION", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_CAUTION), 0)
        .append(", ")
        .append("DANGER", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_DANGER), 0)

    Notification.Builder(context, channelId)
          .setSmallIcon(R.drawable.ic_icon)
          .setContentTitle("Hello World!")
          .setContentText(ssb)
          .setOngoing(true)
              .setRequestPromotedOngoing(true)

واجهة برمجة التطبيقات UWB Downlink-TDoA لنظام Android 17

تتيح ميزة تحديد المدى المستندة إلى الفرق في وقت الوصول بين الإشارات المرسَلة من المحطة الأساسية إلى الجهاز (DL-TDoA) لجهاز ما تحديد موضعه بالنسبة إلى نقاط وصول متعددة من خلال قياس أوقات الوصول النسبية للإشارات.

يوضّح المقتطف التالي كيفية تهيئة Ranging Manager والتحقّق من إمكانات الجهاز وبدء جلسة DL-TDoA:

Kotlin

class RangingApp {

    fun initDlTdoa(context: Context) {
        // Initialize the Ranging Manager
        val rangingManager = context.getSystemService(RangingManager::class.java)

        // Register for device capabilities
        val capabilitiesCallback = object : RangingManager.RangingCapabilitiesCallback {
            override fun onRangingCapabilities(capabilities: RangingCapabilities) {
                // Make sure Dl-TDoA is supported before starting the session
                if (capabilities.uwbCapabilities != null && capabilities.uwbCapabilities!!.isDlTdoaSupported) {
                    startDlTDoASession(context)
                }
            }
        }
        rangingManager.registerCapabilitiesCallback(Executors.newSingleThreadExecutor(), capabilitiesCallback)
    }

    fun startDlTDoASession(context: Context) {

        // Initialize the Ranging Manager
        val rangingManager = context.getSystemService(RangingManager::class.java)

        // Create session and configure parameters
        val executor = Executors.newSingleThreadExecutor()
        val rangingSession = rangingManager.createRangingSession(executor, RangingSessionCallback())
        val rangingRoundIndexes = byteArrayOf(0)
        val config: ByteArray = byteArrayOf() // OOB config data
        val params = DlTdoaRangingParams.createFromFiraConfigPacket(config, rangingRoundIndexes)

        val rangingDevice = RangingDevice.Builder().build()
        val rawTagDevice = RawRangingDevice.Builder()
            .setRangingDevice(rangingDevice)
            .setDlTdoaRangingParams(params)
            .build()

        val dtTagConfig = RawDtTagRangingConfig.Builder(rawTagDevice).build()

        val preference = RangingPreference.Builder(DEVICE_ROLE_DT_TAG, dtTagConfig)
            .setSessionConfig(SessionConfig.Builder().build())
            .build()

        // Start the ranging session
        rangingSession.start(preference)
    }
}

private class RangingSessionCallback : RangingSession.Callback {
    override fun onDlTdoaResults(peer: RangingDevice, measurement: DlTdoaMeasurement) {
        // Process measurement results here
    }
}

Java

public class RangingApp {

    public void initDlTdoa(Context context) {

        // Initialize the Ranging Manager
        RangingManager rangingManager = context.getSystemService(RangingManager.class);

        // Register for device capabilities
        RangingManager.CapabilitiesCallback capabilitiesCallback = new RangingManager.RangingCapabilitiesCallback() {
            @Override
            public void onRangingCapabilities(RangingCapabilities capabilities) {
                // Make sure Dl-TDoA is supported before starting the session
                if (capabilities.getUwbCapabilities() != null && capabilities.getUwbCapabilities().isDlTdoaSupported()) {
                    startDlTDoASession(context);
                }
            }
        };
        rangingManager.registerCapabilitiesCallback(Executors.newSingleThreadExecutor(), capabilitiesCallback);
    }

    public void startDlTDoASession(Context context) {
        RangingManager rangingManager = context.getSystemService(RangingManager.class);

        // Create session and configure parameters
        Executor executor = Executors.newSingleThreadExecutor();
        RangingSession rangingSession = rangingManager.createRangingSession(executor, new RangingSessionCallback());
        byte[] rangingRoundIndexes = new byte[] {0};
        byte[] config = new byte[0]; // OOB config data
        DlTdoaRangingParams params = DlTdoaRangingParams.createFromFiraConfigPacket(config, rangingRoundIndexes);

        RangingDevice rangingDevice = new RangingDevice.Builder().build();
        RawRangingDevice rawTagDevice = new RawRangingDevice.Builder()
                .setRangingDevice(rangingDevice)
                .setDlTdoaRangingParams(params)
                .build();

        RawDtTagRangingConfig dtTagConfig = new RawDtTagRangingConfig.Builder(rawTagDevice).build();

        RangingPreference preference = new RangingPreference.Builder(DEVICE_ROLE_DT_TAG, dtTagConfig)
                .setSessionConfig(new SessionConfig.Builder().build())
                .build();

        // Start the ranging session
        rangingSession.start(preference);
    }

    private static class RangingSessionCallback implements RangingSession.Callback {

        @Override
        public void onDlTdoaResults(RangingDevice peer, DlTdoaMeasurement measurement) {
            // Process measurement results here
        }
    }
}

إعدادات النطاق الخارجي (OOB)

يقدّم المقتطف التالي مثالاً على بيانات إعداد DL-TDoA OOB لشبكة Wi-Fi وBLE:

Java

// Wifi Configuration
byte[] wifiConfig = {
    (byte) 0xDD, (byte) 0x2D, (byte) 0x5A, (byte) 0x18, (byte) 0xFF, // Header
    (byte) 0x5F, (byte) 0x19, // FiRa Sub-Element
    (byte) 0x02, (byte) 0x00, // Profile ID
    (byte) 0x06, (byte) 0x02, (byte) 0x20, (byte) 0x08, // MAC Address
    (byte) 0x14, (byte) 0x01, (byte) 0x0C, // Preamble Index
    (byte) 0x27, (byte) 0x02, (byte) 0x08, (byte) 0x07, // Vendor ID
    (byte) 0x28, (byte) 0x06, (byte) 0xCA, (byte) 0xC8, (byte) 0xA6, (byte) 0xF7, (byte) 0x6F, (byte) 0x08, // Static STS IV
    (byte) 0x08, (byte) 0x02, (byte) 0x60, (byte) 0x09, // Slot Duration
    (byte) 0x1B, (byte) 0x01, (byte) 0x0A, // Slots per RR
    (byte) 0x09, (byte) 0x04, (byte) 0xE8, (byte) 0x03, (byte) 0x00, (byte) 0x00, // Duration
    (byte) 0x9F, (byte) 0x04, (byte) 0x67, (byte) 0x45, (byte) 0x23, (byte) 0x01  // Session ID
};

// BLE Configuration
byte[] bleConfig = {
    (byte) 0x2D, (byte) 0x16, (byte) 0xF4, (byte) 0xFF, // Header
    (byte) 0x5F, (byte) 0x19, // FiRa Sub-Element
    (byte) 0x02, (byte) 0x00, // Profile ID
    (byte) 0x06, (byte) 0x02, (byte) 0x20, (byte) 0x08, // MAC Address
    (byte) 0x14, (byte) 0x01, (byte) 0x0C, // Preamble Index
    (byte) 0x27, (byte) 0x02, (byte) 0x08, (byte) 0x07, // Vendor ID
    (byte) 0x28, (byte) 0x06, (byte) 0xCA, (byte) 0xC8, (byte) 0xA6, (byte) 0xF7, (byte) 0x6F, (byte) 0x08, // Static STS IV
    (byte) 0x08, (byte) 0x02, (byte) 0x60, (byte) 0x09, // Slot Duration
    (byte) 0x1B, (byte) 0x01, (byte) 0x0A, // Slots per RR
    (byte) 0x09, (byte) 0x04, (byte) 0xE8, (byte) 0x03, (byte) 0x00, (byte) 0x00, // Duration
    (byte) 0x9F, (byte) 0x04, (byte) 0x67, (byte) 0x45, (byte) 0x23, (byte) 0x01  // Session ID
};

إذا تعذّر عليك استخدام إعداد OOB لأنّه غير متوفّر، أو إذا كنت بحاجة إلى تغيير القيم التلقائية غير المتوفّرة في إعداد OOB، يمكنك إنشاء مَعلمات باستخدام DlTdoaRangingParams.Builder كما هو موضّح في المقتطف التالي. يمكنك استخدام المَعلمات التالية بدلاً من DlTdoaRangingParams.createFromFiraConfigPacket():

Kotlin

val dlTdoaParams = DlTdoaRangingParams.Builder(1)
    .setComplexChannel(UwbComplexChannel.Builder()
            .setChannel(9).setPreambleIndex(10).build())
    .setDeviceAddress(deviceAddress)
    .setSessionKeyInfo(byteArrayOf(0x01, 0x02, 0x03, 0x04))
    .setRangingIntervalMillis(240)
    .setSlotDuration(UwbRangingParams.DURATION_2_MS)
    .setSlotsPerRangingRound(20)
    .setRangingRoundIndexes(byteArrayOf(0x01, 0x05))
    .build()

Java

DlTdoaRangingParams dlTdoaParams = new DlTdoaRangingParams.Builder(1)
    .setComplexChannel(new UwbComplexChannel.Builder()
            .setChannel(9).setPreambleIndex(10).build())
    .setDeviceAddress(deviceAddress)
    .setSessionKeyInfo(new byte[]{0x01, 0x02, 0x03, 0x04})
    .setRangingIntervalMillis(240)
    .setSlotDuration(UwbRangingParams.DURATION_2_MS)
    .setSlotsPerRangingRound(20)
    .setRangingRoundIndexes(new byte[]{0x01, 0x05})
    .build();