DatovΘ struktury a algoritmy (8.)

V poslednφm dφlu jsme ukonΦili povφdßnφ o datov²ch strukturßch, kterΘ byly implementovßny p°es pole, co₧ nßs v n∞kter²ch ohledech omezovalo. Dnes se zaΦneme v∞novat datovΘ struktu°e seznamu, kterß je zalo₧ena na dynamickΘ alokaci pam∞ti.

8.1 DatovΘ struktury

8.1.1 Seznam

    Sice u₧ jsme se seznamem pracovali, ale neuvedli jsme si jeho charakteristickΘ vlastnosti:

    NßÜ seznam (a¥ u₧ pole, fronta nebo zßsobnφk) implementovan² pomocφ polφ byl statick², museli jsme tedy u₧ p°φmo v programu urΦit kolik bude mφt maximßln∞ prvk∙. Toto omezenφ odstranφme vyu₧itφm spojovΘho seznamu, jeho₧ jednotlivΘ prvky jsou vytvß°eny p°φmo za b∞hu programu. Proto₧e alokace pam∞ti probφhß p°φmo za b∞hu programu, mß jistΘ nev²hody. Za prvΘ je pam∞¥ov∞ nßroΦn∞jÜφ, proto₧e ka₧d² prvek musφ obsahovat ukazatel na dalÜφ prvek (nßsledovnφka, pop°. i p°edch∙dce) a funkce pro alokaci pam∞ti nepat°φ k funkcφm, kterΘ by byly zrovna mßlo Φasov∞ nenßroΦnΘ. Druhou nev²hodou pak je, ₧e na k-t² prvek se m∙₧eme dostat pouze pr∙chodem p°es prvky le₧φcφ p°ed nφm (nebo za nφm, pokud je obousm∞rn²) nelze pou₧φt v²hod indexovßnφ jako u polφ. Poslednφ pak je, ₧e je trochu nßroΦn∞jÜφ pro implementaci.

    My se nynφ budeme v∞novat lineßrnφmu spojovΘmu seznamu a to konkrΘtn∞ jednosm∞rnΘmu - ka₧d² prvek bude obsahovat ukazatel pouze na svΘho nßsledovnφka.

8.1.2 Lineßrnφ spojov² seznam (jednosm∞rn²)

    Je to tedy posloupnost objekt∙ (struktur, t°φd) stejnΘho typu, kterΘ jsou se°azeny do °ady. Jeho funkce je podobnß funkci jednorozm∞rnΘho pole, ovÜem ji₧ s v²Üe uvedenou nev²hodou: nejde indexovat, prochßzφme tedy v₧dy od zaΦßtku. Grafickß reprezentace tΘto datovΘ struktury:

    Malß spodnφ Φßst prvku zobrazuje datov² prvek obsa₧en² ve struktu°e, kter² odkazuje na nßsledujφcφ prvek. Velkß Φßst pak zobrazuje data. Seznam je ukonΦen prvkem, ve kterΘm jsou platnß data, ale ukazatel na dalÜφ prvek mß hodnotu NULL. Druhou mo₧nostφ je zarß₧ka, se kterou se setkßme my. Zarß₧ka neobsahuje data vklßdanß u₧ivatelem, uvidφme jak se ale dß vyu₧φt pro ulehΦenφ n∞kter²ch operacφ:

    Implementovat budeme dv∞ t°φdy a to t°φdu CJednPrvek, kterß bude, jak u₧ nßzev °φkß, prvkem jednosm∞rnΘho lineßrnφho spojovΘho seznamu. Druhou t°φdou bude CLinSpojSeznam, obsahujφcφ jako Φlenskou prom∞nnou typu CJednPrvek urΦujφcφ zaΦßtek seznamu. Proto₧e do seznamu budeme jist∞ p°idßvat prvky, tak se nßm bude hodit i prom∞nnß ukazujφcφ na jeho konec. Pokud by tam nebyla, tak bychom p°i ka₧dΘm vlo₧enφ prvku na konec seznamu museli projφt cel² seznam (tedy vÜech N prßv∞ ulo₧en²ch prvk∙):

#ifndef LINSPOJSEZNAM_H
    #define LINSPOJSEZNAM_H

    template<class T> class CJednPrvek {
    private:
        T m_Data;
        CJednPrvek<T> *m_lpDalsi;
    public:
        CJednPrvek() : m_lpDalsi(NULL) { }
        CJednPrvek(const T& _Data) : m_Data(_Data), m_lpDalsi(NULL) { }

        T VratData() const { return m_Data; }
        CJednPrvek<T>* VratDalsi() const { return m_lpDalsi; }
        void NastavDalsi(CJednPrvek<T> *_lpDalsi) { m_lpDalsi = _lpDalsi; }
        void NastavData(const T& _Data) { m_Data = _Data; }
    };

    template<class T> class CLinSpojSeznam {
    private:
        CJednPrvek<T> *m_lpPrvni, *m_lpPosledni;
    public:
        CLinSpojSeznam() : m_lpPrvni(NULL), m_lpPosledni(NULL) { }
        ~CLinSpojSeznam() { if(!JePrazdny()) { SmazSeznam(); } }

        bool VytvorPrazdny();

        bool SmazSeznam();

        bool JePrazdny() { return (m_lpPrvni == m_lpPosledni); }

        bool VlozNaKonec(const T& _Data);
// vklada na konec seznamu

        void Vypis();
    };

    T°φda CJednPrvek je jasnß, zapouzd°uje pouze nßÜ datov² prvek a ukazatel na dalÜφ objekt CJednPrvek. T°φda CLinSpojSeznam pak pro zaΦßtek obsahuje metody pro zßkladnφ prßci se seznamem, slo₧it∞jÜφ metody budeme p°idßvat postupn∞:

    template<class T> bool CLinSpojSeznam<T>::VytvorPrazdny()
    {
        bool bNavrat = false;

        m_lpPrvni = new CJednPrvek<T>;
// konstruktor automaticky nastavi m_lpDalsi na NULL, jinak bychom to museli udelat my

        if(m_lpPrvni)
        {
            m_lpPosledni = m_lpPrvni;
// zacatek je i koncem
            bNavrat = true;
        }

        return bNavrat;
    }

    K tomuto kousku k≤du je nutnΘ dodat, ₧e aΦkoliv vytvß°φme prßzdn² seznam, obsahuje tento hned jeden prvek. Tento specißlnφ prvek budeme naz²vat zarß₧kou a s v²hodou ji vyu₧ijeme p°i prochßzenφ pole za ·Φelem nalezenφ prvku s urΦitou hodnotou. Na tento prvek bude v₧dy ukazovat Φlenskß prom∞nnß t°φdy CLinSpojSeznam::m_lpPosledni. Proto₧e prßzdn² seznam je celkem k niΦemu, ukß₧eme si jak ulo₧it prvek na konec seznamu:

    template<class T> bool CLinSpojSeznam<T>::VlozNaKonec(const T& _Data)
    {
        bool bNavrat = false;

        if(m_lpPosledni)
// pokud existuje zarazka
        {
           
// vytvorime novy konec (prazdny prvek)- musime dynamicky
            CJednPrvek<T> *lpNovy= new CJednPrvek<T>;

           
// ulozime data do posledniho prvku - zarazky
            m_lpPosledni->NastavData(_Data);
            m_lpPosledni->NastavDalsi(lpNovy);

            m_lpPosledni = lpNovy;
// poslednim prvkem je nove vytvorena zarazka

            bNavrat = true;

           
// pozor prvek novy zde nesmime uvolnit !!!
        }

        return bNavrat;
    }

    Zde vyu₧ijeme toho, ₧e poslednφ prvek je zarß₧ka, kterß neobsahuje pro nßs zajφmavß data. Ulo₧φme tedy data p°edanß jako parametr do tΘto zarß₧ky a vytvo°φme nov² prßzdn² prvek - novou zarß₧ku. Alokaci provedeme dynamicky, proto₧e staticky alokovan² prvek by zanikl po zavolßnφ p°φkazu return. PotΘ nastavφme ukazatel v p∙vodnφ zarß₧ce tak, aby ukazoval na nov∞ vytvo°enou zarß₧ku a koneΦn∞ posuneme ukazatel na zarß₧ku na nov² konec. Toto vÜe provedeme jen za p°edpokladu, ₧e existuje zarß₧ka, tedy seznam byl korektn∞ vytvo°en. Nynφ si ukß₧eme funkci pro v²pis seznamu:

    template<class T> void CLinSpojSeznam<T>::Vypis()
    {
        cout << "Stav seznamu: ";

        if(!JePrazdny())
        {
            CJednPrvek<T> *lpVypisovany= m_lpPrvni;

            while(lpVypisovany!= m_lpPosledni)
// data zarazky pro nas nejsou dulezita
            {
                cout << lpVypisovany->VratData() << " ";

                lpVypisovany= lpVypisovany->VratDalsi();
            }
        }
        else
        {
            cout << " PRAZDNY";
        }

        cout << endl;
    }

    Pokud seznam nenφ prßzdn² (tedy ukazatele m_lpPrvni a m_lpPosledni jsou r∙znΘ), vyu₧ijeme cyklu while a pomocnΘ prom∞nnΘ vypisovany k vypsßnφ obsahu vÜech prvk∙, samoz°ejm∞ krom∞ zarß₧ky. A proto₧e jsme alokovali dynamicky pam∞¥, musφme sezname korektn∞ zruÜit:

    template<class T> bool CLinSpojSeznam<T>::SmazSeznam()
    {
        bool bNavrat = false;

        if(!JePrazdny())
        {
            CJednPrvek<T> *lpMazany;

            while(m_lpPrvni)
            {
                lpMazany= m_lpPrvni;
                m_lpPrvni = m_lpPrvni->VratDalsi();
// posun na dalsi prvek
                delete lpMazany;
            }

            bNavrat = true;
        }

        return bNavrat;
    }

    Seznam uvolnφme pr∙chodem od zaΦßtku postupn²m odmazßvßnφm prvk∙ v po°adφ, ve kterΘm byly ulo₧eny do seznamu. Tφm mßme naÜi zßkladnφ kostru, kterou budeme dßle rozÜi°ovat o u₧iteΦnΘ funkce. Nejprve vyhledßvßnφ prvku:

    template<class T> CJednPrvek<T>* CLinSpojSeznam<T>::Najdi(const T& _Data)
    {
        CJednPrvek<T> *lpNavrat = NULL;
// implicitne nenalezeno

        if(!JePrazdny())
        {
            CJednPrvek<T> *lpZkoumany= m_lpPrvni;

            m_lpPosledni->NastavData(_Data);
// ulozime data do zarazky
            while(lpZkoumany->VratData() != _Data)
            {
                lpZkoumany= lpZkoumany->VratDalsi();
            }

            if(lpZkoumany!= m_lpPosledni)
            {
                lpNavrat = lpZkoumany;
// nalezli jsme prvek
            }
        }

        return lpNavrat;
    }

    Zde jsme s v²hodou vyu₧ili zarß₧ky, do kterΘ ulo₧φme hledanou hodnotu. Potom mßme jistotu, ₧e tuto hodnotu nalezneme v₧dy a nemusφme se zab²vat tφm, jestli jsme dosßhli konce seznamu. Ukß₧eme si, jak by vypadala metoda p°epsanß pro ov∞°ovßnφ konce pole:

    template<class T> CJednPrvek<T>* CLinSpojSeznam<T>::NajdiBezZarazky(const T& _Data)
    {
        CJednPrvek<T> *lpNavrat = NULL;
// implicitne nenalezeno

        if(!JePrazdny())
        {
            CJednPrvek<T> *lpZkoumany= m_lpPrvni;

            while(lpZkoumany!= m_lpPosledni )
            {
                if(_Data == lpZkoumany->VratData())
                {
                    lpNavrat = lpZkoumany;
                    break;   
// preruseni cyklu
                }

                lpZkoumany= lpZkoumany->VratDalsi();   
// posun na dalsi prvek
            }
        }

        return lpNavrat;
    }

    Vidφme, ₧e zarß₧ka je velice u₧iteΦnß v∞c. UÜet°φme si toti₧ nejen problΘmy  s pracφ s ukazateli, ale uÜet°φme si tΘ₧ ov∞°ovßnφ podmφnky na platnost ukazatele v ka₧dΘm kroku. Ob∞ tyto metody vracφ pouze prvnφ v²skyt danΘho prvku. P°idßnφm parametru typu unsigned integer a jednoduchou ·pravou metody bez zarß₧ky lze upravit hledßnφ pro nalezenφ k-tΘho prvku. Ukß₧eme si verzi bez zarß₧ky:

    template<class T> CJednPrvek<T>* CLinSpojSeznam<T>::Najdi(const T& _Data, unsigned int _k)
    {
        CJednPrvek<T> *lpNavrat = NULL; // implicitne nenalezeno

        if(!JePrazdny())
        {
            CJednPrvek<T> *lpZkoumany= m_lpPrvni;

            while((lpZkoumany!= m_lpPosledni) && (_k != 0))
            {
                if(_Data == lpZkoumany->VratData())
                {
                    if(--_k == 0)
                    {
                        lpNavrat = lpZkoumany;
                        break;
                    }
                }

                lpZkoumany= lpZkoumany->VratDalsi();
            }
        }

        return lpNavrat;
    }

    V tomto p°φpad∞ je u₧itφ zarß₧ky slo₧it∞jÜφ. Nynφ metoda pro vlo₧enφ dat na zaΦßtek lineßrnφho spojovΘho seznamu:

    template<class T> bool CLinSpojSeznam<T>::VlozNaZacatek(const T& _Data)
    {
        bool bNavrat = false;

        if(m_lpPrvni)
// pokud je pred co vkladat
        {
            CJednPrvek<T> *lpNovy = new CJednPrvek<T>(_Data);

            if(lpNovy)
            {
                lpNovy->NastavDalsi(m_lpPrvni);
// novy prvek bude ukazovat na puvodni zacatek

                m_lpPrvni = lpNovy;
// a posuneme novy zacatek

                bNavrat = true;
            }

           
// opet novy prvek nemazeme !!!
        }

        return bNavrat;
    }

#endif

    Pouze vytvo°φme nov² prvek a zapojφme zep°edu do seznamu a upravφme ukazatel na prvnφ prvek.

    K≤d naleznete v sekci Downloads (projekt Seznam1).

8.2 Co bude p°φÜt∞

    Pro dneÜek jsme si ukßzali zßkladnφ operace pro prßci s lineßrnφm spojov²m seznamem, v p°φÜtφm dφlu p°idßme dalÜφ metody. Jak bylo slφbeno v dφlech v∞novan²ch front∞ a zßsobnφku, povφme si jak implementovat tyto dv∞ datovΘ struktury pomocφ lineßrnφho spojovΘho seznamu. Implementovat budeme takΘ frontu obousm∞rnou. A mo₧nß u₧ p°φÜt∞ si povφme n∞co o dynamicky vytvß°en²ch stromech.

PřφÜtě nashledanou.

Ond°ej BuriÜin