zoukankan      html  css  js  c++  java
  • 代码分析实例1

    /*
    从一个简单的实例开始:对考试结果进行统计分析(及格率)
    
    int main(int argc, char *argv[])
    {
        float scores[STUDENT_COUNT];
        int passed = 0;
        //initialize scores here...
        for (int i = 0; i != STUDENT_COUNT; i++)
        {
            if (scores[i] >= 60)
            {
                passed++;
            }
        }
        cout << "passed rate=" << (float)passed/STUDENT_COUNT << endl;
        return EXIT_SUCCESS;
    }
    
    责任分解:把分析单独作为一个功能
    void analyze(float *scores, int student_count)
    {
        int passed = 0;
        for (int i = 0; i != student_count; i++)
        {
            if (scores[i] >= 60)
                passed++;
        }
    
        cout << "passing rate = " << (float)passed / student_count << endl;
    }
    
    备注:以上传进去的参数是指针,是否可以使用链表代替数组?
    如果成绩用单项非循环链表呢?
    struct Student
    {
        float score;
        Student* next;
    };
    //..
    Student *head;
    
    //如果成绩用单项非循环链表存储?
    void analyze(Student *scores)
    {
        int passed = 0, count = 0;
        for (Student *p = scores; p != NULL; p = p->next)
        {
            if (p->score >= 60)
                passed++;
            count++;
        }
    
        cout << "passing rate=" << (float)passed/count << endl;
    }
    //遍历:变与不变
    //需要遍历所有学生的成绩(不变),不希望绑定在某种存储方式下(变)
    //分离变与不变(存储)与不变(访问)
    //把“访问”设计为一个接口,针对不同的存储完成这个接口的不同的实现
    for (int i = 0; i != STUDENT_COUNT; i++)
    {
    //access item by p
    }
    
    for (Student *p = scores; p != NULL; p = p->next)
    {
    //access item by p
    }
    
    迭代器:把访问设计成为一个接口
    class Iterator
    {
    public:
        virtual ~Iterator() {}
        virtual bool operator!=(const Iterator &other) const = 0;
        virtual const Iterator& operator++() = 0;
        virtual const Iterator& operator++(int) = 0;
        virtual float& operator*() const = 0;
        virtual float* operator->() const = 0;
        bool operator==(const Iterator &other) const
        {
            return !(*this != other);
        }
    
    };
    
    使用迭代器作为参数传递
    void analyze(Iterator *begin, Iterator* end)
    {
        int passed = 0, count = 0;
        for    (Iterator* p = begin; *p != *end; (*p)++)
        {
            if (**p >=60)
                passed++;
            count++;
        }
        cout << "passing rate" << (float)passed/count << endl;
    }
    //需要给存储对象一个约束:能够返回代表头部和尾部的迭代器
    //使用“左边闭右开区间,即[begin,end)
    class Collection
    {
    public:
        virtual ~Collection() {}
        virtual Iterator* begin() const = 0;
        virtual Iterator* end() const = 0;
        virtual int size() = 0;
    };
    
    int main(int argc, char *argv[])
    {
        Collection *collection;
        //initialize collection here...
        analyze(collection->begin(), collection->end());
    }
    
    //实现基于数组的集合Collection
    class ArrayCollection:public Collection
    {
        friend class ArrayIterator;
        float *_data;
        int _size;
    public:
        ArrayCollection():_size(10){_data = new float0[_size];}
        ArrayCollection(int size,float* data):_size(size)
        {
            _data = new float[_size];
            for    (int i = 0; i < size; i++)
                *(_data+i) = *(data+i);
        }
        ~ArrayCollection() {delete[] _data;}
        int size() {return _size;}
        Iterator* begin() const;
        Iterator* end() const;
    };
    
    Iterator* ArrayCollection::begin() const
    {
        return new ArrayIterator(_data,0);
    }
    
    Iterator* ArrayCollection::end() const
    {
        return new ArrayIterator(_data,_size);
    }
    
    //实现基于数组的迭代器Iterator
    class ArrayIterator:public Iterator
    {
        float *_data;
        int _index;
    public:
        ArrayIterator(float *data,int index):_data(data),_index(index)
        {}
        ArrayIterator(const ArrayIterator& other):_data(other._data),_index(other._index)
        {}
        ~ArrayIterator(){}
        const Iterator& operator++();
        const Iterator& operator++(int);
        float& operator*() const;
        float* operator->() const;
        bool operator!=(const Iterator &other) const;
    };
    
    const Iterator& ArrayIterator::operator++()
    {
        _index++;
        return *this;
    }
    
    const Iterator& ArrayIterator::operator(int)
    {
        _index++;
        return ArrayIterator(_data,_index - 1);
    }
    
    float& ArrayIterator::operator*() const
    {
        return *(_data + _index);
    }
    
    float* ArrayIterator::operator->() const
    {
        return (_data + _index);
    }
    
    bool ArrayIterator::operator!=(const Iterator &other) const
    {
        return (_data != ((ArrayIterator*)(&other))->data ||
        _index != ((ArrayIterator*)(&other))->_index);
    }
    
    
    analyze() &main()
    void analyze(Iterator* begin, Iterator* end)
    {
        int passed = 0, count = 0;
        for    (Iterator* p = begin; *p != *end; (*p)++)
        {
            if (**p >= 60)
                passed++;
            count++;
        }
        cout << "passing rate=" << (float)passed/count << endl;
    }
    
    int main(int argc, char **argv[])
    {
        float scores[] = {90,20,40,40,20,60,70,30,90,100};
        Collection *collection = new ArrayCollection(10,scores);
    
        analyze(collection->begin(), collection->end());
        system("PAUSE");
        return EXIT_SUCCESS;
    }
    
    //总结:迭代器模式
    //提供一种方法顺序访问一个聚合对象中各个元素,而又不需要暴露该对象的内部表示,与对象的内部表示无关(数组还是链表)
    //
    for (Iterator p = begin; p != end; p++)
    {
    //do something with object *p;
    }
    
    
    //另一种常见的迭代器模式
    Iterator it = collection.iterator();
    while(it.hasNext())
    {
        Object object = it.next();
        //do something with object;
    }
    
    变与不变
    不变
    产生迭代器的方法不变
    迭代器遍历集合的接口不变
    变:
    集合的存储方式可变
    迭代器遍历集合的具体实现可变
    实现了遍历与存储方法的隔离;
    更进一步:把analyze看做一个算法,我们实现了算法和数据存储的隔离
    算法--迭代器--数据表示
    算法的通用化
    设计一系列通用算法
    max,min,sort,count,count_if,find...
    模板:在编写代码时候留出一些可变的部分(类型),这些部分在使用前必须作出指明,这样,我们就可以使用通用的
    算法和数据结构,然后再在使用时予以实例化;
    使用模板技术实现泛型;
    
    通用算法:
    template <class _iterator>
    void analyze(_iterator begin,_iterator end)
    {
        int passed = 0, count = 0;
        for    (_iterator p = begin; p != end; p++)
        {
            if (*p >= 60)
                passed++;
            //...
        }
    }
    
    数组容器与迭代器
    template <class T>
    class ArrayCollection
    {
        T* _data;
        int _size;
    public:
        ArrayCollection():_size(10){_data = new T[_size];}
        ArrayCollection(int size):_size(size){_data = new T[_size];}
        ArrayCollection(int size,T* data):_size(size)
        {
            _data = new T[_size];
            for    (int i = 0; i < size; i++)
            {
                *(_data + i) = *(data + i);
            }
        }
        ~ArrayCollection() {delete[] _data;}
        T* begin() {return _data;}
        T* end() {return (_data + _size);}
    };
    
    链表与其迭代器
    template <class T>
    struct LinkedListNode
    {
        T _data;
        LinkedListNode *next;
        LinkedListNode():next(NULL){}
        LinkedListNode(T data):_data(data),next(NULL){}
    };
    
    template <class T>
    struct LinkedListIterator{
        LinkedListNode<T> *pointer;
        LinkedListIterator(LinkedListNode<T> *p):pointer(p){}
        LinkedListIterator(const LinkedListIterator<T>& it):pointer(it.pointer){}
        LinkedListIterator<T>& operator++()
        {    
            pointer = pointer->next;
            return *this;
        }
        const LinkedListIterator<T> operator++(int)
        {
            LinkedListIterator<T> temp = *this;
            pointer = pointer->next;
            return temp;
        }
        T& operator*() const { return pointer->_data;}
        T& operator->() const { return &(pointer->_data);}
        bool operator!=(const LinkedListIterator<T> &other)
        {
            return pointer != other.pointer;
        }
        bool operator==(const LinkedListIterator<T> &other)
        {
            return pointer == other.pointer;
        }
    };
    
    链表容器
    template <class T>
    class LinkedListCollection
    {
        LinkedListNode<T>* _head;
    public:
        LinkedListIterator<T> begin() {return LinkedListIterator<T>(_head);}
        LinkedListIterator<T> end() {return LinkedListIterator<T>(NULL);}
    
        LinkedListCollection():_head(NULL){}
        LinkedListCollection(in size, T* data)
        {
            //...
        }
        ~LinkedListCollection()
        {
            //...
        }
    };
    
    使用算法与迭代器
    int main(int argc, char *argv[])
    {
        float scores[] = {90,....};
        ArrayCollection<float> collection2(10,scores);
        LinkCollection<float> collection1(10,scores);
    
        analyze(scores,scores + 10);
        analyze(collection1.begin(),collection1.end());
        analyze(collection2.begin(),collection2.end());
    
        system("PAUSE");
        return EXIT_SUCCESS;
    
    }
    新的类型
    考试三科都要高于60分,怎么办?
    struct Score
    {
        float value[3];
        Score(){}
        Score(float f1,float f2,float f3)
        {
            value[0] = f1;
            value[1] = f2;
            value[2] = f3;
        }
        Score& operator=(const Score& s)
        {
            value[0] = s.value[0];
            value[1] = s.value[1];
            value[2] = s.value[2];
            return *this;
        }
        bool operator>=(float pass)
        {
            return (value[0] >= pass && value[1] >= pass && value[2] >= pass);
        }
    };
    
    ostream& operator<<(ostream& out, const Score& s)
    {
        out << "{" << s.value[0] << "," << s.value[1] << "," << s.value[2] << "}";
        return out;
    }
    
    适应新的类型
    int main(int argc, char *argv[])
    {
        Score sarray[3];
        sarray[0] = Score(60,60,60);
        sarray[1] = Score(70,70,70);
        sarray[2] = Score(90,80,80);
    
        ArrayCollection<Score> collection3(3,sarray);
        LinkCollection<Score> collection4(3,sarray);
    
        analyze(collection3.begin(),collection3.end());
        analyze(collection4.begin(),collection4.end());
    
        system("PAUSE");
        return EXIT_SUCCESS;
    }
    算法实际上与“可用的操作相关”,与具体的数据类型无关;
    
    抽象结构和类模板
    除了抽象算法外,还有抽象结构:Stack,Linked list,Vector...
    这些抽象结构与存储什么数据没有关系,只与数据的存储方式和访问方式相关;
    可以借助类模板实现:
    
    template <class T>
    struct LinkNode
    {
        T _data;
        LinkNode *_next;
        LinkNode():_next(NULL){}
        LinkNode(T data):_data(data),_next(NULL){}
        LinkNode(T data,T* next):_data(data),_next(next){}
    };
    
    template <class T>
    class LinkedListCollection
    {    
        LinkedListNode<T>* _head;
    public:
        LinkedListCollection():_head(NULL){}
        ~LinkedListCollection(){clear();}
        bool empty(){return (_head == NULL);}
        void addFirst(const T& data) {_head = new LinkedListNode<T>(data,_head);}
        bool removeFirst() {
        if (_head != NULL)
            {
                LinkedListNode<T>* p = _head;
                _head = _head->_next;
                delete p;
                return true;
            }
        else
            return false;        
        }
    }
    
    T* getFirst() { return (_head != NULL) ? &(_head->_data) : NULL;}
    
    LinkedListNode<T>* lastNode()
    {
        LinkedListNode<T>* p;
        for    (p = _head; p->_next != NULL; p = p->_next);
        return p;
    }
    
    void addLast(const T& data)
    {
        if (_head != NULL)
            lastNode()->_next = new LinkedListNode<T>(data);
        else
            _head = new LinkedListNode<T>(data);
    }
    
    T* getLast() { return (_head != NULL) ? &(_lastNode()->_data : NULL);}
    
    bool removeLast()
    {    
        if (_head != NULL)
        {
            if (_head->_next != NULL)
            {
                LinkedListNode<T>* p;
                for    (p = _head; p->_next->_next != NULL; p = p->_next);
                delete p->_next;
                p->_next = NULL;
                return true;
            }
            else
            {
                delete _head;
                _head = NULL;
                return true;
            }
        }
        else
        {
            return false;
        }
    }
    
    void clear()
    {
        while(removeFirst());
    }
    
    LinkedListIterator<T> begin() {return LinkedListIterator<T>(_head);}
    LinkedListIterator<T> end() {return LinkedListIterator<T>(NULL);}
    
    内联函数:
    inline int min(int a, int b) { return (a>b) ? a:b;}
    作用:函数内联展开,避免函数调用开销用空间换时间;
    在类定义体内定义(实现)的函数缺省为内联函数;
    
    通过模板实例化使用Linked List(抽象结构)
    int main(int argc, char *argv[])
    {
        LinkedListCollection<Score> collection4;
        for    (int i = 0; i < 3; i++)
            collection4.addFirst(sarray[i]);
    
        analyze(collection4.begin(),collection4.end());
        //....
        return EXIT_SUCCESS;
    }
    
    
    备注:
    不变:算法和抽象结构的接口与实现不变,数据的访问接口不变(迭代器),数据的可用操作不变(操作符重载);
    数据的组织形式可变,数据的类型可变;
    
    实现“算法/抽象结构”与数据表示之间的分离;
    泛型编程:先实现算法,再充实数据表示(类型);
    
    
    假设算法继续变化
    某些科目的及格线不是60分
    template<class _iterator> void analyze(_iterator begin,_iterator end)
    {
        int passed = 0, count = 0;
    
        for (_iterator p = begin; p != end; p++)
        {
            if(*p >= 60)
                passed++;
            count++;
        }
        cout << "passing rate= " << (float)passed / count << endl;
    }
    
    使用函数指针把判断及格的这个函数传递进去
    template <class _iterator>
    void analyze(_iterator begin, _iterator end, bool (*isPass)(const _iterator&))
    {
        int passed = 0, count = 0;
    
        for    (_iterator p = begin; p != end; p++)
        {
            if(isPass(p))
            {
                passed++;
            }
            count++;
        }
    }
    
    使用函数指针判断及格函数
    template <class _Iterator> bool isPass(const _Iterator& p)
    {
        return (p->value[0] >= 70 && p->value[1] >= 60 && p->value[2] >= 60);
    }//注意这个函数的实现是有缺陷的,只适用于Score类型
    
    int main(int argc, char *argv[])
    {
        //...
        analyze(sarray,sarray + 3,isPass<Score *>);//???
        analyze(collection3.begin,collection3.end(),isPass<Score *>);//???
        analyze(collection4.begin,collection4.end(),isPass<LinkedListIterator<Score> >);//????
        //...
    }
    
    继续解耦
    这两个函数有什么不同吗?
    template <class _Iterator> bool isPass(const _Iterator& p)
    {
        return (p->value[0] >= 60
        && p->value[1] >= 60 && p->value[2] >= 60);
    }
    
    template <class _Iterator> bool isPass(const _Iterator& p)
    {
    return (p->value[0] >= 70
    && p->value[1] >= 60 && p->value[2] >= 60);
    }
    
    如果及格线能够被记住呢?
    Score pass(70,60,60);
    template <class _Iterator> bool isPass(const _Iterator& p)
    {
        return (p->value[0] >= pass.value[0] && 
                p->value[1] >= pass.value[1] &&
                p->value[2] >= pass.value[2]);
    }
    
    文件也可以作为函数“记住”数据的另外一种手段
    Score pass(70,60,60);
    template <class _Iterator>
    bool isPass(const _Iterator& p)
    {
        float pass1, pass2, pass3;
        ifstream is("passScore.txt");
        is >> pass1 >> pass2 >> pass3;
        return (p->value[0] >= pass1 && p->value[1] >= pass2 && p->value[2] >= pass3);
    }
    这样的话通过修改passScore.text文件可以轻松地修改及格的分数线;
    单一实例的问题依然存在; 
    问题在于函数是死的:数据 + 函数 = 对象  函数 + 数据  = 对象
    
    函数调用==()操作符,还原()操作符的本意,操作符重载;
    函数对象:
    通过对函数运算符的重载,我们可以赋予一个对象“函数”的特性,能被调用
    
    template <class _iterator,class T>
    class IsPass{
        T _pass;
    public:
        IsPass(const T& pass):_pass(pass){}
        bool operator() (const _iterator& p)
        {
            return (*p >= _pass);
        }
    };
    
    IsPass就是一个函数对象类模板
    算法的定义:
    
    template <class _iterator , class T>
    void analyze(_iterator begin, _iterator end, IsPass<_iterator,T> isPass)
    {
        int passed = 0, count = 0;
        
        for    (_iterator p = begin; p != end; p++)
            if (isPass(p))
                passed++;
            count++;
    }
    
    使用函数对象
    int main(int argc, char *argv[])
    {
        //...
        sarray[0] = Score(60,60,70);
        sarray[1] = Score(60,60,70);
        sarray[2] = Score(60,60,70);
    
        ArrayCollection<Score> collection3(3,sarray);
        LinkedListCollection<Score> collection4;
        //....
        analyze(sarray,sarray + 3 ,IsPass<Score*, Score> (Score(70,60,60)));
        analyze(collection3.begin(),collection3.end(),IsPass<Score*,Score>(Score(50,60,60)));
        analyze(collection4.begin(),collection4.end(),IsPass<LinkedListIterator<Score>, Score>(Score(60,60,60)));
    
    }
    */
    怕什么真理无穷,进一寸有一寸的欢喜。---胡适
  • 相关阅读:
    「SDOI2018」物理实验
    「SDOI 2018」战略游戏
    「CodeChef Dec13 REALSET」 Petya and Sequence 循环卷积
    关于微信卡券投放code接口报错原因
    composer update maatwebsite/excel 之后 在linux机子上出现500解决方案
    开启mysql 服务【window】
    thinkphp在linux上部署环境(500情况)
    如何推广微信小程序到企业微信
    linux 阿里云远程连接mysql
    php7以上 不支持mcrypt_module_open方法问题【微信开放平台】
  • 原文地址:https://www.cnblogs.com/hujianglang/p/7253440.html
Copyright © 2011-2022 走看看