Дөңгелек тәуелділік - Circular dependency

Жылы бағдарламалық жасақтама, а шеңберлік тәуелділік тікелей немесе жанама түрде дұрыс жұмыс істеуіне бір-біріне тәуелді екі немесе одан да көп модульдер арасындағы қатынас. Мұндай модульдер ретінде белгілі өзара рекурсивті.

Шолу

Дөңгелек тәуелділіктер көп жағдайда табиғи болып табылады домендік модельдер мұнда бір доменнің белгілі бір объектілері бір-біріне тәуелді.[1] Алайда, жылы бағдарламалық жасақтама, үлкен бағдарламалық модульдер арасындағы айналмалы тәуелділіктер ан шаблонға қарсы олардың жағымсыз әсерлеріне байланысты.[1] Осыған қарамастан, осындай циркулярлық (немесе циклдік) тәуелділіктер бағдарламалық жасақтаманың бастапқы файлдары арасында кең тарағаны анықталды.[2] Өзара рекурсивті модульдер, алайда, біршама кең таралған функционалды бағдарламалау, онда индуктивті және рекурсивті анықтамалар жиі көтермеленеді.

Мәселелер

Дөңгелек тәуелділік бағдарламалық жасақтамада көптеген жағымсыз әсерлер тудыруы мүмкін. Бағдарламалық жасақтама тұрғысынан ең проблемалы болып табылады тығыз муфта бір модульді бөлек қайта пайдалануды қысқартатын немесе мүмкін емес ететін өзара тәуелді модульдер.

Дөңгелек тәуелділік а тудыруы мүмкін домино эффектісі бір модульдегі кішігірім жергілікті өзгеріс басқа модульдерге жайылып, қалаусыз ғаламдық эффекттерге ие болған кезде (бағдарлама қателері, компиляция қателері). Дөңгелек тәуелділік сонымен қатар шексіз рекурсияларға немесе басқа күтпеген сәтсіздіктерге әкелуі мүмкін.

Дөңгелек тәуелділіктер де себеп болуы мүмкін жадтың ағуы алдын-ала белгілі бір қарабайыр автоматты қоқыс жинаушылар (пайдаланатындар анықтамалық санау ) пайдаланылмаған объектілерді бөлуден.

Себептері мен шешімдері

Бағдарламалық жасақтаманың өте үлкен дизайнында бағдарламалық жасақтама инженерлері контекстті жоғалтуы мүмкін және байқаусызда дөңгелек тәуелділіктер енгізуі мүмкін. Бағдарламалық жасақтаманы талдауға және қажет емес дөңгелек тәуелділіктерді табуға арналған құралдар бар.[3]

Іске асыру кезінде айналмалы тәуелділіктер енгізілуі мүмкін қайта телефон соғу функционалдылық. Қолдану арқылы бұны болдырмауға болады дизайн үлгілері сияқты бақылаушы үлгісі.

C ++ тіліндегі мысал

C / C ++ тілдеріндегі дөңгелек тәуелділіктерді жүзеге асыру сәл күрделі болуы мүмкін, өйткені кез-келген құрылым немесе сынып анықтамасы оны қолданудан жоғары бір файлға орналастырылуы керек. Сыныптар арасындағы айналмалы тәуелділік A және B осылайша екеуі де анықтамасын қажет етеді A жоғарыда орналастырылуы керек B, және анықтамасы B жоғарыда орналастырылуы керек A, бұл, әрине, мүмкін емес. A алға қарай декларациялау сондықтан оны орындау үшін қажет.

Келесі мысал мұның қалай жасалатынын көрсетеді.

  • Файл а:
#ifndef A_H# A_H анықтаусынып B;	// тікелей декларациясынып A {қоғамдық:	B* б;};#endif // A_H
  • Файл б.х.:
#ifndef B_H# B_H анықтау сынып A;	// тікелей декларациясынып B {қоғамдық:	A* а;};#endif // B_H
  • Файл main.cpp:
# қосу «a.h»# қосу «b.h» int негізгі() {	A а;	B б;	а.б = &б;	б.а = &а;}

Есімде болса да (мысалы, A) бола алады жариялады бірнеше рет, мысалы, тікелей декларациялар сияқты, бұл тек болуы мүмкін анықталған бір рет ( Бір анықтама ережесі ).

Өзіне-өзі сілтеме жасау мысалы

Төменде форвардтық декларацияның тағы бір мысалы келтірілген, егер ол қолданбаға жұмыс уақытында объектілерді өзінен қосуға және жоюға қабілетті объектілер массиві қажет болса пайдалы болуы мүмкін:

  • Файл а:
сынып A {қоғамдық:    статикалық A *бірінші, *соңғы;    A *алдыңғы, *Келесі;    A();    ~A();};

The статикалық айнымалылар бірінші және соңғысын анықтау керек, өйткені олардың декларациясы олар үшін жад кеңістігін сақтамайды. Ескерту: статикалық айнымалылар объектіден объектке өзгермейді және осы берілген класс үшін өзгеріссіз қалады.

Оларды 0 немесе NULL деп инициализациялау керек, сондықтан біз оларды неден бастау керектігін білеміз.

  • Файл a.cpp:
# қосу «a.h»A *A::бірінші=0, *A::соңғы=0; // бұл жерде статикалық сөзді қоймаңыз, ол қате тудырадыA::A() {    егер (бірінші == 0) бірінші=бұл; // бірінші құрылған    алдыңғы = соңғы;    егер (алдыңғы != 0) алдыңғы->Келесі = бұл;    соңғы = бұл;    Келесі = 0;}A::~A() {    егер (алдыңғы != 0) алдыңғы->Келесі = Келесі;    егер (Келесі != 0) Келесі->алдыңғы = алдыңғы;}

Сондай-ақ қараңыз

Әдебиеттер тізімі

  1. ^ а б Лакос, Джон (1996-07-20). Ірі масштабты C ++ бағдарламалық жасақтамасын жобалау (1-ші басылым). Бостон: Аддисон-Уэсли. ISBN  9780201633627.
  2. ^ Мелтон, Хайден; Темперо, Эван (2007-01-12). «Java-дағы сыныптар арасындағы циклдарды эмпирикалық зерттеу». Бағдарламалық жасақтама эмпирикалық. 12 (4): 389–415. CiteSeerX  10.1.1.141.5362. дои:10.1007 / s10664-006-9033-1. ISSN  1382-3256.
  3. ^ Тәуелді емес үшін Java

Сыртқы сілтемелер