রাখার নিয়ম যোগ করুন

একটি উচ্চ স্তরে, একটি রাখা নিয়ম একটি শ্রেণী (বা সাবক্লাস বা বাস্তবায়ন) নির্দিষ্ট করে এবং তারপরে সদস্য-পদ্ধতি, কনস্ট্রাক্টর বা ক্ষেত্রগুলি-কে সংরক্ষণ করার জন্য সেই শ্রেণীর মধ্যে।

একটি Keep নিয়মের জন্য সাধারণ সিনট্যাক্স নিম্নরূপ, তবে কিছু Keep বিকল্প ঐচ্ছিক keep_option_modfier গ্রহণ করে না।


-<keep_option>[,<keep_option_modifier_1>,<keep_option_modifier_2>,...] <class_specification>

নিম্নলিখিত একটি Keep নিয়মের উদাহরণ যা keepclassmembers Keep বিকল্প হিসেবে ব্যবহার করে, অপ্টিমাইজেশনকে মডিফায়ার হিসেবে allowoptimization এবং com.example.MyClass থেকে someSpecificMethod() রাখে:

-keepclassmembers,allowoptimization class com.example.MyClass {
  void someSpecificMethod();
}

বিকল্প রাখুন

কিপ অপশনটি আপনার রাখার নিয়মের প্রথম অংশ। এটি একটি শ্রেণির কোন দিকগুলি সংরক্ষণ করতে হবে তা নির্দিষ্ট করে। এখানে ছয়টি আলাদা রাখার বিকল্প রয়েছে, যেমন keep , keepclassmembers , keepclasseswithmembers , keepnames , keepclassmembernames , keepclasseswithmembernames

নিম্নলিখিত সারণী এই রাখার বিকল্পগুলি বর্ণনা করে:

বিকল্প রাখুন বর্ণনা
keepclassmembers অপ্টিমাইজেশানের পরে ক্লাসটি বিদ্যমান থাকলে শুধুমাত্র নির্দিষ্ট সদস্যদের সংরক্ষণ করে।
keep নির্দিষ্ট শ্রেণী এবং নির্দিষ্ট সদস্য (ক্ষেত্র এবং পদ্ধতি) সংরক্ষণ করে, তাদের অপ্টিমাইজ করা থেকে বাধা দেয়।

দ্রষ্টব্য : keep সাধারণত শুধুমাত্র Keep অপশন মডিফায়ারের সাথে ব্যবহার করা উচিত কারণ keep নিজে থেকে মিলিত ক্লাসে যেকোনো ধরনের অপ্টিমাইজেশন হতে বাধা দেয়।
keepclasseswithmembers একটি শ্রেণী এবং এর নির্দিষ্ট সদস্যদের সংরক্ষণ করে শুধুমাত্র যদি ক্লাসে শ্রেণী স্পেসিফিকেশন থেকে সমস্ত সদস্য থাকে।
keepclassmembernames নির্দিষ্ট শ্রেণীর সদস্যদের পুনঃনামকরণ প্রতিরোধ করে, কিন্তু শ্রেণী বা এর সদস্যদের সরানো থেকে বাধা দেয় না।

দ্রষ্টব্য: এই বিকল্পটির অর্থ প্রায়ই ভুল বোঝা যায়; সমতুল্য -keepclassmembers,allowshrinking
keepnames ক্লাস এবং তাদের সদস্যদের পুনঃনামকরণ রোধ করে, কিন্তু যদি সেগুলি অব্যবহৃত বলে গণ্য করা হয় তবে এটি তাদের সম্পূর্ণরূপে সরানো থেকে বাধা দেয় না।

দ্রষ্টব্য: এই বিকল্পটির অর্থ প্রায়ই ভুল বোঝা যায়; এর পরিবর্তে সমতুল্য -keep,allowshrinking ব্যবহার করার কথা বিবেচনা করুন।
keepclasseswithmembernames ক্লাস এবং তাদের নির্দিষ্ট সদস্যদের পুনঃনামকরণ প্রতিরোধ করে, কিন্তু শুধুমাত্র যদি সদস্যরা চূড়ান্ত কোডে বিদ্যমান থাকে। এটি কোড অপসারণ প্রতিরোধ করে না।

দ্রষ্টব্য: এই বিকল্পটির অর্থ প্রায়ই ভুল বোঝা যায়; সমতুল্য -keepclasseswithmembers,allowshrinking

সঠিক রাখার বিকল্পটি বেছে নিন

আপনার অ্যাপের জন্য সঠিক অপ্টিমাইজেশন নির্ধারণের জন্য সঠিক রাখার বিকল্পটি বাছাই করা অত্যন্ত গুরুত্বপূর্ণ। কিছু রাখার বিকল্পগুলি সঙ্কুচিত কোড, একটি প্রক্রিয়া যার মাধ্যমে রেফারেন্সহীন কোড সরানো হয়, যখন অন্যরা কোডকে অস্পষ্ট করে বা নাম পরিবর্তন করে। নিম্নলিখিত টেবিলটি বিভিন্ন রাখার বিকল্পগুলির ক্রিয়া নির্দেশ করে:

বিকল্প রাখুন ক্লাস সংকুচিত করে ক্লাস অস্পষ্ট করে সদস্যদের সঙ্কুচিত করে সদস্যদের অস্পষ্ট করে
keep
keepclassmembers
keepclasseswithmembers
keepnames
keepclassmembernames
keepclasseswithmembernames

অপশন মডিফায়ার রাখুন

একটি Keep বিকল্প মডিফায়ার একটি Keep নিয়মের সুযোগ এবং আচরণ নিয়ন্ত্রণ করতে ব্যবহৃত হয়। আপনি আপনার Keep নিয়মে 0 বা তার বেশি Keep অপশন মডিফায়ার যোগ করতে পারেন।

একটি Keep অপশন মডিফায়ারের সম্ভাব্য মানগুলি নিম্নলিখিত টেবিলে বর্ণনা করা হয়েছে:

মান বর্ণনা
allowoptimization নির্দিষ্ট উপাদান অপ্টিমাইজেশান অনুমতি দেয়. যাইহোক, নির্দিষ্ট উপাদানের নাম পরিবর্তন বা সরানো হয় না।
allowobfucastion নির্দিষ্ট উপাদানের পুনঃনামকরণের অনুমতি দেয়। যাইহোক, উপাদানগুলি সরানো বা অন্যথায় অপ্টিমাইজ করা হয় না।
allowshrinking R8 তাদের কোনো রেফারেন্স খুঁজে না পেলে নির্দিষ্ট উপাদান অপসারণের অনুমতি দেয়। যাইহোক, উপাদানগুলির নাম পরিবর্তন করা হয় না বা অন্যথায় অপ্টিমাইজ করা হয় না।
includedescriptorclasses R8-কে নির্দেশ দেয় যে সমস্ত শ্রেণীগুলি (প্যারামিটারের ধরন এবং রিটার্নের ধরন) এবং ক্ষেত্রগুলি (ক্ষেত্রের ধরন) রাখা হচ্ছে তার বর্ণনাকারীতে প্রদর্শিত।
allowaccessmodification R8 কে অপ্টিমাইজেশন প্রক্রিয়া চলাকালীন ক্লাস, পদ্ধতি এবং ক্ষেত্রগুলির অ্যাক্সেস মডিফায়ারগুলি ( public , private , protected ) পরিবর্তন করতে (সাধারণত প্রশস্ত) করার অনুমতি দেয়।
allowrepackage R8 কে ডিফল্ট (রুট) প্যাকেজ সহ বিভিন্ন প্যাকেজে ক্লাস সরানোর অনুমতি দেয়।

ক্লাস স্পেসিফিকেশন

রাখা নিয়মের অংশ হিসেবে আপনাকে অবশ্যই একটি ক্লাস, সুপারক্লাস বা বাস্তবায়িত ইন্টারফেস নির্দিষ্ট করতে হবে। java.lang.String মতো java.lang নামস্থান থেকে ক্লাস সহ সমস্ত ক্লাস, তাদের সম্পূর্ণ যোগ্য জাভা নাম ব্যবহার করে নির্দিষ্ট করতে হবে। যে নামগুলি ব্যবহার করা উচিত তা বোঝার জন্য, Get generated Java names- এ বর্ণিত টুলগুলি ব্যবহার করে বাইটকোড পরিদর্শন করুন।

নিম্নলিখিত উদাহরণ দেখায় কিভাবে আপনি MaterialButton ক্লাস নির্দিষ্ট করতে হবে:

  • সঠিক: com.google.android.material.button.MaterialButton
  • ভুল: MaterialButton

ক্লাস স্পেসিফিকেশন এমন একটি ক্লাসের মধ্যে সদস্যদেরও নির্দিষ্ট করে যা রাখা উচিত। নিম্নলিখিত নিয়ম MaterialButton ক্লাস এবং এর সমস্ত সদস্যকে রাখে:

-keep class com.google.android.material.button.MaterialButton { *; }

উপশ্রেণী এবং বাস্তবায়ন

একটি সাবক্লাস বা শ্রেণী লক্ষ্য করতে যা একটি ইন্টারফেস প্রয়োগ করে, যথাক্রমে extend এবং implements ব্যবহার করুন।

উদাহরণস্বরূপ, আপনার যদি নিম্নরূপ সাবক্লাস Foo সহ ক্লাস Bar থাকে:

class Foo : Bar()

নিম্নলিখিত রাখা নিয়মটি Bar সমস্ত উপশ্রেণী সংরক্ষণ করে। মনে রাখবেন যে রাখা নিয়মটি সুপারক্লাস Bar অন্তর্ভুক্ত করে না।

-keep class * extends Bar

আপনার যদি ক্লাস Foo থাকে যা Bar প্রয়োগ করে:

class Foo : Bar

নিম্নলিখিত Keep নিয়মটি Bar প্রয়োগ করে এমন সমস্ত ক্লাস সংরক্ষণ করে। মনে রাখবেন যে রাখার নিয়মটি ইন্টারফেস Bar অন্তর্ভুক্ত করে না।

-keep class * implements Bar

অ্যাক্সেস মডিফায়ার

আপনার রাখার নিয়মগুলিকে আরও সুনির্দিষ্ট করতে আপনি public , private , static এবং final মতো অ্যাক্সেস মডিফায়ারগুলি নির্দিষ্ট করতে পারেন।

উদাহরণস্বরূপ, নিম্নলিখিত নিয়মটি api প্যাকেজ এবং এর সাব-প্যাকেজের মধ্যে সমস্ত public ক্লাস এবং এই ক্লাসগুলিতে সমস্ত পাবলিক এবং সুরক্ষিত সদস্যদের রাখে।

-keep public class com.example.api.** { public protected *; }

আপনি একটি ক্লাসের সদস্যদের জন্য মডিফায়ার ব্যবহার করতে পারেন। উদাহরণস্বরূপ, নিম্নলিখিত নিয়মটি শুধুমাত্র Utils ক্লাসের public static পদ্ধতিগুলিকে রাখে:

-keep class com.example.Utils {
    public static void *(...);
}

কোটলিন-নির্দিষ্ট সংশোধক

R8 internal এবং suspend মতো কোটলিন-নির্দিষ্ট পরিবর্তনকারীকে সমর্থন করে না। এই ধরনের ক্ষেত্রগুলি রাখতে নিম্নলিখিত নির্দেশিকাগুলি ব্যবহার করুন৷

  • একটি internal শ্রেণী, পদ্ধতি বা ক্ষেত্র রাখতে, এটিকে সর্বজনীন হিসাবে বিবেচনা করুন। উদাহরণস্বরূপ, নিম্নলিখিত কোটলিন উত্স বিবেচনা করুন:

    package com.example
    internal class ImportantInternalClass {
      internal f: Int
      internal fun m() {}
    }
    

    internal ক্লাস, পদ্ধতি এবং ক্ষেত্রগুলি Kotlin কম্পাইলার দ্বারা উত্পাদিত .class ফাইলগুলিতে public , তাই আপনাকে অবশ্যই public কীওয়ার্ডটি ব্যবহার করতে হবে যা নিম্নলিখিত উদাহরণে দেখানো হয়েছে:

    -keepclassmembers public class com.example.ImportantInternalClass {
      public int f;
      public void m();
    }
    
  • যখন একজন suspend সদস্যকে কম্পাইল করা হয়, তখন রাখা নিয়মে তার সংকলিত স্বাক্ষর মেলে।

    উদাহরণস্বরূপ, যদি আপনার কাছে fetchUser ফাংশনটি নিম্নলিখিত স্নিপেটে দেখানো হিসাবে সংজ্ঞায়িত করা থাকে:

    suspend fun fetchUser(id: String): User
    

    কম্পাইল করা হলে, বাইটকোডে এর স্বাক্ষর নিচের মত দেখায়:

    public final Object fetchUser(String id, Continuation<? super User> continuation);
    

    এই ফাংশনের জন্য একটি Keep নিয়ম লিখতে, আপনাকে অবশ্যই এই সংকলিত স্বাক্ষরের সাথে মেলে বা ব্যবহার করতে হবে ...

    সংকলিত স্বাক্ষর ব্যবহার করার একটি উদাহরণ নিম্নরূপ:

    -keepclassmembers class com.example.repository.UserRepository {
    public java.lang.Object fetchUser(java.lang.String,  kotlin.coroutines.Continuation);
    }
    

    ... ব্যবহার করে একটি উদাহরণ নিম্নরূপ:

    -keepclassmembers class com.example.repository.UserRepository {
    public java.lang.Object fetchUser(...);
    }
    

সদস্য স্পেসিফিকেশন

ক্লাস স্পেসিফিকেশন ঐচ্ছিকভাবে সংরক্ষিত ক্লাস সদস্যদের অন্তর্ভুক্ত করে। আপনি যদি একটি ক্লাসের জন্য এক বা একাধিক সদস্য নির্দিষ্ট করেন, নিয়মটি শুধুমাত্র সেই সদস্যদের জন্য প্রযোজ্য।

উদাহরণস্বরূপ, একটি নির্দিষ্ট শ্রেণী এবং এর সমস্ত সদস্যদের সংরক্ষণ করতে, নিম্নলিখিতগুলি ব্যবহার করুন:

-keep class com.myapp.MyClass { *; }

শুধুমাত্র ক্লাস সংরক্ষণ করতে এবং এর সদস্যদের নয়, নিম্নলিখিতগুলি ব্যবহার করুন:

-keep class com.myapp.MyClass

বেশিরভাগ সময়, আপনি কিছু সদস্য নির্দিষ্ট করতে চাইবেন। উদাহরণস্বরূপ, নিম্নলিখিত উদাহরণটি MyClass ক্লাসের মধ্যে পাবলিক ফিল্ড text এবং পাবলিক মেথড updateText() রাখে।

-keep class com.myapp.MyClass {
    public java.lang.String text;
    public void updateText(java.lang.String);
}

সমস্ত পাবলিক ক্ষেত্র এবং পাবলিক পদ্ধতি রাখতে, নিম্নলিখিত উদাহরণ দেখুন:

-keep public class com.example.api.ApiClient {
    public *;
}

পদ্ধতি

একটি রাখা নিয়মের জন্য সদস্য স্পেসিফিকেশনে একটি পদ্ধতি নির্দিষ্ট করার জন্য সিনট্যাক্স নিম্নরূপ:

[<access_modifier>] [<return_type>] <method_name>(<parameter_types>);

উদাহরণ স্বরূপ, নিচের Keep নিয়মটি setLabel() নামক একটি পাবলিক পদ্ধতি রাখে যা void ফেরত দেয় এবং একটি String নেয়।

-keep class com.example.MyView {
    public void setLabel(java.lang.String);
}

আপনি <methods> একটি শর্টকাট হিসাবে ব্যবহার করতে পারেন একটি ক্লাসের সমস্ত পদ্ধতির সাথে মেলে নিম্নরূপ:

-keep class com.example.MyView {
    <methods>;
}

রিটার্নের ধরন এবং পরামিতি প্রকারের জন্য প্রকারগুলি কীভাবে নির্দিষ্ট করতে হয় সে সম্পর্কে আরও জানতে, প্রকারগুলি দেখুন।

কনস্ট্রাক্টর

একটি কনস্ট্রাক্টর নির্দিষ্ট করতে, <init> ব্যবহার করুন। একটি রাখা নিয়মের জন্য সদস্য স্পেসিফিকেশনে একটি কনস্ট্রাক্টর নির্দিষ্ট করার জন্য সিনট্যাক্স নিম্নরূপ:

[<access_modifier>] <init>(parameter_types);

উদাহরণস্বরূপ, নিম্নলিখিত Keep নিয়মটি একটি কাস্টম View কনস্ট্রাক্টর রাখে যা একটি Context এবং একটি AttributeSet নেয়।

-keep class com.example.ui.MyCustomView {
    public <init>(android.content.Context, android.util.AttributeSet);
}

সমস্ত পাবলিক কনস্ট্রাক্টর রাখতে, একটি রেফারেন্স হিসাবে নিম্নলিখিত উদাহরণ ব্যবহার করুন:

-keep class com.example.ui.MyCustomView {
    public <init>(...);
}

ক্ষেত্র

একটি রাখা নিয়মের জন্য সদস্য স্পেসিফিকেশনে একটি ক্ষেত্র নির্দিষ্ট করার জন্য সিনট্যাক্স নিম্নরূপ:

[<access_modifier>...] [<type>] <field_name>;

উদাহরণস্বরূপ, নিম্নলিখিত Keep নিয়মটি userId নামে একটি ব্যক্তিগত স্ট্রিং ক্ষেত্র এবং STATUS_ACTIVE নামক একটি পাবলিক স্ট্যাটিক পূর্ণসংখ্যা ক্ষেত্র রাখে:

-keep class com.example.models.User {
    private java.lang.String userId;
    public static int STATUS_ACTIVE;
}

আপনি <fields> একটি শর্টকাট হিসাবে একটি ক্লাসের সমস্ত ক্ষেত্রগুলিকে নিম্নরূপ মেলে ব্যবহার করতে পারেন:

-keep class com.example.models.User {
    <fields>;
}

প্যাকেজ-স্তরের ফাংশন

একটি কোটলিন ফাংশন রেফারেন্স করতে যা একটি ক্লাসের বাইরে সংজ্ঞায়িত করা হয় (সাধারণত শীর্ষ স্তরের ফাংশন বলা হয়), কোটলিন কম্পাইলার দ্বারা অন্তর্নিহিতভাবে যোগ করা ক্লাসের জন্য জেনারেট করা জাভা নামটি ব্যবহার করতে ভুলবেন না। ক্লাসের নাম হল Kotlin ফাইলের নাম Kt যুক্ত। উদাহরণস্বরূপ, যদি আপনার কাছে MyClass.kt নামে একটি Kotlin ফাইল থাকে যা নিম্নরূপ সংজ্ঞায়িত করা হয়েছে:

package com.example.myapp.utils

// A top-level function not inside a class
fun isEmailValid(email: String): Boolean {
    return email.contains("@")
}

isEmailValid ফাংশনের জন্য একটি Keep নিয়ম লিখতে, ক্লাস স্পেসিফিকেশনকে জেনারেট করা ক্লাস MyClassKt টার্গেট করতে হবে:

-keep class com.example.myapp.utils.MyClassKt {
    public static boolean isEmailValid(java.lang.String);
}

প্রকারভেদ

এই বিভাগে কীভাবে রিটার্নের ধরন, প্যারামিটারের ধরন এবং ক্ষেত্রটির ধরনগুলি নিয়ম মেম্বার স্পেসিফিকেশন বজায় রাখতে হয় তা বর্ণনা করা হয়েছে। কোটলিন সোর্স কোড থেকে ভিন্ন হলে প্রকারগুলি নির্দিষ্ট করতে জেনারেট করা জাভা নামগুলি ব্যবহার করতে ভুলবেন না৷

আদিম প্রকার

একটি আদিম প্রকার নির্দিষ্ট করতে, এর জাভা কীওয়ার্ড ব্যবহার করুন। R8 নিম্নলিখিত আদিম প্রকারগুলিকে স্বীকৃতি দেয়: boolean , byte , short , char , int , long , float , double

একটি আদিম টাইপ সহ একটি উদাহরণ নিয়ম নিম্নরূপ:

# Keeps a method that takes an int and a float as parameters.
-keepclassmembers class com.example.Calculator {
    public void setValues(int, float);
}

জেনেরিক প্রকার

কম্পাইলেশনের সময়, Kotlin/Java কম্পাইলার জেনেরিক ধরনের তথ্য মুছে ফেলে, তাই যখন আপনি লিখবেন এমন নিয়মগুলি রাখুন যাতে জেনেরিক প্রকার জড়িত থাকে আপনাকে অবশ্যই আপনার কোডের সংকলিত উপস্থাপনাকে লক্ষ্য করতে হবে , মূল সোর্স কোড নয়। জেনেরিক প্রকারগুলি কীভাবে পরিবর্তিত হয় সে সম্পর্কে আরও জানতে, টাইপ ইরেজার দেখুন।

উদাহরণস্বরূপ, যদি আপনার কাছে Box.kt এ সংজ্ঞায়িত একটি সীমাহীন জেনেরিক টাইপ সহ নিম্নলিখিত কোড থাকে:

package com.myapp.data

class Box<T>(val item: T) {
    fun getItem(): T {
        return item
    }
}

টাইপ ইরেজারের পর, T এর পরিবর্তে Object হয়। ক্লাস কনস্ট্রাক্টর এবং পদ্ধতি রাখার জন্য, আপনার নিয়মটি অবশ্যই জেনেরিক T জায়গায় java.lang.Object ব্যবহার করতে হবে।

একটি উদাহরণ রাখার নিয়ম নিম্নরূপ হবে:

# Keep the constructor and methods of the Box class.
-keep class com.myapp.data.Box {
    public init(java.lang.Object);
    public java.lang.Object getItem();
}

আপনার যদি NumberBox.kt এ একটি আবদ্ধ জেনেরিক টাইপের নিম্নলিখিত কোড থাকে:

package com.myapp.data

// T is constrained to be a subtype of Number
class NumberBox<T : Number>(val number: T)

এই ক্ষেত্রে, টাইপ ইরেজার T এর আবদ্ধ, java.lang.Number দিয়ে প্রতিস্থাপন করে।

একটি উদাহরণ রাখার নিয়ম নিম্নরূপ হবে:

-keep class com.myapp.data.NumberBox {
    public init(java.lang.Number);
}

বেস ক্লাস হিসাবে অ্যাপ-নির্দিষ্ট জেনেরিক প্রকারগুলি ব্যবহার করার সময়, বেস ক্লাসগুলির জন্য নিয়মগুলিও অন্তর্ভুক্ত করা প্রয়োজন৷

উদাহরণস্বরূপ, নিম্নলিখিত কোডের জন্য:

package com.myapp.data

data class UnpackOptions(val useHighPriority: Boolean)

// The generic Box class with UnpackOptions as the bounded type
class Box<T: UnpackOptions>(val item: T) {
}

UnpackOptions ক্লাস এবং Box ক্লাস পদ্ধতি উভয়ই একটি একক নিয়মের সাথে সংরক্ষণ করার জন্য আপনি includedescriptorclasses সহ একটি Keep নিয়ম ব্যবহার করতে পারেন:

-keep,includedescriptorclasses class com.myapp.data.Box {
    public <init>(com.myapp.data.UnpackOptions);
}

একটি নির্দিষ্ট ফাংশন রাখতে যা বস্তুর একটি তালিকা প্রক্রিয়া করে, আপনাকে একটি নিয়ম লিখতে হবে যা ফাংশনের স্বাক্ষরের সাথে অবিকল মেলে। মনে রাখবেন যে জেনেরিক প্রকারগুলি মুছে ফেলার কারণে, List<Product> মতো একটি প্যারামিটার java.util.List হিসাবে দেখা যায়।

উদাহরণস্বরূপ, যদি আপনার কাছে একটি ফাংশন সহ একটি ইউটিলিটি ক্লাস থাকে যা নিম্নরূপ Product বস্তুর একটি তালিকা প্রক্রিয়া করে:

package com.myapp.utils

import com.myapp.data.Product
import android.util.Log

class DataProcessor {
    // This is the function we want to keep
    fun processProducts(products: List<Product>) {
        Log.d("DataProcessor", "Processing ${products.size} products.")
        // Business logic ...
    }
}

// The data class used in the list (from the previous example)
package com.myapp.data
data class Product(val id: String, val name: String)

আপনি শুধুমাত্র processProducts ফাংশন রক্ষা করার জন্য নিম্নলিখিত Keep নিয়ম ব্যবহার করতে পারেন:

-keep class com.myapp.utils.DataProcessor {
    public void processProducts(java.util.List);
}

অ্যারে প্রকার

অ্যারের প্রতিটি মাত্রার জন্য কম্পোনেন্ট টাইপের সাথে [] যুক্ত করে একটি অ্যারের ধরন নির্দিষ্ট করুন। এটি উভয় শ্রেণীর প্রকার এবং আদিম প্রকারের ক্ষেত্রে প্রযোজ্য।

  • এক-মাত্রিক ক্লাস অ্যারে: java.lang.String[]
  • দ্বি-মাত্রিক আদিম অ্যারে: int[][]

উদাহরণস্বরূপ, যদি আপনার নিম্নলিখিত কোড থাকে:

package com.example.data

class ImageProcessor {
  fun process(): ByteArray {
    // process image to return a byte array
  }
}

আপনি নিম্নলিখিত রাখার নিয়ম ব্যবহার করতে পারেন:

# Keeps a method that returns a byte array.
-keepclassmembers class com.example.data.ImageProcessor {
    public byte[] process();
}

ওয়াইল্ডকার্ড

নিম্নলিখিত সারণীটি দেখায় যে কীভাবে ওয়াইল্ডকার্ড ব্যবহার করতে হয় একাধিক ক্লাস বা সদস্যদের জন্য নিয়মগুলি প্রয়োগ করতে যা একটি নির্দিষ্ট প্যাটার্নের সাথে মেলে।

ওয়াইল্ডকার্ড ক্লাস বা সদস্যদের জন্য প্রযোজ্য বর্ণনা
** উভয় সবচেয়ে বেশি ব্যবহৃত হয়। প্যাকেজ বিভাজক যেকোন সংখ্যক সহ যেকোন প্রকারের নামের সাথে মিলে যায়। এটি একটি প্যাকেজ এবং এর সাব-প্যাকেজের মধ্যে সমস্ত ক্লাস মেলানোর জন্য দরকারী।
* উভয় ক্লাস স্পেসিফিকেশনের জন্য, প্যাকেজ বিভাজক ( . ) ধারণ করে না এমন একটি টাইপ নামের যেকোনো অংশের সাথে মেলে
সদস্য স্পেসিফিকেশনের জন্য, যেকোনো পদ্ধতি বা ক্ষেত্রের নামের সাথে মেলে। যখন নিজে ব্যবহার করা হয়, তখন এটি ** এর একটি উপনামও হয়।
? উভয় শ্রেণী বা সদস্যের নামের যেকোনো একক অক্ষর মেলে।
*** সদস্যরা আদিম প্রকার (যেমন int ), ক্লাসের ধরন (যেমন java.lang.String ), এবং যেকোনো মাত্রার অ্যারের প্রকার (যেমন byte[][] ) সহ যেকোনো প্রকারের সাথে মেলে।
... সদস্যরা একটি পদ্ধতির জন্য প্যারামিটারের যেকোনো তালিকার সাথে মেলে।
% সদস্যরা যেকোন আদিম প্রকারের সাথে মেলে (যেমন `int`, `float`, `বুলিয়ান`, বা অন্যান্য)।

এখানে বিশেষ ওয়াইল্ডকার্ডগুলি কীভাবে ব্যবহার করবেন তার কয়েকটি উদাহরণ রয়েছে:

  • আপনার যদি একই নামের একাধিক পদ্ধতি থাকে যা ইনপুট হিসাবে বিভিন্ন আদিম ধরন গ্রহণ করে, আপনি একটি Keep নিয়ম লিখতে % ব্যবহার করতে পারেন যা সেগুলিকে রাখে। উদাহরণস্বরূপ, এই DataStore ক্লাসে একাধিক setValue পদ্ধতি রয়েছে:

    class DataStore {
        fun setValue(key: String, value: Int) { ... }
        fun setValue(key: String, value: Boolean) { ... }
        fun setValue(key: String, value: Float) { ... }
    }
    

    নিম্নলিখিত রাখার নিয়মটি সমস্ত পদ্ধতি বজায় রাখে:

    -keep class com.example.DataStore {
        public void setValue(java.lang.String, %);
    }
    
  • যদি আপনার একাধিক শ্রেণী থাকে যার নাম এক অক্ষর দ্বারা পরিবর্তিত হয়, ব্যবহার করুন ? একটি রাখা নিয়ম লিখতে যা তাদের সব রাখে। উদাহরণস্বরূপ, যদি আপনার নিম্নলিখিত ক্লাস থাকে:

    com.example.models.UserV1 {...}
    com.example.models.UserV2 {...}
    com.example.models.UserV3 {...}
    

    নিম্নলিখিত রাখার নিয়মটি সমস্ত ক্লাস রাখে:

    -keep class com.example.models.UserV?
    
  • ক্লাস Example এবং AnotherExample (যদি সেগুলি রুট-লেভেল ক্লাস হয়), কিন্তু com.foo.Example না হয়, নিম্নলিখিত রাখার নিয়মটি ব্যবহার করুন:

    -keep class *Example
    
  • আপনি যদি নিজে থেকে * ব্যবহার করেন তবে এটি ** এর একটি উপনাম হিসাবে কাজ করে। উদাহরণস্বরূপ, নিম্নলিখিত রাখার নিয়মগুলি সমতুল্য:

    -keepclasseswithmembers class * { public static void main(java.lang.String[];) }
    
    -keepclasseswithmembers class ** { public static void main(java.lang.String[];) }
    

উত্পন্ন জাভা নাম পরিদর্শন করুন

রাখার নিয়ম লেখার সময়, আপনাকে অবশ্যই জাভা বাইটকোডে কম্পাইল করার পরে তাদের নাম ব্যবহার করে ক্লাস এবং অন্যান্য রেফারেন্স প্রকারগুলি নির্দিষ্ট করতে হবে (উদাহরণগুলির জন্য ক্লাস স্পেসিফিকেশন এবং প্রকারগুলি দেখুন)। আপনার কোডের জন্য জেনারেট করা জাভা নামগুলি কী তা পরীক্ষা করতে, অ্যান্ড্রয়েড স্টুডিওতে নিম্নলিখিত যে কোনও একটি টুল ব্যবহার করুন:

  • APK বিশ্লেষক
  • Kotlin সোর্স ফাইল খোলার সাথে, Tools > Kotlin > Show Kotlin Bytecode > Decompile- এ গিয়ে বাইটকোড পরিদর্শন করুন।