வாழ்த்துக்கள், இளம் படவன். இந்த கட்டுரையில், ஜாவா புரோகிராமர்கள் சாத்தியமற்ற சூழ்நிலைகளில் மட்டுமே பயன்படுத்தும் சக்தியைப் பற்றி நான் உங்களுக்கு சொல்கிறேன். ஜாவாவின் இருண்ட பக்கம் பிரதிபலிப்பு API ஆகும். ஜாவாவில், ஜாவா பிரதிபலிப்பு API ஐப் பயன்படுத்தி பிரதிபலிப்பு செயல்படுத்தப்படுகிறது.
எனது தொகுப்பு படிநிலையில், MyClass இன் முழுப் பெயர் "reflection.MyClass" என்று இருக்கும். வகுப்பின் பெயரைக் கற்றுக்கொள்வதற்கான எளிய வழியும் உள்ளது (வகுப்பின் பெயரை ஒரு சரமாகத் திருப்பி விடுங்கள்):
ஜாவா பிரதிபலிப்பு என்றால் என்ன?
இணையத்தில் ஒரு குறுகிய, துல்லியமான மற்றும் பிரபலமான வரையறை உள்ளது. பிரதிபலிப்பு ( லேட் லத்தீன் reflexio - to turn back ) என்பது ஒரு நிரல் இயங்கும் போது அது பற்றிய தரவை ஆராய்வதற்கான ஒரு பொறிமுறையாகும். புலங்கள், முறைகள் மற்றும் வகுப்புக் கட்டமைப்பாளர்கள் பற்றிய தகவல்களை ஆராய பிரதிபலிப்பு உங்களை அனுமதிக்கிறது. தொகுக்கும் நேரத்தில் இல்லாத, ஆனால் இயங்கும் நேரத்தில் கிடைக்கக்கூடிய வகைகளுடன் பணிபுரிய பிரதிபலிப்பு உங்களை அனுமதிக்கிறது. பிரதிபலிப்பு மற்றும் பிழைத் தகவலை வழங்குவதற்கான தர்க்கரீதியாக நிலையான மாதிரியானது சரியான டைனமிக் குறியீட்டை உருவாக்குவதை சாத்தியமாக்குகிறது. வேறு வார்த்தைகளில் கூறுவதானால், ஜாவாவில் பிரதிபலிப்பு எவ்வாறு செயல்படுகிறது என்பதைப் புரிந்துகொள்வது உங்களுக்கு பல அற்புதமான வாய்ப்புகளைத் திறக்கும். நீங்கள் உண்மையில் வகுப்புகள் மற்றும் அவற்றின் கூறுகளை ஏமாற்றலாம். பிரதிபலிப்பு என்ன அனுமதிக்கிறது என்பதற்கான அடிப்படை பட்டியல் இங்கே:- ஒரு பொருளின் வகுப்பைக் கற்றுக்கொள்/தீர்மானித்தல்;
- ஒரு வகுப்பின் மாற்றிகள், புலங்கள், முறைகள், மாறிலிகள், கன்ஸ்ட்ரக்டர்கள் மற்றும் சூப்பர்கிளாஸ்கள் பற்றிய தகவலைப் பெறுங்கள்;
- செயல்படுத்தப்பட்ட இடைமுகம்(களுக்கு) என்ன முறைகள் உள்ளன என்பதைக் கண்டறியவும்;
- இயக்க நேரம் வரை வகுப்பின் பெயர் தெரியாத வகுப்பின் உதாரணத்தை உருவாக்கவும்;
- ஒரு பொருளின் புலங்களின் மதிப்புகளை பெயரால் பெற்று அமைக்கவும்;
- ஒரு பொருளின் முறையை பெயரால் அழைக்கவும்.
MyClass
:
public class MyClass {
private int number;
private String name = "default";
// public MyClass(int number, String name) {
// this.number = number;
// this.name = name;
// }
public int getNumber() {
return number;
}
public void setNumber(int number) {
this.number = number;
}
public void setName(String name) {
this.name = name;
}
private void printData(){
System.out.println(number + name);
}
}
நீங்கள் பார்க்க முடியும் என, இது மிகவும் அடிப்படை வகுப்பு. அளவுருக்கள் கொண்ட கட்டமைப்பாளர் வேண்டுமென்றே கருத்து தெரிவிக்கிறார். அதற்குப் பிறகு வருவோம். வகுப்பின் உள்ளடக்கங்களை நீங்கள் கவனமாகப் பார்த்தீர்கள் என்றால், பெயர் புலத்திற்கான பெறுநர் இல்லாததை நீங்கள் கவனித்திருக்கலாம் . பெயர் புலம் தனிப்பட்ட அணுகல் மாற்றியமைப்புடன் குறிக்கப்பட்டுள்ளது : வகுப்பிற்கு வெளியே எங்களால் அதை அணுக முடியாது, அதாவது அதன் மதிப்பை எங்களால் மீட்டெடுக்க முடியாது . " அதனால் என்ன பிரச்சனை ?" நீ சொல்கிறாய். "ஒரு பெறுநரைச் சேர்க்கவும் அல்லது அணுகல் மாற்றியை மாற்றவும்". நீங்கள் சரியாக இருப்பீர்கள், இல்லையெனில்MyClass
தொகுக்கப்பட்ட AAR நூலகத்தில் அல்லது மாற்றங்களைச் செய்யும் திறன் இல்லாத மற்றொரு தனியார் தொகுதியில் இருந்தது. நடைமுறையில், இது எல்லா நேரத்திலும் நடக்கும். மேலும் சில கவனக்குறைவான புரோகிராமர்கள் ஒரு பெறுபேற்றை எழுத மறந்துவிட்டார்கள் . பிரதிபலிப்பை நினைவில் கொள்ள வேண்டிய நேரம் இது! வகுப்பின் தனிப்பட்ட பெயர் புலத்திற்குச் செல்ல முயற்சிப்போம் MyClass
:
public static void main(String[] args) {
MyClass myClass = new MyClass();
int number = myClass.getNumber();
String name = null; // No getter =(
System.out.println(number + name); // Output: 0null
try {
Field field = myClass.getClass().getDeclaredField("name");
field.setAccessible(true);
name = (String) field.get(myClass);
} catch (NoSuchFieldException | IllegalAccessException e) {
e.printStackTrace();
}
System.out.println(number + name); // Output: 0default
}
என்ன நடந்தது என்பதை பகுப்பாய்வு செய்வோம். ஜாவாவில், ஒரு அற்புதமான வகுப்பு உள்ளது Class
. இது இயங்கக்கூடிய ஜாவா பயன்பாட்டில் வகுப்புகள் மற்றும் இடைமுகங்களைக் குறிக்கிறது. Class
மற்றும் இடையே உள்ள உறவை நாங்கள் மறைக்க மாட்டோம் ClassLoader
, ஏனெனில் அது இந்தக் கட்டுரையின் தலைப்பு அல்ல. அடுத்து, இந்த வகுப்பின் புலங்களை மீட்டெடுக்க, நீங்கள் getFields()
முறையை அழைக்க வேண்டும். இந்த முறை இந்த வகுப்பின் அணுகக்கூடிய புலங்கள் அனைத்தையும் வழங்கும். இது எங்களுக்கு வேலை செய்யாது, ஏனெனில் எங்கள் புலம் தனிப்பட்டது , எனவே நாங்கள் முறையைப் பயன்படுத்துகிறோம் getDeclaredFields()
. இந்த முறை வகுப்பு புலங்களின் வரிசையையும் வழங்குகிறது, ஆனால் இப்போது அது தனிப்பட்ட மற்றும் பாதுகாக்கப்பட்ட புலங்களை உள்ளடக்கியது. இந்த வழக்கில், நாங்கள் ஆர்வமாக உள்ள துறையின் பெயரை நாங்கள் அறிவோம், எனவே நாங்கள் முறையைப் getDeclaredField(String)
பயன்படுத்தலாம்String
விரும்பிய புலத்தின் பெயர். குறிப்பு: getFields()
மேலும் getDeclaredFields()
பெற்றோர் வகுப்பின் புலங்களைத் திருப்பித் தர வேண்டாம்! நன்று. Field
எங்கள் பெயரைக் குறிப்பிடும் ஒரு பொருள் கிடைத்தது . புலம் பொதுவில் இல்லாததால் , அதனுடன் பணிபுரிய அனுமதி வழங்க வேண்டும். முறை setAccessible(true)
மேலும் தொடர உதவுகிறது. இப்போது பெயர் புலம் எங்கள் முழு கட்டுப்பாட்டில் உள்ளது! Field
பொருளின் முறையை அழைப்பதன் மூலம் அதன் மதிப்பை நீங்கள் மீட்டெடுக்கலாம் get(Object)
, அங்கு Object
எங்கள் வகுப்பின் உதாரணம் உள்ளது MyClass
. வகையை மாற்றி , நமது பெயர்String
மாறிக்கு மதிப்பை ஒதுக்குகிறோம் . பெயர் புலத்தில் புதிய மதிப்பை அமைக்க செட்டரைக் கண்டுபிடிக்க முடியவில்லை என்றால் , நீங்கள் செட் முறையைப் பயன்படுத்தலாம்:
field.set(myClass, (String) "new value");
வாழ்த்துகள்! நீங்கள் பிரதிபலிப்பு அடிப்படைகளில் தேர்ச்சி பெற்றுள்ளீர்கள் மற்றும் தனிப்பட்ட துறையை அணுகியுள்ளீர்கள்! try/catch
தொகுதி மற்றும் கையாளப்படும் விதிவிலக்குகளின் வகைகளுக்கு கவனம் செலுத்துங்கள் . அவர்களின் இருப்பு தேவை என்பதை IDE உங்களுக்குத் தெரிவிக்கும், ஆனால் அவர்கள் ஏன் இங்கு இருக்கிறார்கள் என்பதை அவர்களின் பெயர்களால் நீங்கள் தெளிவாகக் கூறலாம். நகர்கிறது! நீங்கள் கவனித்தபடி, எங்கள் MyClass
வகுப்பில் ஏற்கனவே வகுப்புத் தரவைப் பற்றிய தகவலைக் காண்பிக்கும் முறை உள்ளது:
private void printData(){
System.out.println(number + name);
}
ஆனால் இந்த புரோகிராமர் தனது கைரேகையை இங்கேயும் விட்டுவிட்டார். இந்த முறை ஒரு தனிப்பட்ட அணுகல் மாற்றியைக் கொண்டுள்ளது, மேலும் ஒவ்வொரு முறையும் தரவைக் காண்பிக்க எங்கள் சொந்த குறியீட்டை எழுத வேண்டும். என்ன ஒரு குழப்பம். நமது பிரதிபலிப்பு எங்கே போனது? பின்வரும் செயல்பாட்டை எழுதவும்:
public static void printData(Object myClass){
try {
Method method = myClass.getClass().getDeclaredMethod("printData");
method.setAccessible(true);
method.invoke(myClass);
} catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException e) {
e.printStackTrace();
}
}
இங்குள்ள செயல்முறையானது புலத்தை மீட்டெடுப்பதற்குப் பயன்படுத்தப்படும் செயல்முறையைப் போன்றது. நாங்கள் விரும்பிய முறையை பெயரால் அணுகி அதற்கான அணுகலை வழங்குகிறோம். மற்றும் Method
பொருளின் மீது நாம் invoke(Object, Args)
முறை என்று அழைக்கிறோம், அங்கு Object
வகுப்பின் ஒரு உதாரணமும் உள்ளது MyClass
. Args
முறையின் வாதங்கள், எங்களுடையது இல்லை என்றாலும். printData
இப்போது தகவலைக் காண்பிக்க செயல்பாட்டைப் பயன்படுத்துகிறோம் :
public static void main(String[] args) {
MyClass myClass = new MyClass();
int number = myClass.getNumber();
String name = null; //?
printData(myClass); // Output: 0default
try {
Field field = myClass.getClass().getDeclaredField("name");
field.setAccessible(true);
field.set(myClass, (String) "new value");
name = (String) field.get(myClass);
} catch (NoSuchFieldException | IllegalAccessException e) {
e.printStackTrace();
}
printData(myClass);// Output: 0new value
}
ஹர்ரே! இப்போது வகுப்பின் தனிப்பட்ட முறைக்கான அணுகலைப் பெற்றுள்ளோம். ஆனால் முறைக்கு வாதங்கள் இருந்தால் என்ன செய்வது, மற்றும் கட்டமைப்பாளர் ஏன் கருத்து தெரிவிக்கிறார்? எல்லாம் அதன் சொந்த நேரத்தில். ரன் நேரத்தில் (நிரல் இயங்கும் போது) ஒரு வகுப்பின் நிகழ்வுகளை உருவாக்க பிரதிபலிப்பு உங்களை அனுமதிக்கிறது என்பது தொடக்கத்தில் உள்ள வரையறையிலிருந்து தெளிவாகிறது ! வகுப்பின் முழுப் பெயரைப் பயன்படுத்தி ஒரு பொருளை உருவாக்கலாம். வகுப்பின் முழுப் பெயர் அதன் தொகுப்பின் பாதை உட்பட வகுப்பின் பெயராகும் .

MyClass.class.getName()
வகுப்பின் நிகழ்வை உருவாக்க ஜாவா பிரதிபலிப்பைப் பயன்படுத்துவோம்:
public static void main(String[] args) {
MyClass myClass = null;
try {
Class clazz = Class.forName(MyClass.class.getName());
myClass = (MyClass) clazz.newInstance();
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException e) {
e.printStackTrace();
}
System.out.println(myClass); // Output: created object reflection.MyClass@60e53b93
}
ஜாவா பயன்பாடு தொடங்கும் போது, அனைத்து வகுப்புகளும் JVM இல் ஏற்றப்படாது. MyClass
உங்கள் குறியீடு வகுப்பைக் குறிக்கவில்லை என்றால் ClassLoader
, JVM இல் வகுப்புகளை ஏற்றுவதற்குப் பொறுப்பான , வகுப்பை ஏற்றாது. அதாவது, நீங்கள் ClassLoader
அதை ஏற்றுவதற்கு கட்டாயப்படுத்த வேண்டும் மற்றும் ஒரு மாறி வடிவில் ஒரு வகுப்பு விளக்கத்தைப் பெற வேண்டும் Class
. இதனால்தான் எங்களிடம் forName(String)
முறை உள்ளது, String
அதன் விளக்கம் நமக்குத் தேவையான வகுப்பின் பெயர் எங்கே. பொருளைப் பெற்ற பிறகு Сlass
, முறையை அழைப்பது அந்த விளக்கத்தைப் பயன்படுத்தி உருவாக்கப்பட்ட newInstance()
ஒரு பொருளைத் திருப்பித் தரும் Object
. இந்த பொருளை எங்களுக்கு வழங்குவது மட்டுமே எஞ்சியுள்ளதுMyClass
வர்க்கம். குளிர்! அது கடினமாக இருந்தது, ஆனால் புரிந்துகொள்ளக்கூடியதாக இருந்தது, நான் நம்புகிறேன். இப்போது நாம் ஒரு வகுப்பின் உதாரணத்தை ஒரு வரியில் உருவாக்கலாம்! துரதிர்ஷ்டவசமாக, விவரிக்கப்பட்ட அணுகுமுறை இயல்புநிலை கட்டமைப்பாளருடன் மட்டுமே செயல்படும் (அளவுருக்கள் இல்லாமல்). அளவுருக்கள் கொண்ட முறைகள் மற்றும் கட்டமைப்பாளர்களை எப்படி அழைப்பது? எங்கள் கட்டமைப்பாளரைக் கருத்துத் தெரிவிக்க வேண்டிய நேரம் இது. எதிர்பார்த்தபடி, newInstance()
இயல்புநிலை கட்டமைப்பாளரைக் கண்டுபிடிக்க முடியவில்லை, இனி வேலை செய்யாது. கிளாஸ் இன்ஸ்டண்டியேஷனை மீண்டும் எழுதுவோம்:
public static void main(String[] args) {
MyClass myClass = null;
try {
Class clazz = Class.forName(MyClass.class.getName());
Class[] params = {int.class, String.class};
myClass = (MyClass) clazz.getConstructor(params).newInstance(1, "default2");
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | NoSuchMethodException | InvocationTargetException e) {
e.printStackTrace();
}
System.out.println(myClass);// Output: created object reflection.MyClass@60e53b93
}
வகுப்பு கட்டமைப்பாளர்களைப் பெறுவதற்கு இந்த getConstructors()
முறையானது வர்க்க வரையறையில் அழைக்கப்பட வேண்டும், பின்னர் getParameterTypes()
ஒரு கட்டமைப்பாளரின் அளவுருக்களைப் பெற அழைக்கப்பட வேண்டும்:
Constructor[] constructors = clazz.getConstructors();
for (Constructor constructor : constructors) {
Class[] paramTypes = constructor.getParameterTypes();
for (Class paramType : paramTypes) {
System.out.print(paramType.getName() + " ");
}
System.out.println();
}
இது அனைத்து கட்டமைப்பாளர்களையும் அவற்றின் அளவுருக்களையும் பெறுகிறது. எனது எடுத்துக்காட்டில், குறிப்பிட்ட, முன்னர் அறியப்பட்ட அளவுருக்கள் கொண்ட ஒரு குறிப்பிட்ட கட்டமைப்பாளரை நான் குறிப்பிடுகிறேன். இந்த கட்டமைப்பாளரை அழைக்க, newInstance
இந்த அளவுருக்களின் மதிப்புகளை அனுப்பும் முறையைப் பயன்படுத்துகிறோம். invoke
அழைப்பு முறைகளைப் பயன்படுத்தும்போதும் இதுவே இருக்கும் . இது கேள்வியைக் கேட்கிறது: பிரதிபலிப்பு மூலம் கட்டமைப்பாளர்களை அழைப்பது எப்போது பயனுள்ளதாக இருக்கும்? ஆரம்பத்தில் ஏற்கனவே குறிப்பிட்டுள்ளபடி, நவீன ஜாவா தொழில்நுட்பங்கள் ஜாவா பிரதிபலிப்பு API இல்லாமல் பெற முடியாது. எடுத்துக்காட்டாக, டிபென்டென்சி இன்ஜெக்ஷன் (DI), இது குறிப்புகளை முறைகள் மற்றும் கட்டமைப்பாளர்களின் பிரதிபலிப்புடன் ஒருங்கிணைத்து பிரபலமான டேரரை உருவாக்குகிறது.ஆண்ட்ராய்டு மேம்பாட்டிற்கான நூலகம். இந்தக் கட்டுரையைப் படித்த பிறகு, ஜாவா ரிஃப்ளெக்ஷன் ஏபிஐயின் வழிகளில் நீங்கள் படித்தவர் என்று நீங்கள் நம்பிக்கையுடன் கருதலாம். அவர்கள் பிரதிபலிப்பை ஜாவாவின் இருண்ட பக்கமாக அழைப்பதில்லை. இது OOP முன்னுதாரணத்தை முற்றிலும் உடைக்கிறது. ஜாவாவில், என்காப்சுலேஷன் சில நிரல் கூறுகளுக்கு மற்றவர்களின் அணுகலை மறைக்கிறது மற்றும் கட்டுப்படுத்துகிறது. நாங்கள் தனிப்பட்ட மாற்றியைப் பயன்படுத்தும் போது, அந்த புலத்தை அது இருக்கும் வகுப்பிற்குள் இருந்து மட்டுமே அணுக வேண்டும் என்று எண்ணுகிறோம். இந்த கொள்கையின் அடிப்படையில் நிரலின் அடுத்தடுத்த கட்டமைப்பை நாங்கள் உருவாக்குகிறோம். இந்த கட்டுரையில், எங்கும் உங்கள் வழியை கட்டாயப்படுத்த நீங்கள் பிரதிபலிப்பைப் பயன்படுத்துவது எப்படி என்பதைப் பார்த்தோம். படைப்பு வடிவமைப்பு முறை சிங்கிள்டன்ஒரு கட்டடக்கலை தீர்வாக இது ஒரு நல்ல உதாரணம். அடிப்படை யோசனை என்னவென்றால், இந்த முறையை செயல்படுத்தும் ஒரு வர்க்கம் முழு நிரலையும் செயல்படுத்தும் போது ஒரு நிகழ்வு மட்டுமே இருக்கும். இயல்புநிலை கட்டமைப்பாளருடன் தனிப்பட்ட அணுகல் மாற்றியைச் சேர்ப்பதன் மூலம் இது நிறைவேற்றப்படுகிறது. ஒரு புரோகிராமர் பிரதிபலிப்பைப் பயன்படுத்தினால் அது மிகவும் மோசமானதாக இருக்கும். மூலம், சமீபத்தில் ஒரு சக பணியாளர் ஒரு சுவாரஸ்யமான கேள்வியைக் கேட்டேன்: சிங்கிள்டன் முறையைச் செயல்படுத்தும் ஒரு வகுப்பை மரபுரிமையாகப் பெற முடியுமா? இந்த விஷயத்தில், பிரதிபலிப்பு கூட சக்தியற்றதாக இருக்க முடியுமா? கீழே உள்ள கருத்துகளில் கட்டுரை மற்றும் உங்கள் பதிலைப் பற்றிய உங்கள் கருத்தை இடுங்கள், மேலும் உங்கள் சொந்த கேள்விகளைக் கேளுங்கள்!
மேலும் வாசிப்பு: |
---|
GO TO FULL VERSION