编程珠玑(2)第十三章学习笔记
来源:互联网 发布:编程谜题豆瓣 编辑:程序博客网 时间:2024/05/18 22:42
本章题目是搜索。研究的题目是这样的:在没有其他相关数据的情况下,如何存储一组数组?这个问题虽然很小,但却能引发在数据结构实现中出现的许多关键问题。
除了直接调用C++的模板库,作者提出了自己实现接口的四种解决方案。库的作用。C++标准模板库提供了一个实现起来很容易,并且维护和扩展也比较简单的通用解决方案。当遇到设计数据结构的问题时,我们的第一反应是寻求解决问题的通用工具。但本章中,专用的代码可以充分利用特定问题的性质,大大提高运行速度。
我们希望存储的整数是第十二章中产生的随机数,并按顺序存储。
方案一:
线性结构实现。我们知道线性结构有两种实现方式:数组和链表。数组的优点是查询比较方便,缺点是插入和删除需要大量的数据移动;链表恰好相反。
方案二:
二分搜索树实现。大量节省了插入时间。
方案三:
位图实现。位图结构可以节省大量的空间。
方案四:
“数组箱”实现。“数组箱”结合了链表和位向量的优点,有点儿类似散列。
下面是这四种方案实现的代码:
#include<stdio.h>#include<string.h>#include<stdlib.h>using namespace std;#define MAX 200#define MAXAMOUNT 50#define RAND_MAX 10000#define BITSPERWORD 32#define MASK 0x1F#define SHIFT 5int bigrand() { return rand()*RAND_MAX+rand(); } int randint(int l, int u) { return l+rand()%(u-l+1); } //方法一:线性表之数组实现class IntegerSetArray {public:IntegerSetArray(){n = 0;x = new int[1+MAXAMOUNT];x[0] = MAX;}void insert(int t){int i, j;for(i=0; x[i]<t;i++);if(x[i] == t)return;for(j=n; j>=i; j--)x[j+1] = x[j];x[i] = t;this->n++;}int size(){return this->n;}void report(){int i;for(i=0; i<n; i++){printf("%d ", x[i]);}printf("\n");}void genknuth(int m, int n) { int i; for(i=0; i<n; i++){ if(bigrand() % (n-i) < m){ insert(i);m--; } } }private:int n;int *x;};//方法一:线性表之链表实现class IntegerSetList{private:int n;struct node{int val;node *next;node(int v, node *p){val = v;this->next = p;}};node *head;node *sentinel;public:IntegerSetList(){this->sentinel = this->head = new node(MAX, NULL);n = 0;}void insert(int t){this->head = rinsert(t, head);}node *rinsert(int t, node *p){if(p->val < t)p->next = rinsert(t, p->next);else if(p->val > t){p = new node(t, p);n++;}return p;}void report(){node *p;for(p = head; p!=sentinel; p=p->next){printf("%d ", p->val);}printf("\n");}void genknuth(int m, int n) { int i; for(i=0; i<n; i++){ if(bigrand() % (n-i) < m){ insert(i);m--; } } }};//方法二:二分搜索树实现class IntegerBST{private:int n;struct node{int val;node *left, *right;node(int v){val = v;left = right = NULL;};};node *root;public:IntegerBST(){this->root = NULL;n = 0;}void insert(int t){root = rinsert(root, t);}node *rinsert(node *p, int t){if(p == NULL){p = new node(t);n++;}if(p->val < t)p->right = rinsert(p->right, t);else if(p->val > t)p->left = rinsert(p->left, t);return p;}void report(){traverse(this->root);printf("\n");}void traverse(node *p){if(p == NULL)return;traverse(p->left);printf("%d ", p->val);traverse(p->right);}void genknuth(int m, int n) { int i; for(i=0; i<n; i++){ if(bigrand() % (n-i) < m){ insert(i);m--; } } }};//方法三:位图向量实现class IntegerBitVector{private:int n;int *x;public:IntegerBitVector(){x = new int[1+MAX>>5];for(int i=0; i<MAX; i++){clear(i);}n = 0;}void insert(int t){if(test(t))return;set(t);n++;}void report(){int i = 0;for(i=0; i<MAXAMOUNT; i++){if(test(i)){printf("%d ", i);}}}void set(int i){x[i>>SHIFT] |= (1<<(i & MASK));}void clear(int i){x[i>>SHIFT] &= ~(1<<(i & MASK));}int test(int i){return x[i>>SHIFT] & (1<<(i & MASK));}void genknuth(int m, int n) { int i; for(i=0; i<n; i++){ if(bigrand() % (n-i) < m){ insert(i);m--; } } }};//方法四:结合链表和位向量的优点的箱数组实现class IntegerSetBins{private:int n, bins;struct node{int val;node *next;node(int v, node *p){val = v;next = p;};};node **bin, *sentinel;public:IntegerSetBins(){bins = MAXAMOUNT;n = 0;bin = new node*[bins];sentinel = new node(MAX, NULL);for(int i=0; i<bins; i++){bin[i] = sentinel;}}void insert(int t){int i;i = t/(1+MAX/bins);bin[i] = rinsert(t, bin[i]);}node *rinsert(int t, node *p){if(p->val < t)p->next = rinsert(t, p->next);else if(p->val > t){p = new node(t, p);n++;}return p;}void report(){int i, j;for(i=0; i<bins; i++){for(node *p=bin[i]; p!=sentinel; p=p->next){printf("%d ", p->val);}}printf("\n");}void genknuth(int m, int n) { int i; for(i=0; i<n; i++){ if(bigrand() % (n-i) < m){ insert(i);m--; } } }};int main(){int m, n;IntegerSetArray isa = IntegerSetArray();IntegerSetList isl = IntegerSetList();IntegerBST ibst = IntegerBST();IntegerSetBins isb = IntegerSetBins();IntegerBitVector ibv = IntegerBitVector();printf("Input m and n:\n");scanf("%d%d", &m, &n);isa.genknuth(m, n);isa.report();isl.genknuth(m, n);isl.report();ibst.genknuth(m, n);ibst.report();isb.genknuth(m, n);isb.report();ibv.genknuth(m, n);ibv.report();return 0;}
- 编程珠玑(2)第十三章学习笔记
- 编程珠玑(2)第二章学习笔记
- 编程珠玑(2)第五章 学习笔记
- 编程珠玑(2)第八章学习笔记
- 编程珠玑(2)第九章学习笔记
- 编程珠玑(2)第十章学习笔记
- 编程珠玑(2)第十一章学习笔记
- 编程珠玑(2)第十二章学习笔记 取样问题
- 编程珠玑(2)第十四章学习笔记之“堆”
- 编程珠玑(2)第十五章学习笔记
- 【编程珠玑】第十三章 搜索
- 编程珠玑第十三章----查找
- 编程珠玑(2)第六章笔记
- 编程珠玑学习笔记
- 编程珠玑学习笔记
- 编程珠玑第12章(取样问题)学习笔记
- 编程珠玑 第十三章 习题4 思考
- 编程珠玑笔记2
- 使用Shell脚本对Linux系统和进程资源进行监控
- jsp/Servlet
- zoj 2770 Burn the Linked Camp 差分约束
- Java Service Wrapper使用中的问题
- 一款搜索音乐的API
- 编程珠玑(2)第十三章学习笔记
- objective-c 中三种产生随机数的方法
- 浅谈机械磁盘相关概念:磁道、柱面、扇区、寻道时间、旋转延迟、数据传输时间
- 九度OJ 1031 xxx定律
- Android开发中Activity切换导致的onCreate重复执行的问题
- 使用SendMessage模拟某一按钮的点击事件
- android 4.0横屏重复调用onCreate()函数
- Java的内存回收机制
- SQL*Loader