CodeGym /జావా బ్లాగ్ /యాదృచ్ఛికంగా /ఎరేజర్ టైప్ చేయండి
John Squirrels
స్థాయి
San Francisco

ఎరేజర్ టైప్ చేయండి

సమూహంలో ప్రచురించబడింది
హాయ్! మేము జెనరిక్స్‌పై మా పాఠాల శ్రేణిని కొనసాగిస్తాము. అవి ఏమిటో మరియు అవి ఎందుకు అవసరమో మాకు ఇంతకుముందు సాధారణ ఆలోచన వచ్చింది. ఈ రోజు మనం జెనరిక్స్ యొక్క కొన్ని లక్షణాల గురించి మరియు వాటితో పని చేయడం గురించి మరింత తెలుసుకుందాం. వెళ్దాం! గత పాఠంలోఎరేజర్ రకం - 1 , మేము సాధారణ రకాలు మరియు ముడి రకాలు మధ్య వ్యత్యాసం గురించి మాట్లాడాము . ముడి రకం అనేది సాధారణ తరగతి, దీని రకం తీసివేయబడింది.

List list = new ArrayList();
ఇక్కడ ఒక ఉదాహరణ ఉంది. మనలో ఏ రకమైన వస్తువులు ఉంచబడతాయో ఇక్కడ మేము సూచించము List. Listమేము అలాంటి వాటిని సృష్టించి , దానికి కొన్ని వస్తువులను జోడించడానికి ప్రయత్నిస్తే , మేము IDEAలో హెచ్చరికను చూస్తాము:

"Unchecked call to add(E) as a member of raw type of java.util.List".
కానీ మేము జావా 5లో మాత్రమే జెనరిక్స్ కనిపించిన వాస్తవం గురించి కూడా మాట్లాడాము. ఈ సంస్కరణ విడుదలయ్యే సమయానికి, ప్రోగ్రామర్లు ఇప్పటికే ముడి రకాలను ఉపయోగించి అనేక కోడ్‌లను వ్రాశారు, కాబట్టి భాష యొక్క ఈ లక్షణం పని చేయడాన్ని ఆపలేదు మరియు దాని సామర్థ్యం జావాలో ముడి రకాలను సృష్టించడం భద్రపరచబడింది. అయితే, సమస్య మరింత విస్తృతంగా మారింది. మీకు తెలిసినట్లుగా, జావా కోడ్ బైట్‌కోడ్ అని పిలువబడే ప్రత్యేక కంపైల్డ్ ఫార్మాట్‌కి మార్చబడుతుంది, అది జావా వర్చువల్ మెషీన్ ద్వారా అమలు చేయబడుతుంది. కానీ మార్పిడి ప్రక్రియలో మేము టైప్ పారామితుల గురించి సమాచారాన్ని బైట్‌కోడ్‌లో ఉంచినట్లయితే, అది గతంలో వ్రాసిన కోడ్‌ను విచ్ఛిన్నం చేస్తుంది, ఎందుకంటే జావా 5కి ముందు టైప్ పారామితులు లేవు! జెనరిక్స్‌తో పని చేస్తున్నప్పుడు, మీరు గుర్తుంచుకోవలసిన ముఖ్యమైన అంశం ఒకటి ఉంది. దీనిని టైప్ ఎరేజర్ అంటారు. రకం పరామితి గురించి తరగతిలో ఎటువంటి సమాచారం లేదని దీని అర్థం. ఈ సమాచారం సంకలనం సమయంలో మాత్రమే అందుబాటులో ఉంటుంది మరియు రన్‌టైమ్‌కు ముందు తొలగించబడుతుంది (అసాధ్యం అవుతుంది). మీరు మీలో తప్పు రకం వస్తువును ఉంచడానికి ప్రయత్నిస్తే List<String>, కంపైలర్ లోపాన్ని సృష్టిస్తుంది. భాషా సృష్టికర్తలు జెనరిక్స్‌ని సృష్టించినప్పుడు ఇదే సాధించాలనుకుంటున్నారు: కంపైల్-టైమ్ చెక్‌లు. కానీ మీ జావా కోడ్ మొత్తం బైట్‌కోడ్‌గా మారినప్పుడు, అది ఇకపై టైప్ పారామితుల గురించి సమాచారాన్ని కలిగి ఉండదు. బైట్‌కోడ్‌లో, మీ పిల్లుల జాబితా స్ట్రింగ్‌ల List<Cat>కంటే భిన్నంగా లేదు . List<String>బైట్‌కోడ్‌లో, వస్తువుల catsజాబితా అని ఏమీ చెప్పలేదు Cat. అటువంటి సమాచారం సంకలనం సమయంలో తొలగించబడుతుంది - మీరు జాబితాను కలిగి ఉన్నారనే వాస్తవం మాత్రమే List<Object> catsప్రోగ్రామ్ యొక్క బైట్‌కోడ్‌లో ముగుస్తుంది. ఇది ఎలా పని చేస్తుందో చూద్దాం:

public class TestClass<T> {

   private T value1;
   private T value2;

   public void printValues() {
       System.out.println(value1);
       System.out.println(value2);
   }

   public static <T> TestClass<T> createAndAdd2Values(Object o1, Object o2) {
       TestClass<T> result = new TestClass<>();
       result.value1 = (T) o1;
       result.value2 = (T) o2;
       return result;
   }

   public static void main(String[] args) {
       Double d = 22.111;
       String s = "Test String";
       TestClass<Integer> test = createAndAdd2Values(d, s);
       test.printValues();
   }
}
మేము మా స్వంత సాధారణ TestClassతరగతిని సృష్టించాము. ఇది చాలా సులభం: ఇది వాస్తవానికి 2 వస్తువుల యొక్క చిన్న "సేకరణ", ఇది వస్తువు సృష్టించబడిన వెంటనే నిల్వ చేయబడుతుంది. దీనికి 2 Tఫీల్డ్‌లు ఉన్నాయి. పద్ధతిని అమలు చేసినప్పుడు createAndAdd2Values(), రెండు పాస్ చేయబడిన ఆబ్జెక్ట్‌లు ( Object aమరియు Object bతప్పనిసరిగా రకానికి ప్రసారం చేసి T, ఆపై TestClassఆబ్జెక్ట్‌కి జోడించబడాలి. పద్ధతిలో main(), మేము ఒక క్రియేట్ చేస్తాము TestClass<Integer>, అనగా Integerటైప్ ఆర్గ్యుమెంట్ టైప్ పారామీటర్‌ను భర్తీ చేస్తుంది . మేము a మరియు a to Integerకూడా పాస్ చేస్తున్నాము పద్ధతి . మా ప్రోగ్రామ్ పని చేస్తుందని మీరు అనుకుంటున్నారా? అన్నింటికంటే, మేము టైప్ ఆర్గ్యుమెంట్‌గా పేర్కొన్నాము, కానీ ఖచ్చితంగా ఒక కు ప్రసారం చేయడం సాధ్యం కాదు !DoubleStringcreateAndAdd2Values()IntegerStringIntegermain()పద్ధతి మరియు తనిఖీ. కన్సోల్ అవుట్‌పుట్:

22.111 
Test String
అది ఊహించనిది! ఇలా ఎందుకు జరిగింది? ఇది రకం ఎరేజర్ యొక్క ఫలితం. కోడ్ కంపైల్ చేయబడినప్పుడు Integerమా ఆబ్జెక్ట్‌ను ఇన్‌స్టాంటియేట్ చేయడానికి ఉపయోగించే టైప్ ఆర్గ్యుమెంట్ గురించిన సమాచారం తొలగించబడింది. TestClass<Integer> testఫీల్డ్ అవుతుంది TestClass<Object> test. మా Doubleమరియు Stringవాదనలు సులభంగా వస్తువులుగా మార్చబడ్డాయి Object(అవి మనం ఊహించిన విధంగా వస్తువులుగా మార్చబడవు Integer!) మరియు నిశ్శబ్దంగా కు జోడించబడ్డాయి TestClass. రకం ఎరేజర్ యొక్క మరొక సాధారణ కానీ చాలా బహిర్గతం చేసే ఉదాహరణ ఇక్కడ ఉంది:

import java.util.ArrayList;
import java.util.List;

public class Main {

   private class Cat {

   }

   public static void main(String[] args) {

       List<String> strings = new ArrayList<>();
       List<Integer> numbers = new ArrayList<>();
       List<Cat> cats = new ArrayList<>();

       System.out.println(strings.getClass() == numbers.getClass());
       System.out.println(numbers.getClass() == cats.getClass());

   }
}
కన్సోల్ అవుట్‌పుట్:

true 
true
మేము మూడు విభిన్న రకాల ఆర్గ్యుమెంట్‌లతో సేకరణలను సృష్టించినట్లు కనిపిస్తోంది — String, Integer, మరియు మా స్వంత Catతరగతి. కానీ బైట్‌కోడ్‌కి మార్చే సమయంలో, మూడు జాబితాలు మారతాయి List<Object>, కాబట్టి ప్రోగ్రామ్ రన్ అయినప్పుడు మనం మూడు సందర్భాల్లోనూ ఒకే తరగతిని ఉపయోగిస్తున్నామని చెబుతుంది.

శ్రేణులు మరియు జెనరిక్స్‌తో పని చేస్తున్నప్పుడు ఎరేజర్‌ని టైప్ చేయండి

శ్రేణులు మరియు సాధారణ తరగతులతో (ఉదాహరణకు) పని చేస్తున్నప్పుడు స్పష్టంగా అర్థం చేసుకోవలసిన ముఖ్యమైన అంశం ఉంది List. మీ ప్రోగ్రామ్ కోసం డేటా స్ట్రక్చర్‌లను ఎంచుకునేటప్పుడు కూడా మీరు దానిని పరిగణనలోకి తీసుకోవాలి. జెనరిక్స్ రకం ఎరేజర్‌కు లోబడి ఉంటాయి. రన్‌టైమ్‌లో టైప్ పారామీటర్‌ల గురించి సమాచారం అందుబాటులో లేదు. దీనికి విరుద్ధంగా, ప్రోగ్రామ్ రన్ అవుతున్నప్పుడు శ్రేణులకు వాటి డేటా రకం గురించి తెలుసు మరియు దాని గురించిన సమాచారాన్ని ఉపయోగించవచ్చు. శ్రేణిలో చెల్లని రకాన్ని ఉంచడానికి ప్రయత్నిస్తే మినహాయింపు విసిరివేయబడుతుంది:

public class Main2 {

   public static void main(String[] args) {

       Object x[] = new String[3];
       x[0] = new Integer(222);
   }
}
కన్సోల్ అవుట్‌పుట్:

Exception in thread "main" java.lang.ArrayStoreException: java.lang.Integer
శ్రేణులు మరియు జెనరిక్స్ మధ్య చాలా పెద్ద వ్యత్యాసం ఉన్నందున, వాటికి అనుకూలత సమస్యలు ఉండవచ్చు. అన్నింటికంటే మించి, మీరు సాధారణ వస్తువుల శ్రేణిని లేదా కేవలం పారామితి చేయబడిన శ్రేణిని కూడా సృష్టించలేరు. అది కాస్త గందరగోళంగా అనిపిస్తుందా? ఒకసారి చూద్దాము. ఉదాహరణకు, మీరు జావాలో వీటిలో దేనినీ చేయలేరు:

new List<T>[]
new List<String>[]
new T[]
మేము ఆబ్జెక్ట్‌ల శ్రేణిని సృష్టించడానికి ప్రయత్నిస్తే List<String>, సాధారణ శ్రేణి సృష్టి గురించి ఫిర్యాదు చేసే సంకలన లోపం మనకు వస్తుంది:

import java.util.List;

public class Main2 {

   public static void main(String[] args) {

       // Compilation error! Generic array creation
       List<String>[] stringLists = new List<String>[1];
   }
}
అయితే ఇది ఎందుకు జరుగుతుంది? అటువంటి శ్రేణుల సృష్టి ఎందుకు అనుమతించబడదు? ఇదంతా రకం భద్రతను అందించడానికి. కంపైలర్ అటువంటి జెనరిక్ ఆబ్జెక్ట్‌ల శ్రేణులను సృష్టించడానికి వీలు కల్పిస్తే, మనమే టన్ను సమస్యలను సృష్టించుకోవచ్చు. జాషువా బ్లాచ్ పుస్తకం "ఎఫెక్టివ్ జావా" నుండి ఇక్కడ ఒక సాధారణ ఉదాహరణ:

public static void main(String[] args) {

   List<String>[] stringLists = new List<String>[1];  //  (1)
   List<Integer> intList = Arrays.asList(42, 65, 44);  //  (2)
   Object[] objects = stringLists;  //  (3)
   objects[0] = intList;  //  (4)
   String s = stringLists[0].get(0);  //  (5)
}
List<String>[] stringListsఅటువంటి శ్రేణిని సృష్టించడం అనుమతించబడిందని మరియు సంకలన దోషాన్ని సృష్టించదని ఊహించండి . ఇది నిజమైతే, మనం చేయగలిగే కొన్ని విషయాలు ఇక్కడ ఉన్నాయి: లైన్ 1లో, మేము జాబితాల శ్రేణిని సృష్టిస్తాము: List<String>[] stringLists. మా శ్రేణిలో ఒకటి ఉంది List<String>. లైన్ 2 లో, మేము సంఖ్యల జాబితాను సృష్టిస్తాము: List<Integer>. 3వ పంక్తిలో, మనము List<String>[]వేరియబుల్‌కు కేటాయిస్తాము Object[] objects. జావా భాష దీనిని అనుమతిస్తుంది: వస్తువుల శ్రేణి అన్ని సబ్‌క్లాస్‌ల వస్తువులు మరియు వస్తువులను Xనిల్వ చేయగలదు . దీని ప్రకారం, మీరు శ్రేణిలో ఏదైనా ఉంచవచ్చు . లైన్ 4లో, మేము శ్రేణి (a ) యొక్క ఏకైక మూలకాన్ని a తో భర్తీ చేస్తాము . అందువల్ల, మేము నిల్వ చేయడానికి మాత్రమే ఉద్దేశించిన శ్రేణిలో ఉంచాముXXObjectobjects()List<String>List<Integer>List<Integer>List<String>వస్తువులు! మేము లైన్ 5ని అమలు చేసినప్పుడు మాత్రమే లోపాన్ని ఎదుర్కొంటాము. A ClassCastExceptionరన్‌టైమ్‌లో విసిరివేయబడుతుంది. దీని ప్రకారం, అటువంటి శ్రేణుల సృష్టిపై నిషేధం జావాకు జోడించబడింది. ఇది అలాంటి పరిస్థితులను నివారించవచ్చు.

నేను టైప్ ఎరేజర్‌ని ఎలా పొందగలను?

బాగా, మేము టైప్ ఎరేజర్ గురించి తెలుసుకున్నాము. వ్యవస్థను మోసగించడానికి ప్రయత్నిద్దాం! :) టాస్క్: మాకు సాధారణ TestClass<T>తరగతి ఉంది. createNewT()మేము ఈ తరగతికి ఒక కొత్త ఆబ్జెక్ట్‌ని సృష్టించి, తిరిగి ఇచ్చే పద్ధతిని వ్రాయాలనుకుంటున్నాము T. కానీ ఇది అసాధ్యం, సరియైనదా? సంకలనం సమయంలో రకం గురించిన మొత్తం సమాచారం Tతొలగించబడుతుంది మరియు రన్‌టైమ్‌లో మనం ఏ రకమైన వస్తువును సృష్టించాలో నిర్ణయించలేము. దీన్ని చేయడానికి నిజానికి ఒక గమ్మత్తైన మార్గం ఉంది. జావాకు క్లాస్ ఉందని మీరు బహుశా గుర్తుంచుకోవచ్చు Class. మన వస్తువులలో ఏదైనా తరగతిని నిర్ణయించడానికి మేము దీన్ని ఉపయోగించవచ్చు:

public class Main2 {

   public static void main(String[] args) {

       Class classInt = Integer.class;
       Class classString = String.class;

       System.out.println(classInt);
       System.out.println(classString);
   }
}
కన్సోల్ అవుట్‌పుట్:

class java.lang.Integer 
class java.lang.String
అయితే ఇక్కడ మనం మాట్లాడని అంశం ఒకటి ఉంది. ఒరాకిల్ డాక్యుమెంటేషన్‌లో, క్లాస్ క్లాస్ జెనరిక్ అని మీరు చూస్తారు! ఎరేజర్ రకం - 3

https://docs.oracle.com/javase/8/docs/api/java/lang/Class.html

డాక్యుమెంటేషన్ ఇలా చెబుతోంది, "T - ఈ తరగతి వస్తువు ద్వారా రూపొందించబడిన తరగతి రకం." దీన్ని డాక్యుమెంటేషన్ భాష నుండి సాదా ప్రసంగానికి అనువదించడం, ఆబ్జెక్ట్ యొక్క తరగతి Integer.classకేవలం కాదు Class, కానీ Class<Integer>. ఆబ్జెక్ట్ రకం String.classకేవలం కాదు Class, కానీ Class<String>, మొదలైనవి. ఇది ఇప్పటికీ స్పష్టంగా లేకుంటే, మునుపటి ఉదాహరణకి టైప్ పరామితిని జోడించి ప్రయత్నించండి:

public class Main2 {

   public static void main(String[] args) {

       Class<Integer> classInt = Integer.class;
       // Compilation error!
       Class<String> classInt2 = Integer.class;
      
      
       Class<String> classString = String.class;
       // Compilation error!
       Class<Double> classString2 = String.class;
   }
}
ఇప్పుడు, ఈ జ్ఞానాన్ని ఉపయోగించి, మేము టైప్ ఎరేజర్‌ను దాటవేయవచ్చు మరియు మా పనిని సాధించవచ్చు! రకం పరామితి గురించి సమాచారాన్ని పొందడానికి ప్రయత్నిద్దాం. మా రకం వాదన ఇలా ఉంటుంది MySecretClass:

public class MySecretClass {

   public MySecretClass() {

       System.out.println("A MySecretClass object was created successfully!");
   }
}
మరియు ఇక్కడ మేము మా పరిష్కారాన్ని ఆచరణలో ఎలా ఉపయోగిస్తాము:

public class TestClass<T> {

   Class<T> typeParameterClass;

   public TestClass(Class<T> typeParameterClass) {
       this.typeParameterClass = typeParameterClass;
   }

   public T createNewT() throws IllegalAccessException, InstantiationException {
       T t = typeParameterClass.newInstance();
       return t;
   }

   public static void main(String[] args) throws InstantiationException, IllegalAccessException {

       TestClass<MySecretClass> testString = new TestClass<>(MySecretClass.class);
       MySecretClass secret = testString.createNewT();

   }
}
కన్సోల్ అవుట్‌పుట్:

A MySecretClass object was created successfully!
మేము మా సాధారణ తరగతి యొక్క కన్స్ట్రక్టర్‌కు అవసరమైన క్లాస్ ఆర్గ్యుమెంట్‌ని పంపాము:

TestClass<MySecretClass> testString = new TestClass<>(MySecretClass.class);
ఇది టైప్ ఆర్గ్యుమెంట్ గురించి సమాచారాన్ని సేవ్ చేయడానికి మమ్మల్ని అనుమతించింది, ఇది పూర్తిగా తొలగించబడకుండా నిరోధించింది. ఫలితంగా, మేము ఒక సృష్టించగలిగాముTవస్తువు! :) దాంతో ఈరోజు పాఠం ముగిసిపోతుంది. జెనరిక్స్‌తో పని చేస్తున్నప్పుడు మీరు ఎల్లప్పుడూ టైప్ ఎరేజర్‌ని గుర్తుంచుకోవాలి. ఈ ప్రత్యామ్నాయం చాలా సౌకర్యవంతంగా కనిపించడం లేదు, కానీ జెనరిక్స్ సృష్టించబడినప్పుడు జావా భాషలో భాగం కాదని మీరు అర్థం చేసుకోవాలి. పారామితి చేయబడిన సేకరణలను రూపొందించడంలో మరియు సంకలనం సమయంలో లోపాలను గుర్తించడంలో మాకు సహాయపడే ఈ ఫీచర్, తర్వాత పరిష్కరించబడింది. మొదటి వెర్షన్ నుండి జెనరిక్స్‌ని చేర్చిన కొన్ని ఇతర భాషలలో, టైప్ ఎరేజర్ లేదు (ఉదాహరణకు, C#లో). మార్గం ద్వారా, మేము జెనరిక్స్ అధ్యయనం పూర్తి చేయలేదు! తదుపరి పాఠంలో, మీరు జెనరిక్స్ యొక్క మరికొన్ని లక్షణాలతో పరిచయం పొందుతారు. ప్రస్తుతానికి, రెండు పనులను పరిష్కరించడం మంచిది! :)
వ్యాఖ్యలు
TO VIEW ALL COMMENTS OR TO MAKE A COMMENT,
GO TO FULL VERSION