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

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

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

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

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

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

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

يضيف Android 17 عدة عوامل تشغيل جديدة على مستوى النظام إلى ProfilingManager لمساعدتك في جمع بيانات مفصّلة لتصحيح مشاكل الأداء.

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

  • TRIGGER_TYPE_COLD_START: يتم تشغيل المشغِّل أثناء تشغيل التطبيق على البارد. ويوفّر في الردّ نموذجًا لحزمة التنفيذ وتتبُّعًا للنظام.
  • TRIGGER_TYPE_OOM: يتم تشغيل العامل عندما يعرض التطبيق OutOfMemoryError ويوفّر تفريغًا لذاكرة Java المؤقتة في الردّ.
  • TRIGGER_TYPE_KILL_EXCESSIVE_CPU_USAGE: يتم تشغيل المشغّل عندما يتم إيقاف التطبيق بسبب الاستخدام غير الطبيعي والمفرط لوحدة المعالجة المركزية، ويوفّر نموذجًا لحزمة التنفيذ في الردّ.
  • TRIGGER_TYPE_ANOMALY: يتم تشغيل العامل لرصد الحالات غير الطبيعية في أداء النظام، مثل عدد كبير من طلبات الإجراءات في Binder والاستخدام المفرط للذاكرة.

لفهم كيفية إعداد عامل تشغيل النظام، يمكنك الاطّلاع على المستندات حول إنشاء ملفات الأداء استنادًا إلى عوامل التشغيل وكيفية استرداد بيانات الأداء وتحليلها المستندات.

عامل تشغيل ملفات الأداء للحالات غير الطبيعية في التطبيق

يقدّم Android 17 خدمة لرصد الحالات غير الطبيعية على الجهاز فقط تراقب السلوكيات التي تستهلك الكثير من الموارد وحالات التراجع المحتملة في التوافق. تسمح هذه الخدمة لتطبيقك، عند دمجها مع ProfilingManager، بتلقّي بيانات الأداء التي يتم تشغيلها من خلال أحداث معيّنة يرصدها النظام.

استخدِم المشغِّل TRIGGER_TYPE_ANOMALY لرصد مشاكل في أداء النظام مثل عدد كبير من طلبات الإجراءات في Binder والاستخدام المفرط للذاكرة. عندما يخالف التطبيق حدود الذاكرة التي يحدّدها نظام التشغيل، يسمح عامل تشغيل الحالات غير الطبيعية للمطوّرين بتلقّي تفريغات لذاكرة التطبيق المؤقتة للمساعدة في تحديد مشاكل الذاكرة وحلّها. بالإضافة إلى ذلك، بالنسبة إلى عدد كبير من طلبات الإجراءات في Binder، يوفّر عامل تشغيل الحالات غير الطبيعية ملفًا شخصيًا لعينات مكدس طلبات الإجراءات بشأن معاملات Binder.

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

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)

Android 17 引入了对加密客户端 Hello (ECH) 的平台支持,这是对网络通信的一项重大隐私增强功能。ECH 是一项 TLS 1.3 扩展,可在初始 TLS 握手期间加密服务器名称指示 (SNI)。这种加密有助于保护用户隐私,因为它可以让网络中介更难识别应用连接到的特定网域。

该平台现在包含网络库实现 ECH 所需的 API。这包括 DnsResolver 中的新功能,用于查询包含 ECH 配置的 HTTPS DNS 记录;以及 Conscrypt 的 SSLEngine 和 SSLSocket 中的新方法,用于在连接到网域时传入这些配置来启用 ECH。开发者可以通过网络安全配置文件中的新 <domainEncryption> 元素来配置 ECH 偏好设置,例如机会性地启用 ECH 或强制使用 ECH,这些设置可全局应用,也可按网域应用。

预计 HttpEngine、WebView 和 OkHttp 等热门联网库将在未来的更新中集成这些平台 API,从而使应用能够更轻松地采用 ECH 并增强用户隐私保护。

如需了解详情,请参阅加密的客户端 Hello 文档。

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

&quot;منتقي جهات الاتصال في Android&quot; هو واجهة موحّدة يمكن للمستخدمين تصفّحها لمشاركة جهات الاتصال مع تطبيقك. ويتوفّر هذا المنتقي على الأجهزة التي تعمل بالإصدار 17 من نظام التشغيل Android (مستوى واجهة برمجة التطبيقات 37) أو الإصدارات الأحدث، وهو يوفّر بديلاً يحافظ على الخصوصية للإذن الواسع النطاق READ_CONTACTS. بدلاً من طلب الوصول إلى دفتر العناوين الكامل للمستخدم، يحدّد تطبيقك حقول البيانات التي يحتاجها، مثل أرقام الهواتف أو عناوين البريد الإلكتروني، ويختار المستخدم جهات اتصال معيّنة لمشاركتها. يمنح هذا الإذن تطبيقك إذن الوصول للقراءة إلى البيانات المحدّدة فقط، ما يضمن التحكّم الدقيق مع توفير تجربة مستخدم متّسقة تتضمّن إمكانات البحث المضمّنة والتبديل بين الملفات الشخصية والاختيار المتعدّد بدون الحاجة إلى إنشاء واجهة المستخدم أو صيانتها.

لمزيد من المعلومات، اطّلِع على مستندات أداة اختيار جهات الاتصال.

الأمان

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

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

Android 高级保护模式为 Android 用户提供了一套强大的新安全功能,标志着在保护用户(尤其是面临较高风险的用户)免遭复杂攻击方面迈出了重要一步。AAPM 是一项选择启用功能,只需进行一项配置设置即可激活。用户可以随时启用该功能,以应用一套主观的安全保护措施。

这些核心配置包括:禁止安装未知来源的应用(旁加载)、限制 USB 数据信号传输,以及强制执行 Google Play 保护机制扫描,从而显著减小设备的攻击面。 开发者可以使用 AdvancedProtectionManager API 与此功能集成,以检测模式的状态,从而使应用能够在用户选择启用此模式时自动采用强化型安全姿态或限制高风险功能。

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

يتوافق نظام التشغيل Android الآن مع مخطّط توقيع حِزم APK المختلط لحماية هوية توقيع تطبيقك من التهديدات المحتملة للهجمات التي تستخدم الحوسبة الكمية. توفّر هذه الميزة مخطّطًا جديدًا لتوقيع حزمة APK، يتيح لك إقران مفتاح توقيع تقليدي (مثل RSA أو EC) بخوارزمية جديدة للتشفير بعد الكم (ML-DSA).

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

التأثير على المطوّرين

  • التطبيقات التي تستخدم ميزة "توقيع التطبيق" من Play: إذا كنت تستخدم ميزة "توقيع التطبيق" من Play، يمكنك الانتظار إلى أن يتيح لك Google Play خيار ترقية التوقيع المختلط باستخدام مفتاح PQC تم إنشاؤه من خلال Google Play، ما يضمن حماية تطبيقك بدون الحاجة إلى إدارة المفاتيح يدويًا.
  • التطبيقات التي تستخدم مفاتيح مُدارة ذاتيًا: يمكن للمطوّرين الذين يديرون مفاتيح التوقيع الخاصة بهم الاستفادة من أدوات إنشاء Android المعدَّلة (مثل apksigner) لتغيير المفتاح إلى هوية مختلطة تجمع بين مفتاح مقاوم للكمّ ومفتاح تقليدي جديد. (يجب إنشاء مفتاح كلاسيكي جديد، ولا يمكنك إعادة استخدام المفتاح القديم).

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

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

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

实现优化,使应用能够在低带宽卫星网络上有效运行。

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

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

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

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

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

التسليم (Handoff)

‫Handoff هي ميزة وواجهة برمجة تطبيقات جديدة ستتوفّر في Android 17، ويمكن لمطوّري التطبيقات دمجها لتوفير تجربة متواصلة للمستخدمين على جميع الأجهزة. تتيح هذه الميزة للمستخدم بدء نشاط تطبيق على أحد أجهزة Android ونقله إلى جهاز Android آخر. تعمل ميزة &quot;نقل النشاط&quot; في خلفية جهاز المستخدم، وتعرض الأنشطة المتاحة من أجهزة المستخدم الأخرى القريبة من خلال نقاط دخول مختلفة، مثل مشغّل التطبيقات وشريط المهام، على الجهاز المستلِم.

يمكن للتطبيقات تحديد ميزة &quot;نقل البيانات&quot; لتشغيل تطبيق Android الأصلي نفسه، إذا كان مثبّتًا ومتوفّرًا على الجهاز المستلِم. في مسار التنقّل من تطبيق إلى تطبيق، يتم توجيه المستخدم إلى النشاط المحدّد من خلال رابط لصفحة في التطبيق. يمكن بدلاً من ذلك توفير ميزة "التسليم من التطبيق إلى الويب" كخيار احتياطي أو تنفيذها مباشرةً باستخدام ميزة "التسليم من عنوان URL".

يتم تنفيذ ميزة Handoff على أساس كل نشاط. لتفعيل ميزة "التسليم"، استدعِ طريقة setHandoffEnabled() للنشاط. قد يلزم تمرير بيانات إضافية مع عملية التسليم حتى يتمكّن النشاط الذي تم إنشاؤه من جديد على الجهاز المستلِم من استعادة الحالة المناسبة. نفِّذ onHandoffActivityDataRequested() الدالة التنفيذية لعرض عنصر HandoffActivityData يحتوي على تفاصيل تحدّد كيفية تعامل ميزة "التسليم" مع النشاط وإعادة إنشائه على الجهاز المستلِم.

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

في نظام التشغيل 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();