برای اینکه برنامه خود را تا حد امکان کوچک و سریع کنید، باید نسخه انتشار خود را با isMinifyEnabled = true
بهینه سازی و کوچک کنید.
انجام این کار کوچک کردن را فعال می کند که کد و منابع استفاده نشده را حذف می کند. مبهم سازی ، که نام کلاس ها و اعضای برنامه شما را کوتاه می کند. و بهینه سازی ، که استراتژی های تهاجمی تری را برای کاهش بیشتر اندازه و بهبود عملکرد برنامه شما اعمال می کند. این صفحه توضیح می دهد که R8 چگونه این وظایف زمان کامپایل را برای پروژه شما انجام می دهد و چگونه می توانید آنها را سفارشی کنید.
وقتی پروژه خود را با استفاده از افزونه Android Gradle نسخه 3.4.0 یا بالاتر میسازید، افزونه دیگر از ProGuard برای بهینهسازی کد در زمان کامپایل استفاده نمیکند. در عوض، این افزونه با کامپایلر R8 کار می کند تا وظایف زمان کامپایل زیر را انجام دهد:
- کوچک کردن کد (یا تکان دادن درخت): کلاسها، فیلدها، روشها و ویژگیهای استفاده نشده را از برنامه شما و وابستگیهای کتابخانهاش شناسایی و با خیال راحت حذف میکند (که آن را به ابزاری ارزشمند برای کار در حدود 64 هزار مرجع تبدیل میکند). برای مثال، اگر فقط از چند API وابسته به کتابخانه استفاده میکنید، کوچک کردن میتواند کد کتابخانهای را که برنامه شما از آن استفاده نمیکند شناسایی کند و فقط آن کد را از برنامه شما حذف کند. برای کسب اطلاعات بیشتر، به بخش نحوه کوچک کردن کد خود بروید.
- کاهش منابع: منابع استفاده نشده را از برنامه بسته بندی شده شما حذف می کند، از جمله منابع استفاده نشده در وابستگی های کتابخانه برنامه شما. این در ارتباط با کوچک کردن کد کار می کند به طوری که پس از حذف کدهای استفاده نشده، هر منبعی که دیگر به آن ارجاع داده نمی شود، می تواند با خیال راحت حذف شود. برای کسب اطلاعات بیشتر، به بخش نحوه کاهش منابع خود بروید.
- بهینه سازی: کد شما را بازرسی و بازنویسی می کند تا عملکرد زمان اجرا را بهبود بخشد و حجم فایل های DEX برنامه شما را کاهش دهد. این عملکرد زمان اجرا کد را تا 30 درصد بهبود می بخشد و به شدت راه اندازی و زمان بندی فریم را بهبود می بخشد. برای مثال، اگر R8 تشخیص دهد که شاخه
else {}
برای یک عبارت if/else داده شده هرگز گرفته نشده است، R8 کد شاخهelse {}
را حذف می کند. برای کسب اطلاعات بیشتر، به بخش بهینه سازی کد بروید. - مبهم سازی (یا کوچک سازی شناسه): نام کلاس ها و اعضا را کوتاه می کند که منجر به کاهش اندازه فایل DEX می شود. برای کسب اطلاعات بیشتر، به بخش نحوه مبهم کردن کد خود بروید.
هنگام ساخت نسخه انتشار برنامه خود، R8 می تواند پیکربندی شود تا وظایف زمان کامپایل که در بالا توضیح داده شد را برای شما انجام دهد. همچنین میتوانید وظایف خاصی را غیرفعال کنید یا رفتار R8 را از طریق فایلهای قوانین ProGuard سفارشی کنید. در واقع، R8 با تمام فایلهای قوانین ProGuard موجود شما کار میکند ، بنابراین بهروزرسانی پلاگین Android Gradle برای استفاده از R8 نباید شما را ملزم به تغییر قوانین موجود خود کند.
کوچک کردن، مبهم سازی و بهینه سازی را فعال کنید
هنگامی که از Android Studio 3.4 یا Android Gradle 3.4.0 و بالاتر استفاده می کنید، R8 کامپایلر پیش فرضی است که بایت کد جاوا پروژه شما را به فرمت DEX که روی پلتفرم اندروید اجرا می شود تبدیل می کند. با این حال، هنگامی که یک پروژه جدید با استفاده از Android Studio ایجاد می کنید، بهینه سازی کد، کوچک شدن، مبهم سازی و کد به طور پیش فرض فعال نیست. دلیلش این است که این بهینهسازیهای زمان کامپایل زمان ساخت پروژه شما را افزایش میدهند و در صورتی که کد مورد نظر را به اندازه کافی سفارشی نکنید، ممکن است اشکالاتی را ایجاد کنند.
بنابراین، بهتر است هنگام ساختن نسخه نهایی برنامه خود که قبل از انتشار آزمایش می کنید، این وظایف زمان کامپایل را فعال کنید. برای فعال کردن کوچک کردن، مبهم سازی و بهینه سازی، موارد زیر را در اسکریپت ساخت سطح پروژه خود بگنجانید.
android { buildTypes { getByName("release") { // Enables code shrinking, obfuscation, and optimization for only // your project's release build type. Make sure to use a build // variant with `isDebuggable=false`. isMinifyEnabled = true // Enables resource shrinking, which is performed by the // Android Gradle plugin. isShrinkResources = true proguardFiles( // Includes the default ProGuard rules files that are packaged with // the Android Gradle plugin. To learn more, go to the section about // R8 configuration files. getDefaultProguardFile("proguard-android-optimize.txt"), // Includes a local, custom Proguard rules file "proguard-rules.pro" ) } } ... }
android { buildTypes { release { // Enables code shrinking, obfuscation, and optimization for only // your project's release build type. Make sure to use a build // variant with `debuggable false`. minifyEnabled true // Enables resource shrinking, which is performed by the // Android Gradle plugin. shrinkResources true // Includes the default ProGuard rules files that are packaged with // the Android Gradle plugin. To learn more, go to the section about // R8 configuration files. proguardFiles getDefaultProguardFile( 'proguard-android-optimize.txt'), 'proguard-rules.pro' } } ... }
فایل های پیکربندی R8
R8 از فایلهای قوانین ProGuard استفاده میکند تا رفتار پیشفرض خود را اصلاح کند و ساختار برنامهتان را بهتر بشناسد، مانند کلاسهایی که به عنوان نقاط ورودی به کد برنامه شما عمل میکنند. اگرچه میتوانید برخی از این فایلهای قوانین را تغییر دهید، برخی از قوانین ممکن است بهطور خودکار توسط ابزارهای زمان کامپایل مانند AAPT2 ایجاد شوند یا از وابستگیهای کتابخانه برنامه شما به ارث برده شوند. جدول زیر منابع فایل های قوانین ProGuard را که R8 استفاده می کند، توضیح می دهد.
منبع | مکان | توضیحات |
اندروید استودیو | <module-dir>/proguard-rules.pro | هنگامی که یک ماژول جدید با استفاده از Android Studio ایجاد می کنید، IDE یک فایل proguard-rules.pro را در فهرست اصلی آن ماژول ایجاد می کند.به طور پیش فرض، این فایل هیچ قانونی را اعمال نمی کند. بنابراین، قوانین ProGuard خود را در اینجا وارد کنید، مانند قوانین حفظ سفارشی خود. |
پلاگین اندروید Gradle | در زمان کامپایل توسط افزونه Android Gradle ایجاد شده است. | افزونه Android Gradle proguard-android-optimize.txt را ایجاد می کند که شامل قوانینی است که برای اکثر پروژه های Android مفید است و حاشیه نویسی @Keep* را فعال می کند.به طور پیش فرض، هنگام ایجاد یک ماژول جدید با استفاده از اندروید استودیو، اسکریپت ساخت سطح ماژول شامل این فایل قوانین در بیلد انتشار شما برای شما می شود. توجه: افزونه Android Gradle شامل فایلهای قوانین ProGuard از پیش تعریفشده اضافی است، اما توصیه میشود از |
وابستگی های کتابخانه | در یک کتابخانه AAR: در یک کتابخانه JAR: علاوه بر این مکانها، افزونه Android Gradle نسخه 3.6 یا بالاتر نیز از قوانین کوچک کردن هدفمند پشتیبانی میکند. | اگر یک کتابخانه AAR یا JAR با فایل قوانین خاص خود منتشر شود و شما آن کتابخانه را به عنوان یک وابستگی زمان کامپایل قرار دهید، R8 به طور خودکار آن قوانین را هنگام کامپایل پروژه شما اعمال می کند. علاوه بر قوانین معمولی ProGuard، افزونه Android Gradle نسخه 3.6 یا بالاتر نیز از قوانین کوچک کردن هدفمند پشتیبانی می کند. اینها قوانینی هستند که کوچک کننده های خاص (R8 یا ProGuard) و همچنین نسخه های کوچک کننده خاص را هدف قرار می دهند. استفاده از فایلهای قوانین که با کتابخانهها بستهبندی شدهاند، در صورتی مفید است که قوانین خاصی برای عملکرد صحیح کتابخانه مورد نیاز باشد - یعنی توسعهدهنده کتابخانه مراحل عیبیابی را برای شما انجام داده است. با این حال، باید توجه داشته باشید که از آنجایی که قوانین افزودنی هستند، برخی از قوانینی که وابستگی کتابخانه شامل آن می شود را نمی توان حذف کرد و ممکن است بر گردآوری قسمت های دیگر برنامه شما تأثیر بگذارد. به عنوان مثال، اگر یک کتابخانه شامل یک قانون برای غیرفعال کردن بهینه سازی کد باشد، آن قانون بهینه سازی را برای کل پروژه شما غیرفعال می کند. |
Android Asset Package Tool 2 (AAPT2) | پس از ساختن پروژه خود با minifyEnabled true : <module-dir>/build/intermediates/aapt_proguard_file/.../aapt_rules.txt | AAPT2 بر اساس ارجاع به کلاسها در مانیفست، طرحبندیها و سایر منابع برنامه، قوانین حفظ را ایجاد میکند. به عنوان مثال، AAPT2 برای هر فعالیتی که در مانیفست برنامه خود به عنوان نقطه ورودی ثبت می کنید، یک قانون نگه داشتن دارد. |
فایل های پیکربندی سفارشی | بهطور پیشفرض، وقتی یک ماژول جدید با استفاده از Android Studio ایجاد میکنید، IDE <module-dir>/proguard-rules.pro را برای شما ایجاد میکند تا قوانین خود را اضافه کنید. | می توانید پیکربندی های اضافی اضافه کنید و R8 آنها را در زمان کامپایل اعمال می کند. |
هنگامی که ویژگی minifyEnabled
را روی true
تنظیم می کنید، R8 قوانین را از تمام منابع موجود لیست شده در بالا ترکیب می کند. این مهم است که هنگام عیبیابی با R8 به خاطر داشته باشید، زیرا وابستگیهای زمان کامپایل دیگر، مانند وابستگیهای کتابخانه، ممکن است تغییراتی را در رفتار R8 ایجاد کنند که شما از آن اطلاعی ندارید.
برای خروجی یک گزارش کامل از تمام قوانینی که R8 هنگام ساخت پروژه خود اعمال می کند، موارد زیر را در فایل proguard-rules.pro
ماژول خود قرار دهید:
// You can specify any path and filename.
-printconfiguration ~/tmp/full-r8-config.txt
قوانین کوچک کردن هدفمند
افزونه Android Gradle نسخه 3.6 یا بالاتر از قوانین کتابخانههایی پشتیبانی میکند که کوچککنندههای خاص (R8 یا ProGuard) و همچنین نسخههای کوچککننده خاص را هدف قرار میدهند. این به توسعهدهندگان کتابخانه اجازه میدهد تا قوانین خود را برای کار بهینه در پروژههایی که از نسخههای کوچککننده جدید استفاده میکنند، تنظیم کنند، در حالی که اجازه میدهد قوانین موجود همچنان در پروژههایی با نسخههای کوچککننده قدیمیتر استفاده شوند.
برای تعیین قوانین کوچک کردن هدفمند، توسعه دهندگان کتابخانه باید آنها را در مکان های خاصی در داخل یک کتابخانه AAR یا JAR قرار دهند، همانطور که در زیر توضیح داده شده است.
In an AAR library:
proguard.txt (legacy location)
classes.jar
└── META-INF
└── com.android.tools (targeted shrink rules location)
├── r8-from-<X>-upto-<Y>/<R8-rules-file>
└── proguard-from-<X>-upto-<Y>/<ProGuard-rules-file>
In a JAR library:
META-INF
├── proguard/<ProGuard-rules-file> (legacy location)
└── com.android.tools (targeted shrink rules location)
├── r8-from-<X>-upto-<Y>/<R8-rules-file>
└── proguard-from-<X>-upto-<Y>/<ProGuard-rules-file>
این بدان معناست که قوانین کوچک کردن هدفمند در فهرست META-INF/com.android.tools
یک JAR یا در فهرست META-INF/com.android.tools
داخل classes.jar
یک AAR ذخیره میشوند.
در زیر آن دایرکتوری، میتوان چندین دایرکتوری با نامهایی به شکل r8-from-<X>-upto-<Y>
یا proguard-from-<X>-upto-<Y>
وجود داشته باشد تا مشخص شود کدام نسخههای کوچککننده قوانین داخل دایرکتوری ها برای آنها نوشته شده است. توجه داشته باشید که قسمت های -from-<X>
و -upto-<Y>
اختیاری هستند، نسخه <Y>
انحصاری است و محدوده های نسخه باید پیوسته باشند.
برای مثال، r8-upto-8.0.0
، r8-from-8.0.0-upto-8.2.0
و r8-from-8.2.0
مجموعه معتبری از قوانین کوچک کردن هدفمند را تشکیل می دهند. قوانین تحت پوشه r8-from-8.0.0-upto-8.2.0
توسط R8 از نسخه 8.0.0 تا نسخه 8.0.0 استفاده می شود، اما شامل نسخه 8.2.0 نمی شود .
با توجه به این اطلاعات، افزونه Android Gradle نسخه 3.6 یا بالاتر، قوانین را از فهرست های R8 منطبق انتخاب می کند. اگر یک کتابخانه قوانین کوچک کردن هدفمند را مشخص نکند، افزونه Android Gradle قوانین را از مکانهای قدیمی انتخاب میکند ( proguard.txt
برای AAR یا META-INF/proguard/<ProGuard-rules-file>
برای JAR).
توسعهدهندگان کتابخانه میتوانند برای حفظ سازگاری با افزونه Android Gradle قدیمیتر از 3.6 یا ابزارهای دیگر، قوانین هدفمند کوچک کردن یا قوانین ProGuard قدیمی را در کتابخانههای خود یا هر دو نوع را در کتابخانههای خود بگنجانند.
شامل تنظیمات اضافی
هنگامی که یک پروژه یا ماژول جدید با استفاده از Android Studio ایجاد می کنید، IDE یک فایل <module-dir>/proguard-rules.pro
را برای شما ایجاد می کند تا قوانین خود را در آن لحاظ کنید. همچنین میتوانید با افزودن آنها به ویژگی proguardFiles
در اسکریپت ساخت ماژول، قوانین اضافی را از فایلهای دیگر اضافه کنید.
به عنوان مثال، میتوانید با افزودن یک ویژگی proguardFiles
دیگر در بلوک productFlavor
مربوطه، قوانینی را اضافه کنید که مختص هر نوع ساخت هستند. فایل Gradle زیر flavor2-rules.pro
را به طعم محصول flavor2
اضافه می کند. اکنون، flavor2
از هر سه قانون ProGuard استفاده میکند، زیرا قوانین مربوط به بلوک release
نیز اعمال میشوند.
علاوه بر این، میتوانید ویژگی testProguardFiles
را اضافه کنید که فهرستی از فایلهای ProGuard را که فقط در APK آزمایشی گنجانده شدهاند را مشخص میکند:
android { ... buildTypes { getByName("release") { isMinifyEnabled = true proguardFiles( getDefaultProguardFile("proguard-android-optimize.txt"), // List additional ProGuard rules for the given build type here. By default, // Android Studio creates and includes an empty rules file for you (located // at the root directory of each module). "proguard-rules.pro" ) testProguardFiles( // The proguard files listed here are included in the // test APK only. "test-proguard-rules.pro" ) } } flavorDimensions.add("version") productFlavors { create("flavor1") { ... } create("flavor2") { proguardFile("flavor2-rules.pro") } } }
android { ... buildTypes { release { minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), // List additional ProGuard rules for the given build type here. By default, // Android Studio creates and includes an empty rules file for you (located // at the root directory of each module). 'proguard-rules.pro' testProguardFiles // The proguard files listed here are included in the // test APK only. 'test-proguard-rules.pro' } } flavorDimensions "version" productFlavors { flavor1 { ... } flavor2 { proguardFile 'flavor2-rules.pro' } } }
کد خود را کوچک کنید
هنگامی که ویژگی minifyEnabled
روی true
تنظیم می کنید، کوچک شدن کد با R8 به طور پیش فرض فعال می شود.
کوچک کردن کد (همچنین به عنوان تکان دادن درخت شناخته می شود)، فرآیند حذف کدی است که R8 تشخیص می دهد در زمان اجرا لازم نیست. اگر برای مثال، برنامه شما دارای وابستگی های کتابخانه ای زیادی باشد اما تنها از بخش کوچکی از عملکرد آنها استفاده کند، این فرآیند می تواند اندازه برنامه شما را تا حد زیادی کاهش دهد.
برای کوچک کردن کد برنامه، R8 ابتدا تمام نقاط ورود به کد برنامه شما را بر اساس مجموعه فایلهای پیکربندی ترکیبی تعیین میکند. این نقاط ورودی شامل تمام کلاسهایی است که پلتفرم Android ممکن است برای باز کردن فعالیتها یا خدمات برنامه شما از آنها استفاده کند. با شروع از هر نقطه ورودی، R8 کد برنامه شما را بررسی می کند تا نموداری از همه روش ها، متغیرهای عضو و سایر کلاس هایی که برنامه شما ممکن است در زمان اجرا به آنها دسترسی داشته باشد بسازد. کدی که به آن نمودار متصل نیست غیرقابل دسترس تلقی می شود و ممکن است از برنامه حذف شود.
شکل 1 یک برنامه با وابستگی به کتابخانه زمان اجرا را نشان می دهد. در حین بررسی کد برنامه، R8 تشخیص می دهد که متدهای foo()
, faz()
و bar()
از نقطه ورودی MainActivity.class
قابل دسترسی هستند. با این حال، کلاس OkayApi.class
یا متد baz()
هرگز توسط برنامه شما در زمان اجرا استفاده نمی شود و R8 آن کد را هنگام کوچک کردن برنامه شما حذف می کند.
R8 نقاط ورودی را از طریق قوانین -keep
در فایل های پیکربندی R8 پروژه تعیین می کند. به این معنی که قوانین حفظ، کلاسهایی را مشخص میکنند که R8 نباید هنگام کوچک کردن برنامهتان آنها را کنار بگذارد و R8 آن کلاسها را به عنوان نقاط ورود احتمالی به برنامه شما در نظر میگیرد. افزونه Android Gradle و AAPT2 بهطور خودکار قوانین نگهداری را ایجاد میکنند که اکثر پروژههای برنامه برای شما مورد نیاز هستند، مانند فعالیتها، بازدیدها و خدمات برنامهتان. با این حال، اگر میخواهید این رفتار پیشفرض را با قوانین نگهداری اضافی سفارشی کنید، بخش نحوه سفارشیسازی کد نگهداری را بخوانید.
اگر در عوض فقط به کاهش حجم منابع برنامه خود علاقه دارید، به بخش نحوه کوچک کردن منابع خود بروید.
توجه داشته باشید که اگر یک پروژه کتابخانه کوچک شود، برنامهای که به آن کتابخانه وابسته است شامل کلاسهای کتابخانه کوچک شده است. اگر کلاسهایی در APK کتابخانه وجود ندارد، ممکن است لازم باشد قوانین نگهداری کتابخانه را تنظیم کنید. اگر در حال ساخت و انتشار یک کتابخانه با فرمت AAR هستید، فایلهای JAR محلی که کتابخانه شما به آن وابسته است، در فایل AAR کوچک نمیشوند .
کدی را که باید حفظ شود سفارشی کنید
برای اکثر مواقع، فایل قوانین پیشفرض ProGuard ( proguard-android-optimize.txt
) برای R8 کافی است تا فقط کدهای استفاده نشده را حذف کند. با این حال، تجزیه و تحلیل درست برخی از موقعیتها برای R8 دشوار است و ممکن است کدهایی را که برنامه شما واقعاً به آن نیاز دارد حذف کند. چند نمونه از مواردی که ممکن است کد را به اشتباه حذف کند عبارتند از:
- وقتی برنامه شما روشی را از رابط بومی جاوا (JNI) فراخوانی میکند.
- وقتی برنامه شما کد را در زمان اجرا جستجو می کند (مانند بازتاب)
آزمایش برنامه شما باید خطاهای ناشی از کد حذف نامناسب را نشان دهد، اما همچنین میتوانید کد حذف شده را با ایجاد گزارش کد حذف شده بررسی کنید.
برای رفع خطاها و مجبور کردن R8 به نگه داشتن کد خاص، یک خط -keep
را در فایل قوانین ProGuard اضافه کنید. به عنوان مثال:
-keep public class MyClass
از طرف دیگر، میتوانید حاشیهنویسی @Keep
به کدی که میخواهید نگه دارید اضافه کنید. افزودن @Keep
به یک کلاس کل کلاس را همانطور که هست نگه می دارد. افزودن آن به یک متد یا فیلد، متد/فیلد (و نام آن) و همچنین نام کلاس را دست نخورده نگه میدارد. توجه داشته باشید که این حاشیهنویسی فقط در صورت استفاده از کتابخانه Annotations AndroidX و زمانی که فایل قوانین ProGuard را که با افزونه Android Gradle بستهبندی شده است، در دسترس قرار میدهید، همانطور که در بخش نحوه فعال کردن کوچک کردن توضیح داده شده است.
هنگام استفاده از گزینه -keep
باید ملاحظات زیادی را رعایت کنید. برای اطلاعات بیشتر در مورد سفارشی کردن فایل قوانین خود، راهنمای ProGuard را بخوانید. بخش عیبیابی سایر مشکلات رایجی را که ممکن است هنگام حذف کدتان با آن مواجه شوید، تشریح میکند.
کتابخانه های بومی را حذف کنید
بهطور پیشفرض، کتابخانههای کد بومی در نسخههای انتشار برنامه شما حذف میشوند. این حذف شامل حذف جدول نمادها و اشکال زدایی اطلاعات موجود در کتابخانه های بومی مورد استفاده برنامه شما است. حذف کتابخانه های کد بومی منجر به صرفه جویی قابل توجهی در اندازه می شود. با این حال، تشخیص خرابی در کنسول Google Play به دلیل اطلاعات از دست رفته (مانند نام کلاس و عملکرد) غیرممکن است.
پشتیبانی از خرابی بومی
کنسول Google Play خرابی های بومی تحت حیاتی اندروید را گزارش می کند. با چند مرحله، می توانید یک فایل نمادهای اشکال زدایی بومی را برای برنامه خود ایجاد و آپلود کنید. این فایل ردیابی پشته تصادف بومی نمادین (شامل نام کلاس و توابع) را در حیاتی اندروید فعال میکند تا به شما کمک کند برنامه خود را در مرحله تولید اشکالزدایی کنید. این مراحل بسته به نسخه پلاگین Android Gradle مورد استفاده در پروژه شما و خروجی ساخت پروژه شما متفاوت است.
پلاگین Android Gradle نسخه 4.1 یا بالاتر
اگر پروژه شما یک Android App Bundle می سازد، می توانید به طور خودکار فایل نمادهای اشکال زدایی بومی را در آن قرار دهید. برای گنجاندن این فایل در نسخههای انتشار، موارد زیر را به فایل build.gradle.kts
برنامه خود اضافه کنید:
android.buildTypes.release.ndk.debugSymbolLevel = { SYMBOL_TABLE | FULL }
سطح نماد اشکال زدایی را از موارد زیر انتخاب کنید:
- از
SYMBOL_TABLE
برای دریافت نام توابع در ردپای نمادین پشته کنسول Play استفاده کنید. این سطح از سنگ قبرها پشتیبانی می کند. -
FULL
برای دریافت نام توابع، فایلها و شماره خطوط در ردیابی پشته نمادین کنسول Play استفاده کنید.
اگر پروژه شما یک APK میسازد، از تنظیمات ساخت build.gradle.kts
که قبلاً نشان داده شده بود استفاده کنید تا فایل نمادهای اشکال زدایی بومی را جداگانه ایجاد کنید. فایل نمادهای اشکال زدایی بومی را به صورت دستی در کنسول Google Play آپلود کنید . به عنوان بخشی از فرآیند ساخت، افزونه Android Gradle این فایل را در محل پروژه زیر خروجی میدهد:
app/build/outputs/native-debug-symbols/ variant-name /native-debug-symbols.zip
پلاگین Android Gradle نسخه 4.0 یا بالاتر (و سایر سیستم های ساخت)
به عنوان بخشی از فرآیند ساخت، پلاگین Android Gradle یک کپی از کتابخانههای جدا نشده را در فهرست پروژه نگه میدارد. این ساختار دایرکتوری شبیه به زیر است:
app/build/intermediates/cmake/universal/release/obj/
├── armeabi-v7a/
│ ├── libgameengine.so
│ ├── libothercode.so
│ └── libvideocodec.so
├── arm64-v8a/
│ ├── libgameengine.so
│ ├── libothercode.so
│ └── libvideocodec.so
├── x86/
│ ├── libgameengine.so
│ ├── libothercode.so
│ └── libvideocodec.so
└── x86_64/
├── libgameengine.so
├── libothercode.so
└── libvideocodec.so
محتویات این دایرکتوری را فشرده کنید:
cd app/build/intermediates/cmake/universal/release/obj
zip -r symbols.zip .
فایل
symbols.zip
را به صورت دستی در کنسول Google Play آپلود کنید .
منابع خود را کوچک کنید
کوچک کردن منابع فقط در رابطه با کوچک کردن کد کار می کند. پس از حذف کدهای کوچککننده، همه کدهای استفاده نشده را حذف میکند، کوچککننده منابع میتواند شناسایی کند که برنامه هنوز از چه منابعی استفاده میکند. این امر مخصوصاً زمانی صادق است که شما کتابخانههای کدی را که شامل منابع هستند اضافه میکنید—شما باید کد کتابخانه استفادهنشده را حذف کنید تا منابع کتابخانه بیارجاع شوند و در نتیجه توسط کوچککننده منبع قابل حذف باشند.
برای فعال کردن کوچک شدن منابع، ویژگی shrinkResources
را در اسکریپت ساخت خود روی true
تنظیم کنید (در کنار minifyEnabled
برای کوچک کردن کد). به عنوان مثال:
android { ... buildTypes { getByName("release") { isShrinkResources = true isMinifyEnabled = true proguardFiles( getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro" ) } } }
android { ... buildTypes { release { shrinkResources true minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } }
اگر قبلاً برنامه خود را با استفاده از minifyEnabled
برای کوچک کردن کد نساختهاید، قبل از فعال کردن shrinkResources
آن را امتحان کنید، زیرا ممکن است لازم باشد فایل proguard-rules.pro
خود را ویرایش کنید تا کلاسها یا روشهایی را که ایجاد یا فراخوانی میشوند به صورت پویا قبل از شروع حذف نگه دارید. منابع
شخصی سازی کنید که کدام منابع را نگه دارید
اگر منابع خاصی وجود دارد که میخواهید آن را نگه دارید یا کنار بگذارید، یک فایل XML در پروژه خود با تگ <resources>
ایجاد کنید و هر منبع را در ویژگی tools:keep
و هر منبع را در ویژگی tools:discard
مشخص کنید. هر دو ویژگی فهرستی از نام منابع را میپذیرند که با کاما جدا شدهاند. می توانید از کاراکتر ستاره به عنوان کارت وحشی استفاده کنید.
به عنوان مثال:
<?xml version="1.0" encoding="utf-8"?> <resources xmlns:tools="http://schemas.android.com/tools" tools:keep="@layout/l_used*_c,@layout/l_used_a,@layout/l_used_b*" tools:discard="@layout/unused2" />
این فایل را در منابع پروژه خود ذخیره کنید، به عنوان مثال، در res/raw/my.package.keep.xml
. بیلد این فایل را در برنامه شما بسته بندی نمی کند.
توجه: حتما از یک نام منحصر به فرد برای فایل keep
استفاده کنید. وقتی کتابخانههای مختلف با هم مرتبط میشوند، قوانین نگهداری آنها در غیر این صورت با هم تضاد میکنند، و باعث ایجاد مشکلات بالقوه با قوانین نادیده گرفته شده یا منابع نگهداری غیر ضروری میشود.
مشخص کردن منابعی که باید حذف شوند ممکن است احمقانه به نظر برسد، اما این می تواند هنگام استفاده از انواع ساخت مفید باشد. به عنوان مثال، ممکن است تمام منابع خود را در دایرکتوری مشترک پروژه قرار دهید، سپس یک فایل my.package.build.variant.keep.xml
متفاوت برای هر نوع ساخت ایجاد کنید، زمانی که می دانید به نظر می رسد یک منبع داده شده در کد استفاده می شود (و بنابراین توسط شرینکر حذف نمی شود) اما می دانید که در واقع برای نوع ساخت داده شده استفاده نخواهد شد. همچنین این امکان وجود دارد که ابزارهای ساخت به اشتباه منبع مورد نیاز را شناسایی کرده باشند، که این امکان وجود دارد زیرا کامپایلر شناسه های منبع را به صورت درون خطی اضافه می کند و سپس تحلیلگر منبع ممکن است تفاوت بین یک منبع ارجاع شده واقعی و یک مقدار صحیح در کد را که اتفاق می افتد نداند. ارزش یکسانی دارند
بررسی های مرجع دقیق را فعال کنید
به طور معمول، کوچک کننده منبع می تواند به دقت تعیین کند که آیا از یک منبع استفاده شده است یا خیر. با این حال، اگر کد شما با Resources.getIdentifier()
تماس بگیرد (یا اگر هر یک از کتابخانه های شما این کار را انجام دهد—کتابخانه AppCompat این کار را می کند)، به این معنی است که کد شما نام منابع را بر اساس رشته های تولید شده به صورت پویا جستجو می کند. وقتی این کار را انجام میدهید، کوچککننده منبع بهطور پیشفرض رفتار تدافعی دارد و همه منابع را با قالب نام منطبق بهعنوان بالقوه استفاده شده و غیرقابل حذف علامتگذاری میکند.
به عنوان مثال، کد زیر باعث می شود که تمام منابع با پیشوند img_
به عنوان استفاده شده علامت گذاری شوند.
val name = String.format("img_%1d", angle + 1) val res = resources.getIdentifier(name, "drawable", packageName)
String name = String.format("img_%1d", angle + 1); res = getResources().getIdentifier(name, "drawable", getPackageName());
کوچککننده منبع همچنین تمام ثابتهای رشتهای در کد شما و همچنین منابع مختلف res/raw/
را بررسی میکند و به دنبال URLهای منبع در قالبی مشابه file:///android_res/drawable//ic_plus_anim_016.png
میگردد. اگر رشتههایی مانند این یا رشتههای دیگری را پیدا کند که به نظر میرسد میتوان از آنها برای ساخت نشانیهای اینترنتی مانند این استفاده کرد، آنها را حذف نمیکند.
اینها نمونه هایی از حالت انقباض ایمن هستند که به طور پیش فرض فعال است. با این حال، میتوانید این مدیریت «ایمنتر از متأسفانه» را خاموش کنید و مشخص کنید که کوچککننده منابع فقط منابعی را نگه میدارد که مطمئن است استفاده میشوند. برای انجام این کار، shrinkMode
در فایل keep.xml
به صورت strict
گیر کنید:
<?xml version="1.0" encoding="utf-8"?> <resources xmlns:tools="http://schemas.android.com/tools" tools:shrinkMode="strict" />
اگر حالت انقباض دقیق را فعال کنید و کد شما همچنین به منابعی با رشته های تولید شده به صورت پویا ارجاع می دهد، همانطور که در بالا نشان داده شده است، باید به صورت دستی آن منابع را با استفاده از ویژگی tools:keep
نگه دارید.
منابع جایگزین استفاده نشده را حذف کنید
کوچککننده منابع Gradle فقط منابعی را حذف میکند که کد برنامه شما به آنها ارجاع نمیدهد، به این معنی که منابع جایگزین را برای پیکربندیهای مختلف دستگاه حذف نمیکند. در صورت لزوم، میتوانید از ویژگی resConfigs
افزونه Android Gradle برای حذف فایلهای منبع جایگزینی که برنامه شما به آن نیازی ندارد، استفاده کنید.
برای مثال، اگر از کتابخانهای استفاده میکنید که شامل منابع زبانی است (مانند AppCompat یا Google Play Services)، برنامه شما شامل تمام رشتههای زبان ترجمهشده برای پیامهای موجود در آن کتابخانهها میشود، چه بقیه برنامه شما به همان زبانها ترجمه شده باشد یا نه اگر میخواهید فقط زبانهایی را نگه دارید که برنامه شما به طور رسمی از آنها پشتیبانی میکند، میتوانید آن زبانها را با استفاده از ویژگی resConfig
مشخص کنید. هر منبعی برای زبان های مشخص نشده حذف می شود.
قطعه زیر نشان می دهد که چگونه منابع زبان خود را فقط به انگلیسی و فرانسوی محدود کنید:
android { defaultConfig { ... resourceConfigurations.addAll(listOf("en", "fr")) } }
android { defaultConfig { ... resConfigs "en", "fr" } }
هنگام انتشار یک برنامه با استفاده از قالب Android App Bundle، به طور پیشفرض فقط زبانهای پیکربندی شده در دستگاه کاربر هنگام نصب برنامه دانلود میشوند. به طور مشابه، تنها منابعی که با تراکم صفحه دستگاه مطابقت دارند و کتابخانه های بومی مطابق با ABI دستگاه در دانلود گنجانده شده است. برای اطلاعات بیشتر به پیکربندی Android App Bundle مراجعه کنید.
برای برنامههای قدیمی که با APK منتشر میشوند (که قبل از آگوست 2021 ایجاد شدهاند)، میتوانید با ساختن چندین APK که هر کدام پیکربندی دستگاه متفاوتی را هدف قرار میدهند، تراکم صفحه یا منابع ABI را در APK خود تنظیم کنید.
ادغام منابع تکراری
بهطور پیشفرض، Gradle منابعی با نام یکسان را ادغام میکند. این رفتار توسط ویژگی shrinkResources
کنترل نمیشود و نمیتوان آن را غیرفعال کرد، زیرا لازم است وقتی چندین منبع با نامی که کد شما جستجو میشود مطابقت دارند از خطا جلوگیری شود.
ادغام منابع تنها زمانی اتفاق می افتد که دو یا چند فایل نام، نوع و واجد شرایط منبع یکسانی را به اشتراک بگذارند. Gradle انتخاب می کند که کدام فایل را بهترین انتخاب در بین موارد تکراری می داند (بر اساس ترتیب اولویت که در زیر توضیح داده شده است) و فقط آن یک منبع را برای توزیع در آرتیفکت نهایی به AAPT ارسال می کند.
Gradle به دنبال منابع تکراری در مکانهای زیر میگردد:
- منابع اصلی، مرتبط با مجموعه منبع اصلی، معمولاً در
src/main/res/
قرار دارند. - پوشش های مختلف، از نوع ساخت و طعم های ساخت.
- وابستگی های پروژه کتابخانه
Gradle منابع تکراری را به ترتیب اولویت آبشاری زیر ادغام می کند:
وابستگی ها → اصلی → طعم ساخت → نوع ساخت
به عنوان مثال، اگر یک منبع تکراری هم در منابع اصلی و هم در طعم ساخت ظاهر می شود، Gradle منبعی را در طعم ساخت انتخاب می کند.
اگر منابع یکسان در مجموعه منبع یکسان ظاهر شوند، Gradle نمی تواند آنها را ادغام کند و یک خطای ادغام منابع منتشر می کند. اگر چندین مجموعه منبع را در ویژگی sourceSet
فایل build.gradle.kts
خود تعریف کنید، ممکن است اتفاق بیفتد - برای مثال اگر هر دو src/main/res/
و src/main/res2/
دارای منابع یکسان باشند.
کد خود را مبهم کنید
هدف از مبهم سازی کاهش اندازه برنامه شما با کوتاه کردن نام کلاس ها، روش ها و فیلدهای برنامه شما است. مثال زیر نمونه ای از مبهم سازی با استفاده از R8 است:
androidx.appcompat.app.ActionBarDrawerToggle$DelegateProvider -> a.a.a.b:
androidx.appcompat.app.AlertController -> androidx.appcompat.app.AlertController:
android.content.Context mContext -> a
int mListItemLayout -> O
int mViewSpacingRight -> l
android.widget.Button mButtonNeutral -> w
int mMultiChoiceItemLayout -> M
boolean mShowTitle -> P
int mViewSpacingLeft -> j
int mButtonPanelSideLayout -> K
در حالی که مبهم سازی کد را از برنامه شما حذف نمی کند، صرفه جویی در اندازه قابل توجهی را می توان در برنامه هایی با فایل های DEX مشاهده کرد که بسیاری از کلاس ها، روش ها و فیلدها را فهرست می کنند. با این حال، از آنجایی که مبهم سازی قسمتهای مختلف کد شما را تغییر نام میدهد، وظایف خاصی مانند بررسی ردپای پشته، به ابزارهای اضافی نیاز دارد. برای درک stacktrace خود پس از مبهم کردن، بخش نحوه رمزگشایی یک stack trace مبهم را بخوانید.
بهعلاوه، اگر کد شما متکی به نامگذاری قابل پیشبینی برای روشها و کلاسهای برنامهتان است—مثلاً هنگام استفاده از بازتاب، باید آن امضاها را بهعنوان نقاط ورودی در نظر بگیرید و قوانین حفظ را برای آنها مشخص کنید، همانطور که در بخش نحوه سفارشی کردن کد توضیح داده شد. نگه داشتن . این قوانین حفظ به R8 می گویند که نه تنها آن کد را در DEX نهایی برنامه شما نگه دارد، بلکه نام اصلی خود را نیز حفظ کند.
رمزگشایی یک رد پشته مبهم
بعد از اینکه R8 کد شما را مبهم کرد، درک ردیابی پشته دشوار است (اگر غیرممکن نباشد) زیرا ممکن است نام کلاس ها و متدها تغییر کرده باشند. برای به دست آوردن ردیابی پشته اصلی باید ردیابی پشته را دوباره دنبال کنید.
بهینه سازی کد
به منظور بهینه سازی برنامه شما حتی بیشتر، R8 کد شما را در سطح عمیق تری بازرسی می کند تا کدهای استفاده نشده بیشتری را حذف کند یا، در صورت امکان، کد شما را بازنویسی کند تا کمتر پرمخاطب باشد. در زیر چند نمونه از این بهینه سازی ها آورده شده است:
- اگر کد شما هرگز شاخه
else {}
را برای یک عبارت if/else نمی گیرد، R8 ممکن است کد شعبهelse {}
را حذف کند. - اگر کد شما یک روش را فقط در چند مکان فراخوانی میکند، R8 ممکن است روش را حذف کرده و آن را در چند سایت تماس وارد کند.
- اگر R8 تشخیص دهد که یک کلاس فقط یک زیر کلاس منحصر به فرد دارد، و خود کلاس نمونه سازی نشده است (به عنوان مثال، یک کلاس پایه انتزاعی فقط توسط یک کلاس پیاده سازی مشخص استفاده می شود)، آنگاه R8 می تواند دو کلاس را ترکیب کرده و یک کلاس را از برنامه حذف کند. .
- برای کسب اطلاعات بیشتر، پست های وبلاگ بهینه سازی R8 توسط جیک وارتون را بخوانید.
R8 به شما اجازه نمی دهد که بهینه سازی های گسسته را غیرفعال یا فعال کنید، یا رفتار یک بهینه سازی را تغییر دهید. در واقع، R8 قوانین ProGuard را که سعی در اصلاح بهینهسازیهای پیشفرض دارند، مانند -optimizations
و -optimizationpasses
را نادیده میگیرد. این محدودیت مهم است زیرا با ادامه بهبود R8، حفظ یک رفتار استاندارد برای بهینهسازی به تیم اندروید استودیو کمک میکند تا به راحتی هر مشکلی را که ممکن است با آن مواجه شوید عیبیابی و حل کند.
توجه داشته باشید که فعال کردن بهینه سازی، ردیابی پشته را برای برنامه شما تغییر می دهد. برای مثال، درونسازی، فریمهای پشته را حذف میکند. برای یادگیری نحوه به دست آوردن ردیابی پشته اصلی به بخش ردیابی مجدد مراجعه کنید.
تاثیر بر عملکرد زمان اجرا
اگر کوچک کردن، مبهم سازی و بهینهسازی همگی فعال باشند، R8 عملکرد زمان اجرا کد (از جمله زمان راهاندازی و فریم در رشته رابط کاربر) را تا 30 درصد بهبود میبخشد. غیرفعال کردن هر یک از این موارد، مجموعه بهینهسازیهایی را که R8 استفاده میکند، به شدت محدود میکند.
اگر R8 فعال است، برای عملکرد راهاندازی بهتر نیز باید نمایههای راهاندازی ایجاد کنید .
بهینه سازی های تهاجمی تر را فعال کنید
R8 شامل مجموعه ای از بهینه سازی های اضافی است (که به آن "حالت کامل" گفته می شود) که باعث می شود رفتار متفاوتی با ProGuard داشته باشد. این بهینهسازیها بهطور پیشفرض از پلاگین Android Gradle نسخه 8.0.0 فعال هستند.
می توانید این بهینه سازی های اضافی را با قرار دادن موارد زیر در فایل gradle.properties
پروژه خود غیرفعال کنید:
android.enableR8.fullMode=false
از آنجایی که بهینهسازیهای اضافی باعث میشود R8 متفاوت از ProGuard رفتار کند، ممکن است از شما بخواهد که قوانین ProGuard اضافی را برای جلوگیری از مشکلات زمان اجرا در صورتی که از قوانین طراحیشده برای ProGuard استفاده میکنید، اضافه کنید. به عنوان مثال، بگویید که کد شما از طریق Java Reflection API به یک کلاس ارجاع می دهد. وقتی از "حالت کامل" استفاده نمیکنید ، R8 فرض میکند که شما قصد دارید اشیاء آن کلاس را در زمان اجرا بررسی و دستکاری کنید - حتی اگر کد شما واقعاً این کار را نمیکند - و به طور خودکار کلاس و مقدار اولیه استاتیک آن را نگه میدارد.
با این حال، هنگام استفاده از "حالت کامل"، R8 این فرض را نمیکند و اگر R8 ادعا کند که در غیر این صورت کد شما هرگز از کلاس در زمان اجرا استفاده نمیکند، کلاس را از DEX نهایی برنامه شما حذف میکند. یعنی اگر میخواهید کلاس و استاتیک اولیه آن را حفظ کنید، باید یک قانون keep را در فایل قوانین خود قرار دهید تا این کار را انجام دهید.
اگر هنگام استفاده از "حالت کامل" R8 با مشکلی مواجه شدید، برای راه حل احتمالی به صفحه پرسش و پاسخ R8 مراجعه کنید. اگر نمی توانید مشکل را حل کنید، لطفاً یک اشکال را گزارش کنید .
ردیابی مجدد stacktraces
کد پردازش شده توسط R8 به روشهای مختلفی تغییر میکند که میتواند درک ردپای پشته را سختتر کند، زیرا ردیابی پشته دقیقاً با کد منبع مطابقت ندارد. این می تواند در مورد تغییرات در شماره خطوط زمانی که اطلاعات اشکال زدایی نگهداری نمی شود. این می تواند به دلیل بهینه سازی هایی مانند inlining و outlining باشد. بزرگترین عامل مبهم سازی است که در آن حتی کلاس ها و متدها نیز نام ها را تغییر می دهند.
برای بازیابی پشته اصلی، R8 ابزار خط فرمان retrace را ارائه میکند که با بسته ابزار خط فرمان همراه است.
برای پشتیبانی از ردیابی مجدد ردیابی پشته برنامه خود، باید با افزودن قوانین زیر به فایل proguard-rules.pro
ماژول خود، اطمینان حاصل کنید که ساخت اطلاعات کافی برای ردیابی مجدد را حفظ می کند:
-keepattributes LineNumberTable,SourceFile
-renamesourcefileattribute SourceFile
ویژگی LineNumberTable
اطلاعات موقعیتی را در روش هایی حفظ می کند که این موقعیت ها در ردیابی پشته چاپ می شوند. ویژگی SourceFile
تضمین میکند که تمام زمانهای اجرا بالقوه واقعاً اطلاعات موقعیتی را چاپ میکنند. دستور -renamesourcefileattribute
نام فایل منبع در stack traces را فقط به SourceFile
تنظیم می کند. نام فایل منبع اصلی واقعی هنگام ردیابی مجدد لازم نیست زیرا فایل نگاشت حاوی فایل منبع اصلی است.
R8 هر بار که اجرا میشود یک فایل mapping.txt
ایجاد میکند که حاوی اطلاعات مورد نیاز برای ترسیم ردپای پشتهها به دنبالههای پشته اصلی است. Android Studio فایل را در فهرست <module-name> /build/outputs/mapping/ <build-type> /
ذخیره می کند.
هنگام انتشار برنامه خود در Google Play، می توانید فایل mapping.txt
را برای هر نسخه از برنامه خود آپلود کنید. هنگام انتشار با استفاده از بسته نرم افزاری Android، این فایل به طور خودکار به عنوان بخشی از محتوای بسته برنامه گنجانده می شود. سپس Google Play ردیابی پشته های ورودی از مشکلات گزارش شده توسط کاربر را مجدداً ردیابی می کند تا بتوانید آنها را در Play Console مرور کنید. برای اطلاعات بیشتر، به مقاله مرکز راهنمایی درباره نحوه رفع ابهام از ردیابی پشته خرابی مراجعه کنید.
عیب یابی با R8
این بخش برخی از استراتژیها را برای عیبیابی مشکلات هنگام فعال کردن کوچک کردن، مبهمسازی و بهینهسازی با استفاده از R8 توضیح میدهد. اگر در زیر راهحلی برای مشکل خود پیدا نکردید، صفحه پرسشهای متداول R8 و راهنمای عیبیابی ProGuard را نیز بخوانید.
گزارشی از کد حذف شده (یا نگهداری شده) ایجاد کنید
برای کمک به عیب یابی برخی از مشکلات R8، دیدن گزارشی از تمام کدهایی که R8 از برنامه شما حذف کرده است، ممکن است مفید باشد. برای هر ماژولی که میخواهید این گزارش را برای آن ایجاد کنید، -printusage <output-dir>/usage.txt
را به فایل قوانین سفارشی خود اضافه کنید. وقتی R8 را فعال میکنید و برنامه خود را میسازید، R8 گزارشی با مسیر و نام فایلی که شما مشخص کردهاید را خروجی میدهد. گزارش کد حذف شده شبیه به زیر است:
androidx.drawerlayout.R$attr
androidx.vectordrawable.R
androidx.appcompat.app.AppCompatDelegateImpl
public void setSupportActionBar(androidx.appcompat.widget.Toolbar)
public boolean hasWindowFeature(int)
public void setHandleNativeActionModesEnabled(boolean)
android.view.ViewGroup getSubDecor()
public void setLocalNightMode(int)
final androidx.appcompat.app.AppCompatDelegateImpl$AutoNightModeManager getAutoNightModeManager()
public final androidx.appcompat.app.ActionBarDrawerToggle$Delegate getDrawerToggleDelegate()
private static final boolean DEBUG
private static final java.lang.String KEY_LOCAL_NIGHT_MODE
static final java.lang.String EXCEPTION_HANDLER_MESSAGE_SUFFIX
...
اگر در عوض میخواهید گزارشی از نقاط ورودی که R8 از قوانین حفظ پروژه خود تعیین میکند، مشاهده کنید، -printseeds <output-dir>/seeds.txt
را در فایل قوانین سفارشی خود قرار دهید. وقتی R8 را فعال میکنید و برنامه خود را میسازید، R8 گزارشی با مسیر و نام فایلی که شما مشخص کردهاید را خروجی میدهد. گزارش نقاط ورودی نگهداری شده مشابه موارد زیر است:
com.example.myapplication.MainActivity
androidx.appcompat.R$layout: int abc_action_menu_item_layout
androidx.appcompat.R$attr: int activityChooserViewStyle
androidx.appcompat.R$styleable: int MenuItem_android_id
androidx.appcompat.R$styleable: int[] CoordinatorLayout_Layout
androidx.lifecycle.FullLifecycleObserverAdapter
...
عیب یابی کاهش منابع
وقتی منابع را کوچک می کنید، Build پنجره خلاصه ای از منابع حذف شده از برنامه را نشان می دهد. (باید ابتدا روی Toggle view کلیک کنید در سمت چپ پنجره برای نمایش خروجی متن دقیق از Gradle.) به عنوان مثال:
:android:shrinkDebugResources
Removed unused resources: Resource data reduced from 2570KB to 1711KB: Removed 33%
:android:validateDebugSigning
Gradle همچنین یک فایل تشخیصی به نام resources.txt
در <module-name>/build/outputs/mapping/release/
(همان پوشه فایل های خروجی ProGuard) ایجاد می کند. این فایل شامل جزئیاتی مانند اینکه کدام منابع به منابع دیگر ارجاع می دهند و کدام منابع استفاده شده یا حذف شده است.
برای مثال، برای اینکه بفهمید چرا @drawable/ic_plus_anim_016
هنوز در برنامه شما است، فایل resources.txt
را باز کنید و نام آن فایل را جستجو کنید. ممکن است متوجه شوید که از منبع دیگری ارجاع داده شده است، به شرح زیر:
16:25:48.005 [QUIET] [system.out] @drawable/add_schedule_fab_icon_anim : reachable=true
16:25:48.009 [QUIET] [system.out] @drawable/ic_plus_anim_016
اکنون باید بدانید که چرا @drawable/add_schedule_fab_icon_anim
قابل دسترسی است—و اگر به سمت بالا جستجو کنید میبینید که منبع در زیر «منابع قابل دسترسی ریشه هستند:» فهرست شده است. این بدان معناست که یک مرجع کد به add_schedule_fab_icon_anim
وجود دارد (یعنی شناسه R.drawable آن در کد قابل دسترسی پیدا شده است).
اگر از بررسی دقیق استفاده نمیکنید، شناسههای منبع را میتوان بهعنوان قابل دسترسی علامتگذاری کرد اگر ثابتهای رشتهای وجود داشته باشند که به نظر میرسد ممکن است برای ساخت نام منابع برای منابع بارگذاری شده پویا استفاده شوند. در این صورت، اگر خروجی ساخت را برای نام منبع جستجو کنید، ممکن است پیامی مانند این پیدا کنید:
10:32:50.590 [QUIET] [system.out] Marking drawable:ic_plus_anim_016:2130837506
used because it format-string matches string pool constant ic_plus_anim_%1$d.
اگر یکی از این رشته ها را می بینید و مطمئن هستید که از رشته برای بارگذاری دینامیک منبع داده شده استفاده نمی شود، می توانید از ویژگی tools:discard
برای اطلاع رسانی به سیستم ساخت برای حذف آن استفاده کنید، همانطور که در بخش نحوه انجام آن توضیح داده شد. شخصی سازی کنید که کدام منابع را نگه دارید .