ตัวอย่างต่อไปนี้อิงตามสถานการณ์ทั่วไปที่คุณใช้ R8 เพื่อเพิ่มประสิทธิภาพ แต่ต้องมีคำแนะนำขั้นสูงในการร่างกฎการเก็บรักษา
เงาสะท้อน
โดยทั่วไป เราไม่แนะนำให้ใช้การสะท้อนเพื่อประสิทธิภาพสูงสุด อย่างไรก็ตาม ในบางสถานการณ์ คุณอาจหลีกเลี่ยงไม่ได้ ตัวอย่างต่อไปนี้ให้คำแนะนำเกี่ยวกับกฎการเก็บรักษาในสถานการณ์ทั่วไปที่ใช้การสะท้อน
การสะท้อนด้วยคลาสที่โหลดตามชื่อ
ไลบรารีมักจะโหลดคลาสแบบไดนามิกโดยใช้ชื่อคลาสเป็น String
อย่างไรก็ตาม R8 จะตรวจหาคลาสที่โหลดด้วยวิธีนี้ไม่ได้ และอาจนำคลาสที่พิจารณาว่าไม่ได้ใช้ออก
ตัวอย่างเช่น ลองพิจารณาสถานการณ์ต่อไปนี้ที่คุณมีไลบรารีและแอปที่ใช้ไลบรารี โดยโค้ดจะแสดงตัวโหลดไลบรารีที่สร้างอินเทอร์เฟซ StartupTask ซึ่งแอปใช้
โค้ดไลบรารีมีดังนี้
// The interface for a task that runs once.
interface StartupTask {
fun run()
}
// The library object that loads and executes the task.
object TaskRunner {
fun execute(className: String) {
// R8 won't retain classes specified by this string value at runtime
val taskClass = Class.forName(className)
val task = taskClass.getDeclaredConstructor().newInstance() as StartupTask
task.run()
}
}
แอปที่ใช้ไลบรารีมีโค้ดดังนี้
// The app's task to pre-cache data.
// R8 will remove this class because it's only referenced by a string.
class PreCacheTask : StartupTask {
override fun run() {
// This log will never appear if the class is removed by R8.
Log.d("AppTask", "Warming up the cache...")
}
}
fun onCreate() {
// The library is told to run the app's task by its name.
TaskRunner.execute("com.example.app.PreCacheTask")
}
ในสถานการณ์นี้ ไลบรารีของคุณควรรวมไฟล์กฎการเก็บรักษาของผู้ใช้ที่มีกฎการเก็บรักษาต่อไปนี้
-keep class * implements com.example.library.StartupTask {
<init>();
}
หากไม่มีกฎนี้ R8 จะนำ PreCacheTask ออกจากแอปเนื่องจากแอปไม่ได้ใช้คลาสโดยตรง ซึ่งจะทำให้การผสานรวมหยุดทำงาน กฎนี้จะค้นหาคลาสที่ใช้การติดตั้งใช้งานอินเทอร์เฟซ StartupTask ของไลบรารีและเก็บคลาสเหล่านั้นไว้พร้อมกับคอนสตรักเตอร์ที่ไม่มีอาร์กิวเมนต์ ซึ่งจะช่วยให้ไลบรารีสร้างอินสแตนซ์และเรียกใช้ PreCacheTask ได้สำเร็จ
การสะท้อนด้วย ::class.java
ไลบรารีสามารถโหลดคลาสได้โดยให้แอปส่งออบเจ็กต์ Class โดยตรง ซึ่งเป็นวิธีที่เสถียรกว่าการโหลดคลาสตามชื่อ วิธีนี้จะสร้างการอ้างอิงที่ชัดเจนไปยังคลาสที่ R8 ตรวจหาได้ อย่างไรก็ตาม แม้ว่าวิธีนี้จะป้องกันไม่ให้ R8 นำคลาสออก แต่คุณยังคงต้องใช้กฎการเก็บรักษาเพื่อประกาศว่าคลาสได้รับการสร้างอินสแตนซ์แบบสะท้อนและเพื่อปกป้องสมาชิกที่เข้าถึงแบบสะท้อน เช่น คอนสตรักเตอร์
ตัวอย่างเช่น ลองพิจารณาสถานการณ์ต่อไปนี้ที่คุณมีไลบรารีและแอปที่ใช้ไลบรารี โดยตัวโหลดไลบรารีจะสร้างอินสแตนซ์อินเทอร์เฟซ StartupTask
โดยส่งการอ้างอิงคลาสโดยตรง
โค้ดไลบรารีมีดังนี้
// The interface for a task that runs once.
interface StartupTask {
fun run()
}
// The library object that loads and executes the task.
object TaskRunner {
fun execute(taskClass: Class<out StartupTask>) {
// The class isn't removed, but its constructor might be.
val task = taskClass.getDeclaredConstructor().newInstance()
task.run()
}
}
แอปที่ใช้ไลบรารีมีโค้ดดังนี้
// The app's task is to pre-cache data.
class PreCacheTask : StartupTask {
override fun run() {
Log.d("AppTask", "Warming up the cache...")
}
}
fun onCreate() {
// The library is given a direct reference to the app's task class.
TaskRunner.execute(PreCacheTask::class.java)
}
ในสถานการณ์นี้ ไลบรารีของคุณควรรวมไฟล์กฎการเก็บรักษาของผู้ใช้ที่มีกฎการเก็บรักษาต่อไปนี้
# Allow any implementation of StartupTask to be removed if unused.
-keep,allowobfuscation,allowshrinking class * implements com.example.library.StartupTask
# Keep the default constructor, which is called via reflection.
-keepclassmembers class * implements com.example.library.StartupTask {
<init>();
}
กฎเหล่านี้ออกแบบมาให้ทำงานได้อย่างสมบูรณ์แบบกับการสะท้อนประเภทนี้ ซึ่งช่วยให้เพิ่มประสิทธิภาพได้สูงสุดพร้อมกับตรวจสอบว่าโค้ดทำงานอย่างถูกต้อง กฎเหล่านี้ช่วยให้ R8 ปรับชื่อคลาสให้ยากต่อการอ่าน (Obfuscate) และลดขนาดหรือนำการติดตั้งใช้งานคลาส StartupTask ออกหากแอปไม่เคยใช้คลาสนี้ อย่างไรก็ตาม
สำหรับการติดตั้งใช้งานใดๆ เช่น PrecacheTask ที่ใช้ในตัวอย่าง
กฎเหล่านี้จะเก็บคอนสตรักเตอร์เริ่มต้น (<init>()) ที่ไลบรารีต้อง
เรียก
-keep,allowobfuscation,allowshrinking class * implements com.example.library.StartupTask: กฎนี้กำหนดเป้าหมายไปยังคลาสใดก็ตามที่ ใช้การติดตั้งใช้งานอินเทอร์เฟซStartupTask-keep class * implements com.example.library.StartupTask: กฎนี้จะเก็บคลาสใดก็ตาม (*) ที่ใช้การติดตั้งใช้งานอินเทอร์เฟซ,allowobfuscation: กฎนี้จะสั่งให้ R8 เปลี่ยนชื่อหรือปรับชื่อคลาสให้ยากต่อการอ่าน (Obfuscate) แม้ว่าจะเก็บคลาสไว้ก็ตาม การดำเนินการนี้ปลอดภัยเนื่องจากไลบรารีไม่ได้ขึ้นอยู่กับชื่อคลาส แต่จะรับออบเจ็กต์Classโดยตรง,allowshrinking: ตัวแก้ไขนี้จะสั่งให้ R8 นำคลาสออกได้หากไม่ได้ใช้ ซึ่งจะช่วยให้ R8 ลบการติดตั้งใช้งานStartupTaskที่ไม่เคยส่งไปยังTaskRunner.execute()ได้อย่างปลอดภัย กล่าวโดยย่อคือกฎนี้หมายความว่า หากแอปใช้คลาสที่ใช้การติดตั้งใช้งานStartupTaskR8 จะเก็บคลาสไว้ R8 สามารถเปลี่ยนชื่อคลาสเพื่อลดขนาดและลบคลาสได้หากแอปไม่ได้ใช้
-keepclassmembers class * implements com.example.library.StartupTask { <init>(); }: กฎนี้กำหนดเป้าหมายไปยังสมาชิกที่เฉพาะเจาะจงของคลาสที่ระบุไว้ใน กฎแรก ซึ่งในกรณีนี้คือคอนสตรักเตอร์-keepclassmembers class * implements com.example.library.StartupTask: กฎนี้จะเก็บสมาชิกที่เฉพาะเจาะจง (เมธอด, ฟิลด์) ของคลาสที่ใช้การติดตั้งใช้งานอินเทอร์เฟซStartupTaskแต่จะเก็บไว้ก็ต่อเมื่อมีการเก็บคลาสที่ใช้การติดตั้งใช้งานไว้ด้วย{ <init>(); }: นี่คือตัวเลือกสมาชิก<init>คือ ชื่อภายในพิเศษสำหรับคอนสตรักเตอร์ใน Java bytecode ส่วนนี้กำหนดเป้าหมายไปยังคอนสตรักเตอร์เริ่มต้นที่ไม่มีอาร์กิวเมนต์โดยเฉพาะ- กฎนี้มีความสำคัญเนื่องจากโค้ดของคุณเรียก
getDeclaredConstructor().newInstance()โดยไม่มีอาร์กิวเมนต์ ซึ่งจะเรียกคอนสตรักเตอร์เริ่มต้นแบบสะท้อน หากไม่มีกฎนี้ R8 จะเห็นว่าไม่มีโค้ดใดเรียกnew PreCacheTask()โดยตรง จึงสันนิษฐานว่าคอนสตรักเตอร์ไม่ได้ใช้และนำคอนสตรักเตอร์ออก ซึ่งจะทำให้แอปขัดข้องขณะรันไทม์ด้วยInstantiationException
การสะท้อนตามคำอธิบายประกอบเมธอด
ไลบรารีมักจะกำหนดคำอธิบายประกอบที่นักพัฒนาแอปใช้เพื่อติดแท็กเมธอดหรือฟิลด์
จากนั้นไลบรารีจะใช้การสะท้อนเพื่อค้นหาสมาชิกที่มีคำอธิบายประกอบเหล่านี้ขณะรันไทม์ ตัวอย่างเช่น คำอธิบายประกอบ @OnLifecycleEvent ใช้เพื่อค้นหาเมธอดที่จำเป็นขณะรันไทม์
ตัวอย่างเช่น ลองพิจารณาสถานการณ์ต่อไปนี้ที่คุณมีไลบรารีและแอปที่ใช้ไลบรารี โดยตัวอย่างจะแสดง Event Bus ที่ค้นหาและเรียกใช้เมธอดที่มีคำอธิบายประกอบด้วย @OnEvent
โค้ดไลบรารีมีดังนี้
@Retention(AnnotationRetention.RUNTIME)
@Target(AnnotationTarget.FUNCTION)
annotation class OnEvent
class EventBus {
fun dispatch(listener: Any) {
// Find all methods annotated with @OnEvent and invoke them
listener::class.java.declaredMethods.forEach { method ->
if (method.isAnnotationPresent(OnEvent::class.java)) {
try {
method.invoke(listener)
} catch (e: Exception) { /* ... */ }
}
}
}
}
แอปที่ใช้ไลบรารีมีโค้ดดังนี้
class MyEventListener {
@OnEvent
fun onSomethingHappened() {
// This method will be removed by R8 without a keep rule
Log.d(TAG, "Event received!")
}
}
fun onCreate() {
// Instantiate the listener and the event bus
val listener = MyEventListener()
val eventBus = EventBus()
// Dispatch the listener to the event bus
eventBus.dispatch(listener)
}
ไลบรารีควรรวมไฟล์กฎการเก็บรักษาของผู้ใช้ที่จะเก็บเมธอดใดก็ตามที่ใช้คำอธิบายประกอบโดยอัตโนมัติ
-keepattributes RuntimeVisibleAnnotations
-keep @interface com.example.library.OnEvent;
-keepclassmembers class * {
@com.example.library.OnEvent <methods>;
}
-keepattributes RuntimeVisibleAnnotations: กฎนี้จะเก็บ คำอธิบายประกอบที่ตั้งใจให้อ่านขณะรันไทม์-keep @interface com.example.library.OnEvent: กฎนี้จะเก็บคลาสคำอธิบายประกอบOnEventไว้-keepclassmembers class * {@com.example.library.OnEvent <methods>;}: กฎนี้จะเก็บคลาสและสมาชิกที่เฉพาะเจาะจงไว้ก็ต่อเมื่อมีการใช้คลาสและคลาสมีสมาชิกเหล่านั้น-keepclassmembers: กฎนี้จะเก็บคลาสและสมาชิกที่เฉพาะเจาะจงไว้ก็ต่อเมื่อมีการใช้คลาสและคลาสมีสมาชิกเหล่านั้นclass *: กฎนี้ใช้กับคลาสใดก็ได้@com.example.library.OnEvent <methods>;: กฎนี้จะเก็บคลาสใดก็ตาม ที่มีเมธอดอย่างน้อย 1 รายการ (<methods>) ที่มีคำอธิบายประกอบด้วย@com.example.library.OnEventและเก็บเมธอดที่มีคำอธิบายประกอบไว้ด้วย
การสะท้อนตามคำอธิบายประกอบชั้นเรียน
ไลบรารีสามารถใช้การสะท้อนเพื่อสแกนหาคลาสที่มีคำอธิบายประกอบที่เฉพาะเจาะจง
ในกรณีนี้ คลาส Task Runner จะค้นหาคลาสทั้งหมดที่มีคำอธิบายประกอบด้วย ReflectiveExecutor โดยใช้การสะท้อนและเรียกใช้เมธอด execute
ตัวอย่างเช่น ลองพิจารณาสถานการณ์ต่อไปนี้ที่คุณมีไลบรารีและแอปที่ใช้ไลบรารี
ไลบรารีมีโค้ดดังนี้
@Retention(AnnotationRetention.RUNTIME)
@Target(AnnotationTarget.CLASS)
annotation class ReflectiveExecutor
class TaskRunner {
fun process(task: Any) {
val taskClass = task::class.java
if (taskClass.isAnnotationPresent(ReflectiveExecutor::class.java)) {
val methodToCall = taskClass.getMethod("execute")
methodToCall.invoke(task)
}
}
}
แอปที่ใช้ไลบรารีมีโค้ดดังนี้
// In consumer app
@ReflectiveExecutor
class ImportantBackgroundTask {
fun execute() {
// This class will be removed by R8 without a keep rule
Log.e("ImportantBackgroundTask", "Executing the important background task...")
}
}
// Usage of ImportantBackgroundTask
fun onCreate(){
val task = ImportantBackgroundTask()
val runner = TaskRunner()
runner.process(task)
}
เนื่องจากไลบรารีใช้การสะท้อนแบบสะท้อนเพื่อรับคลาสที่เฉพาะเจาะจง ไลบรารีจึงควรรวมไฟล์กฎการเก็บรักษาของผู้ใช้ที่มีกฎการเก็บรักษาต่อไปนี้
# Retain annotation metadata for runtime reflection.
-keepattributes RuntimeVisibleAnnotations
# Keep the annotation interface itself.
-keep @interface com.example.library.ReflectiveExecutor
# Keep the execute method in the classes which are being used
-keepclassmembers @com.example.library.ReflectiveExecutor class * {
public void execute();
}
การกำหนดค่านี้มีประสิทธิภาพสูงเนื่องจากจะบอก R8 อย่างชัดเจนว่าจะเก็บอะไรไว้
การสะท้อนเพื่อรองรับทรัพยากร Dependency ที่ไม่บังคับ
กรณีการใช้งานทั่วไปสำหรับการสะท้อนคือการสร้างทรัพยากร Dependency แบบยืดหยุ่นระหว่างไลบรารีหลักกับไลบรารีส่วนเสริมที่ไม่บังคับ ไลบรารีหลักสามารถตรวจสอบว่ามีการรวมส่วนเสริมไว้ในแอปหรือไม่ และหากรวมไว้ ก็จะเปิดใช้ฟีเจอร์เพิ่มเติมได้ ซึ่งจะช่วยให้คุณจัดส่งโมดูลส่วนเสริมได้โดยไม่ต้องบังคับให้ไลบรารีหลักมีทรัพยากร Dependency โดยตรงกับโมดูลเหล่านั้น
ไลบรารีหลักใช้การสะท้อน (Class.forName) เพื่อค้นหาคลาสที่เฉพาะเจาะจงตามชื่อ หากพบคลาส ระบบจะเปิดใช้ฟีเจอร์ หากไม่พบ ระบบจะหยุดทำงานอย่างราบรื่น
ตัวอย่างเช่น ลองพิจารณาโค้ดต่อไปนี้ที่ AnalyticsManager หลักจะตรวจสอบคลาส VideoEventTracker ที่ไม่บังคับเพื่อเปิดใช้การวิเคราะห์วิดีโอ
ไลบรารีหลักมีโค้ดดังนี้
object AnalyticsManager {
private const val VIDEO_TRACKER_CLASS = "com.example.analytics.video.VideoEventTracker"
fun initialize() {
try {
// Attempt to load the optional module's class using reflection
Class.forName(VIDEO_TRACKER_CLASS).getDeclaredConstructor().newInstance()
Log.d(TAG, "Video tracking enabled.")
} catch (e: ClassNotFoundException) {
Log.d(TAG,"Video tracking module not found. Skipping.")
} catch (e: Exception) {
Log.e(TAG, e.printStackTrace())
}
}
}
ไลบรารีวิดีโอที่ไม่บังคับมีโค้ดดังนี้
package com.example.analytics.video
class VideoEventTracker {
// This constructor must be kept for the reflection call to succeed.
init { /* ... */ }
}
นักพัฒนาแอปของไลบรารีที่ไม่บังคับมีหน้าที่รับผิดชอบในการจัดเตรียมกฎการเก็บรักษาของผู้ใช้ที่จำเป็น กฎการเก็บรักษานี้จะช่วยให้แอปใดก็ตามที่ใช้ไลบรารีที่ไม่บังคับเก็บโค้ดที่ไลบรารีหลักต้องค้นหาไว้
# In the video library's consumer keep rules file
-keep class com.example.analytics.video.VideoEventTracker {
<init>();
}
หากไม่มีกฎนี้ R8 มีแนวโน้มที่จะนำ VideoEventTracker ออกจากไลบรารีที่ไม่บังคับเนื่องจากไม่มีสิ่งใดในโมดูลนั้นใช้คลาสนี้โดยตรง กฎการเก็บรักษาจะเก็บคลาสและคอนสตรักเตอร์ไว้ ซึ่งจะช่วยให้ไลบรารีหลักสร้างอินสแตนซ์คลาสนี้ได้สำเร็จ
การสะท้อนเพื่อเข้าถึงสมาชิกส่วนตัว
การใช้การสะท้อนเพื่อเข้าถึงโค้ดส่วนตัวหรือโค้ดที่ได้รับการปกป้องซึ่งไม่ได้เป็นส่วนหนึ่งของ API สาธารณะของไลบรารีอาจทำให้เกิดปัญหาที่สำคัญ โค้ดดังกล่าวอาจมีการเปลี่ยนแปลงโดยไม่ต้องแจ้งให้ทราบ ซึ่งอาจนำไปสู่ลักษณะการทำงานที่ไม่คาดคิดหรือการขัดข้องในแอปพลิเคชัน
เมื่อคุณใช้การสะท้อนสำหรับ API ที่ไม่ใช่สาธารณะ คุณอาจพบปัญหาต่อไปนี้
- การอัปเดตถูกบล็อก: การเปลี่ยนแปลงโค้ดส่วนตัวหรือโค้ดที่ได้รับการปกป้องอาจทำให้คุณอัปเดตเป็นไลบรารีเวอร์ชันที่สูงขึ้นไม่ได้
- พลาดสิทธิประโยชน์: คุณอาจพลาดฟังก์ชันการทำงานใหม่ๆ การแก้ไขข้อบกพร่องที่ทำให้เกิดการขัดข้องที่สำคัญ หรือการอัปเดตความปลอดภัยที่จำเป็น
การเพิ่มประสิทธิภาพของ R8 และการสะท้อน
หากคุณต้องสะท้อนโค้ดส่วนตัวหรือโค้ดที่ได้รับการปกป้องของไลบรารี โปรดให้ความสนใจกับการเพิ่มประสิทธิภาพของ R8 หากไม่มีการอ้างอิงโดยตรงไปยังสมาชิกเหล่านี้ R8 อาจสันนิษฐานว่าสมาชิกเหล่านี้ไม่ได้ใช้และนำออกหรือเปลี่ยนชื่อในภายหลัง
ซึ่งอาจนำไปสู่การขัดข้องขณะรันไทม์ โดยมักจะมีข้อความแสดงข้อผิดพลาดที่ทำให้เข้าใจผิด เช่น NoSuchMethodException หรือ NoSuchFieldException
ตัวอย่างเช่น ลองพิจารณาสถานการณ์ต่อไปนี้ที่แสดงวิธีเข้าถึงฟิลด์ส่วนตัวจากคลาสไลบรารี
ไลบรารีที่คุณไม่ได้เป็นเจ้าของมีโค้ดดังนี้
class LibraryClass {
private val secretMessage = "R8 will remove me"
}
แอปของคุณมีโค้ดดังนี้
fun accessSecretMessage(instance: LibraryClass) {
// Use Java reflection from Kotlin to access the private field
val secretField = instance::class.java.getDeclaredField("secretMessage")
secretField.isAccessible = true
// This will crash at runtime with R8 enabled
val message = secretField.get(instance) as String
}
เพิ่มกฎ -keep ในแอปเพื่อป้องกันไม่ให้ R8 นำฟิลด์ส่วนตัวออก
-keepclassmembers class com.example.LibraryClass {
private java.lang.String secretMessage;
}
-keepclassmembers: กฎนี้จะเก็บสมาชิกที่เฉพาะเจาะจงของคลาสไว้ก็ต่อเมื่อมีการเก็บคลาสไว้ด้วยclass com.example.LibraryClass: กฎนี้กำหนดเป้าหมายไปยังคลาสที่แน่นอนซึ่งมีฟิลด์private java.lang.String secretMessage;: กฎนี้จะระบุฟิลด์ส่วนตัวที่เฉพาะเจาะจงตามชื่อและประเภท
Java Native Interface (JNI)
การเพิ่มประสิทธิภาพของ R8 อาจทำให้เกิดปัญหาเมื่อทำงานกับการเรียกขึ้นจากโค้ดแบบเนทีฟ (C/C++) ไปยัง Java หรือ Kotlin แม้ว่าการเรียกขึ้นจาก Java หรือ Kotlin ไปยังโค้ดแบบเนทีฟก็อาจทำให้เกิดปัญหาเช่นกัน แต่ไฟล์ proguard-android-optimize.txt เริ่มต้นจะมีกฎต่อไปนี้เพื่อเก็บการเรียกขึ้นไว้ กฎนี้จะป้องกันไม่ให้มีการตัดเมธอดแบบเนทีฟออก
-keepclasseswithmembernames,includedescriptorclasses class * {
native <methods>;
}
การโต้ตอบกับโค้ดแบบเนทีฟผ่าน Java Native Interface (JNI)
เมื่อแอปใช้ JNI เพื่อทำการเรียกขึ้นจากโค้ดแบบเนทีฟ (C/C++) ไปยัง Java หรือ Kotlin, R8 จะมองไม่เห็นว่ามีการเรียกเมธอดใดจากโค้ดแบบเนทีฟ หากไม่มีการอ้างอิงโดยตรงไปยังเมธอดเหล่านี้ในแอป R8 จะสันนิษฐานอย่างไม่ถูกต้องว่าเมธอดเหล่านี้ไม่ได้ใช้และนำออก ซึ่งจะทำให้แอปขัดข้อง
ตัวอย่างต่อไปนี้แสดงคลาส Kotlin ที่มีเมธอดที่ตั้งใจให้เรียกจากไลบรารีแบบเนทีฟ ไลบรารีแบบเนทีฟจะสร้างอินสแตนซ์ประเภทแอปพลิเคชันและส่งข้อมูลจากโค้ดแบบเนทีฟไปยังโค้ด Kotlin
package com.example.models
// This class is used in the JNI bridge method signature
data class NativeData(val id: Int, val payload: String)
package com.example.app
// In package com.example.app
class JniBridge {
/**
* This method is called from the native side.
* R8 will remove it if it's not kept.
*/
fun onNativeEvent(data: NativeData) {
Log.d(TAG, "Received event from native code: $data")
}
// Use 'external' to declare a native method
external fun startNativeProcess()
companion object {
init {
// Load the native library
System.loadLibrary("my-native-lib")
}
}
}
ในกรณีนี้ คุณต้องแจ้งให้ R8 ทราบเพื่อป้องกันไม่ให้มีการเพิ่มประสิทธิภาพประเภทแอปพลิเคชัน นอกจากนี้ หากเมธอดที่เรียกจากโค้ดแบบเนทีฟใช้คลาสของคุณเองในลายเซ็นเป็นพารามิเตอร์หรือประเภทการแสดงผล คุณต้องตรวจสอบด้วยว่าไม่มีการเปลี่ยนชื่อคลาสเหล่านั้น
เพิ่มกฎการเก็บรักษาต่อไปนี้ลงในแอป
-keepclassmembers,includedescriptorclasses class com.example.JniBridge {
public void onNativeEvent(com.example.model.NativeData);
}
-keep class NativeData{
<init>(java.lang.Integer, java.lang.String);
}
กฎการเก็บรักษาเหล่านี้จะป้องกันไม่ให้ R8 นำเมธอด onNativeEvent และประเภทพารามิเตอร์ของเมธอดออกหรือเปลี่ยนชื่อ
-keepclassmembers,includedescriptorclasses class com.example.JniBridge{ public void onNativeEvent(com.example.model.NativeData);}: กฎนี้จะเก็บสมาชิกที่เฉพาะเจาะจงของคลาสไว้ก็ต่อเมื่อมีการสร้างอินสแตนซ์คลาส ในโค้ด Kotlin หรือ Java ก่อน ซึ่งจะบอก R8 ว่าแอปกำลังใช้คลาสและ ควรเก็บสมาชิกที่เฉพาะเจาะจงของคลาสไว้-keepclassmembers: กฎนี้จะเก็บสมาชิกที่เฉพาะเจาะจงของคลาสไว้ก็ต่อเมื่อมีการสร้างอินสแตนซ์คลาสในโค้ด Kotlin หรือ Java ก่อน ซึ่งจะบอก R8 ว่าแอปกำลังใช้คลาสและควรเก็บสมาชิกที่เฉพาะเจาะจงของคลาสไว้class com.example.JniBridge: กฎนี้กำหนดเป้าหมายไปยังคลาสที่แน่นอนซึ่งมีฟิลด์includedescriptorclasses: ตัวแก้ไขนี้จะเก็บคลาสใดก็ตามที่พบในลายเซ็นหรือตัวอธิบายของเมธอดไว้ด้วย ในกรณีนี้ ตัวแก้ไขจะป้องกันไม่ให้ R8 เปลี่ยนชื่อหรือนำคลาสcom.example.models.NativeDataซึ่งใช้เป็นพารามิเตอร์ออก หากมีการเปลี่ยนชื่อNativeData(เช่น เป็นa.a) ลายเซ็นเมธอดจะไม่ตรงกับสิ่งที่โค้ดแบบเนทีฟคาดหวังอีกต่อไป ซึ่งจะทำให้เกิดการขัดข้องpublic void onNativeEvent(com.example.models.NativeData);: กฎนี้จะระบุลายเซ็น Java ที่แน่นอนของเมธอดที่จะเก็บไว้
-keep class NativeData{<init>(java.lang.Integer, java.lang.String);}: แม้ว่าincludedescriptorclassesจะช่วยให้มั่นใจได้ว่ามีการเก็บคลาสNativeDataไว้ แต่สมาชิก (ฟิลด์หรือเมธอด) ใดก็ตามภายในNativeDataที่เข้าถึงโดยตรงจากโค้ด JNI แบบเนทีฟจะต้องมีกฎการเก็บรักษาของตัวเอง-keep class NativeData: กฎนี้กำหนดเป้าหมายไปยังคลาสที่ชื่อNativeDataและบล็อกจะระบุสมาชิกที่จะเก็บไว้ภายในคลาสNativeDataclass to keep.<init>(java.lang.Integer, java.lang.String): นี่คือ ลายเซ็นของคอนสตรักเตอร์ ลายเซ็นนี้จะระบุคอนสตรักเตอร์ที่ใช้พารามิเตอร์ 2 รายการอย่างไม่ซ้ำกัน โดยพารามิเตอร์แรกคือIntegerและพารามิเตอร์ที่ 2 คือString
การเรียกแพลตฟอร์มโดยอ้อม
โอนข้อมูลด้วยการติดตั้งใช้งาน Parcelable
เฟรมเวิร์ก Android ใช้การสะท้อนเพื่อสร้างอินสแตนซ์ออบเจ็กต์ Parcelable ในการพัฒนา Kotlin สมัยใหม่ คุณควรใช้ปลั๊กอิน kotlin-parcelize ซึ่งจะสร้างการติดตั้งใช้งาน Parcelable ที่จำเป็นโดยอัตโนมัติ รวมถึงฟิลด์ CREATOR และเมธอดที่เฟรมเวิร์กต้องการ
ตัวอย่างเช่น ลองพิจารณาตัวอย่างต่อไปนี้ที่ใช้ปลั๊กอิน kotlin-parcelize เพื่อสร้างคลาส Parcelable
import android.os.Parcelable
import kotlinx.parcelize.Parcelize
// Add the @Parcelize annotation to your data class
@Parcelize
data class UserData(
val name: String,
val age: Int
) : Parcelable
ในสถานการณ์นี้ ไม่มีกฎการเก็บรักษาที่แนะนำ ปลั๊กอิน kotlin-parcelize Gradle จะสร้างกฎการเก็บรักษาที่จำเป็นสำหรับคลาสที่คุณใส่คำอธิบายประกอบด้วย @Parcelize โดยอัตโนมัติ ปลั๊กอินจะจัดการความซับซ้อนให้คุณ โดยตรวจสอบว่ามีการเก็บ CREATOR และคอนสตรักเตอร์ที่สร้างขึ้นไว้สำหรับการเรียกการสะท้อนของเฟรมเวิร์ก Android
หากคุณเขียนคลาส Parcelable ด้วยตนเองใน Kotlin โดยไม่ใช้ @Parcelize,
คุณมีหน้าที่รับผิดชอบในการเก็บฟิลด์ CREATOR และคอนสตรักเตอร์ที่
รับ Parcel หากลืมดำเนินการดังกล่าว แอปจะขัดข้องเมื่อระบบพยายามยกเลิกการซีเรียลไลซ์ออบเจ็กต์ การใช้ @Parcelize เป็นแนวทางปฏิบัติมาตรฐานที่ปลอดภัยกว่า
เมื่อใช้ปลั๊กอิน kotlin-parcelize โปรดคำนึงถึงสิ่งต่อไปนี้
- ปลั๊กอินจะสร้างฟิลด์
CREATORโดยอัตโนมัติระหว่างการคอมไพล์ - ไฟล์
proguard-android-optimize.txtมีกฎkeepที่จำเป็นเพื่อเก็บฟิลด์เหล่านี้ไว้เพื่อให้ทำงานได้อย่างถูกต้อง - นักพัฒนาแอปต้องตรวจสอบว่ามีกฎ
keepที่จำเป็นทั้งหมด โดยเฉพาะอย่างยิ่งสำหรับการติดตั้งใช้งานที่กำหนดเองหรือทรัพยากร Dependency ของบุคคลที่สาม
ไลบรารียอดนิยม
ไลบรารีที่ใช้การสะท้อนหรือการแปลง bytecode จะเข้าถึงโค้ดแบบไดนามิกขณะรันไทม์ หาก R8 นำคลาส ฟิลด์ หรือเมธอดที่เข้าถึงด้วยวิธีนี้ออกหรือเปลี่ยนชื่อ แอปอาจขัดข้อง
อย่างไรก็ตาม ไลบรารีของบุคคลที่สามยอดนิยม (เช่น Gson, Retrofit และ Kotlinx Serialization) จะรวมกฎการเก็บรักษาของผู้ใช้ R8 ของตัวเองไว้โดยอัตโนมัติ เมื่อใช้ไลบรารีเวอร์ชันล่าสุด คุณไม่จำเป็นต้องเพิ่มกฎการเก็บรักษาด้วยตนเองลงในโปรเจ็กต์
Gson
Gson เป็นไลบรารีการซีเรียลไลซ์และการยกเลิกการซีเรียลไลซ์ JSON ซึ่งใช้การสะท้อนอย่างมาก เมื่อคุณใช้ โหมดเต็ม เพื่อเพิ่มประสิทธิภาพแอป ระบบจะนำลายเซ็นประเภททั่วไป คอนสตรัคเตอร์เริ่มต้น และฟิลด์ที่ไม่มีคำอธิบายประกอบออก เว้นแต่ จะได้รับคำสั่งอย่างชัดเจน
หากต้องการให้ Gson ทำงานอย่างถูกต้อง ให้เพิ่มกฎที่เฉพาะเจาะจงเพื่อเก็บฟิลด์ที่ไม่ใช่ชั่วคราวไว้ในคลาสโมเดลข้อมูลและเก็บลำดับชั้น TypeToken ไว้
# Preserve generic type information required for deserialization
-keepattributes Signature
# Keep all non-transient fields in your data model classes for reflection
-keepclassmembers class com.example.models.** {
!transient <fields>;
}
# Keep TypeToken itself and any anonymous classes extending it
-keep,allowobfuscation,allowshrinking,allowoptimization class com.google.gson.reflect.TypeToken { *; }
-keep,allowobfuscation,allowshrinking,allowoptimization class * extends com.google.gson.reflect.TypeToken
Gson จะไม่สนใจฟิลด์ที่มีตัวแก้ไข transient ระหว่างการ
ซีเรียลไลซ์และการยกเลิกการซีเรียลไลซ์ ซึ่งเป็นเหตุผลที่กฎการเก็บรักษากำหนดเป้าหมายไปยังฟิลด์ที่ไม่ใช่ชั่วคราว (!transient) โดยเฉพาะ
Retrofit
Retrofit เป็นไลบรารีเครือข่ายที่ตรวจสอบเมธอดอินเทอร์เฟซบริการที่มีคำอธิบายประกอบ HTTP (เช่น @GET หรือ @POST) โดยใช้การสะท้อนเพื่อสร้างคำขอเครือข่ายและแปลงการตอบกลับ
Retrofit จะสร้างการติดตั้งใช้งานอินเทอร์เฟซ API แบบไดนามิกขณะรันไทม์โดยใช้ Proxy.newProxyInstance() เนื่องจาก R8 ไม่เห็นคลาสใดที่ใช้การติดตั้งใช้งานอินเทอร์เฟซเหล่านี้แบบคงที่ จึงอาจนำเมธอดหรือประเภทการแสดงผลทั่วไปของเมธอดออก
กฎการเก็บรักษาที่รวมไว้
Retrofit ใช้การสะท้อนขณะรันไทม์เพื่อตรวจสอบพารามิเตอร์ทั่วไป คำอธิบายประกอบเมธอด และคำอธิบายประกอบพารามิเตอร์ หากไม่มีการกำหนดค่าที่เหมาะสม โหมดเต็มของ R8 อาจนำลายเซ็นทั่วไปออกจากประเภทการแสดงผล การดำเนินการต่อของ Kotlin และคลาสการตอบกลับออกอย่างละเอียด หรือแม้แต่แทนที่ค่าอินเทอร์เฟซด้วยค่า Null เนื่องจากอินเทอร์เฟซ Retrofit ได้รับการสร้างอินสแตนซ์แบบไดนามิกด้วยพร็อกซี
ตั้งแต่ Retrofit 2.10.0 เป็นต้นไป ไลบรารีจะรวมกฎการเก็บรักษาอย่างเป็นทางการที่จำเป็นเพื่อเก็บค่าเริ่มต้นของคำอธิบายประกอบ พารามิเตอร์เมธอดบริการ และข้อมูลเมตาของคลาสที่จำเป็นไว้โดยอัตโนมัติ ดูข้อมูลเพิ่มเติมได้ที่ กฎที่ Retrofit ใช้
เก็บประเภทการแสดงผลทั่วไปไว้
Retrofit จะตรวจสอบลายเซ็นทั่วไปของประเภทการแสดงผล (เช่น
Observable<Data>) เพื่อยกเลิกการซีเรียลไลซ์การตอบกลับของเครือขัยอย่างถูกต้อง หาก R8 นำลายเซ็นทั่วไปออก Retrofit จะแทนที่ออบเจ็กต์ที่สร้างอินสแตนซ์ด้วย null
หากต้องการป้องกันไม่ให้โหมดเต็มของ R8 นำลายเซ็นทั่วไปของประเภทการแสดงผลออก ให้ใช้กฎแบบมีเงื่อนไขต่อไปนี้
# Preserve generic type information for Call/Observable return types
-keepattributes Signature
# If an interface has a Retrofit HTTP annotation, keep its return type (class <3>)
-if interface * {
@retrofit2.http.* public *** *(...);
}
-keep,allowoptimization,allowshrinking,allowobfuscation class <3>
นอกจากนี้ คลาสโมเดลข้อมูลจริงที่แสดงผล (เช่น Data ใน
Observable<Data>) ต้องเก็บไว้ด้วย เนื่องจาก
ตัวแปลง (เช่น Gson) จะสร้างคลาสนี้แบบสะท้อน
Coroutine
เมื่อคุณใช้ Coroutine ของ Kotlin คอมไพเลอร์ Kotlin จะแปลงฟังก์ชัน suspend โดยเพิ่มพารามิเตอร์ Continuation ลงในลายเซ็นเมธอดที่คอมไพล์
เมื่อไลบรารี เช่น Retrofit อ่านลายเซ็นทั่วไปของฟังก์ชัน suspend แบบสะท้อน ไลบรารีจะใช้พารามิเตอร์ Continuation นั้น เมื่อใช้โหมดเต็ม ระบบจะเก็บแอตทริบิวต์ Signature ไว้สำหรับคลาสที่เก็บไว้อย่างชัดเจนเท่านั้น เนื่องจาก Continuation เป็นพารามิเตอร์สังเคราะห์ R8 จึงนำลายเซ็นของพารามิเตอร์ออกโดยค่าเริ่มต้น ซึ่งจะทำให้การสะท้อนหยุดทำงาน
หากต้องการป้องกันการนำลายเซ็นออกและตรวจสอบความเข้ากันได้ขณะรันไทม์ในโหมดเต็ม ให้รวมกฎต่อไปนี้
# Keep the signature attribute globally
-keepattributes Signature
# Explicitly keep the Continuation class so its signature is not stripped
-keep class kotlin.coroutines.Continuation