zoukankan      html  css  js  c++  java
  • 数据结构与算法

    1.B+树


    2.快速排序,堆排序,插入排序(其实八大排序算法都应该了解)

    Java内部八种排序
    1.    冒泡排序
    基本思想:在要排序的一组数中,对当前还未排好序的范围内的全部数,自上而下对相邻的两个数依次进行比较和调整,让较大的数往下沉,较小的往上冒。即:每当两相邻的数比较后发现它们的排序与排序要求相反时,就将它们互换
    /**
         * 冒泡排序
         * @param data    要排序的数组
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, boolean reverse) {
            if (data.length == 1) {
                return;
            }
            for (int i = 0; i < data.length - 1; i++) {
                int tmp = 0;
                for (int j = 0; j < data.length - i - 1; j++) {
                    if (reverse) {  //从小到大(ture)
                        if (data[j] >= data[j+1]) {
                            tmp = data[j];
                            data[j] = data[j +1 ];
                            data[j+1] = tmp;
                        }
                    } else {    //从大到小(false)
                        if (data[j] <= data[j+1]) {
                            tmp = data[j+1];
                            data[j+1] = data[j];
                            data[j] = tmp;
                        }
                    }
                }
            }
    }
    2.    堆排序
    基本思想:堆排序是一种树形选择排序,是对直接选择排序的有效改进。
    堆的定义如下:具有n个元素的序列(h1,h2,...,hn),当且仅当满足(hi>=h2i,hi>=2i+1)或(hi<=h2i,hi<=2i+1)(i=1,2,...,n/2)时称之为堆。在这里只讨论满足前者条件的堆。由堆的定义可以看出,堆顶元素(即第一个元素)必为最大项(大顶堆)。完全二叉树可以很直观地表示堆的结构。堆顶为根,其它为左子树、右子树。初始时把要排序的数的序列看作是一棵顺序存储的二叉树,调整它们的存储序,使之成为一个堆,这时堆的根节点的数最大。然后将根节点与堆的最后一个节点交换。然后对前面(n-1)个数重新调整使之成为堆。依此类推,直到只有两个节点的堆,并对它们作交换,最后得到有n个节点的有序序列。从算法描述来看,堆排序需要两个过程,一是建立堆,二是堆顶与堆的最后一个元素交换位置。所以堆排序有两个函数组成。一是建堆的渗透函数,二是反复调用渗透函数实现排序的函数 
    /**
         * 堆排序
         * @param data    要排序的数组
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, boolean reverse) {
            if (data.length == 1) {
                return;
            }
            for (int i = 0; i < data.length; i++) {
                //建堆
                buildHeap(data, 0, data.length -1 - i, reverse);
                int tmp = data[0];
                data[0] = data[data.length - 1 - i];
                data[data.length - 1 - i] = tmp;
            }
        }
    
        /**
         * 将指定开始和结束段的数据建堆
         * @param data
         * @param beginIndex
         * @param endIndex
         * @param reverse
         */
        public static void buildHeap(int[] data, int beginIndex, int endIndex, boolean reverse) {
            if (beginIndex >= endIndex) {
                return;
            }
            for (int i = (endIndex + beginIndex - 1) / 2; i >= beginIndex; i--) {
                int cur = i;
                if (reverse) {   //大顶堆,用来从小到大排序
                    //发生交换之后需要检查孙子节点,重孙子节点...
                    while (2 * cur + 1 <= endIndex) {
                        int biggerChildIndex = 2 * cur + 1;
                        if (biggerChildIndex + 1 <= endIndex) {
                            if (data[biggerChildIndex] < data[biggerChildIndex + 1]) {
                                biggerChildIndex = biggerChildIndex + 1;
                            }
                        }
                        //找到最大子节点,如果比当前节点大,就交换
                        if (data[i] < data[biggerChildIndex]) {
                            int tmp = data[i];
                            data[i] = data[biggerChildIndex];
                            data[biggerChildIndex] = tmp;
                            //准备检查孙子节点
                            cur = biggerChildIndex;
                        } else {
                            break;
                        }
                    }
                } else {    //小顶堆,用来从大到小排序
                    //发生交换之后需要检查孙子节点,重孙子节点...
                    while (2 * cur + 1 <= endIndex) {
                        int samllerChildIndex = 2 * i + 1;
                        if (samllerChildIndex + 1 <= endIndex) {
                            if (data[samllerChildIndex] > data[samllerChildIndex + 1]) {
                                samllerChildIndex = samllerChildIndex + 1;
                            }
                        }
                        //找到最小子节点,如果比当前节点小,就交换
                        if (data[i] > data[samllerChildIndex]) {
                            int tmp = data[i];
                            data[i] = data[samllerChildIndex];
                            data[samllerChildIndex] = tmp;
                            cur = samllerChildIndex;
                        } else {
                            break;
                        }
                    }
                }
            }
    }
    3.    直接插入排序
    基本思想:在要排序的一组数中,假设前面(n-1)[n>=2]个数已经是排好顺序的,现在要把第n个数插到前面的有序数中,使得这n个数也是排好顺序的。如此反复循环,直到全部排好顺序。
    /**
         *  插入排序
         * @param data  要排序的数组
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, boolean reverse) {
            if (data.length == 1) {
                return;
            }
            int tmp = 0;
            for (int i = 1; i < data.length; i++) {
                tmp = data[i];
                int n = i - 1;
                for (; n >= 0; n--) {
                    if (reverse) {   //从小到大排序
                        if (data[n] >= tmp) {
                            data[n + 1] = data[n];  //将大于当前值的数后移一个位置
                        } else {
                            break;
                        }
                    } else {    //从大到小排序
                        if (data[n] <= tmp) {
                            data[n + 1] = data[n];  //将小于当前值的数后移一个位置
                        } else {
                            break;
                        }
                    }
                }
                data[n+1] = tmp;
            }
    }
    4.    归并排序
    基本思想:归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的。然后再把有序子序列合并为整体有序序列。
    /**
         * 归并排序
         * @param data    要排序的数组
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, int left, int right, boolean reverse) {
            if (left >= right) {
                return;
            }
            int mid = (left + right) / 2;
            sort(data, left, mid, reverse);
            sort(data, mid + 1, right, reverse);
            merge(data, left, mid, right, reverse);
        }
    
        /**
         * 合并已排序好的两段
         * @param data
         * @param left
         * @param mid
         * @param right
         * @param reverse
         */
        public static void merge(int[] data, int left, int mid, int right, boolean reverse) {
            int[] tmp = new int[right - left + 1];
            int i = left;
            int j = mid + 1;
            int n = 0;
            while (i <= mid && j <= right) {
                if (reverse) {  //从小到大
                    if (data[i] <= data[j]) {
                        tmp[n++] = data[i++];
                    } else {
                        tmp[n++] = data[j++];
                    }
                } else {    //从大到小
                    if (data[i] <= data[j]) {
                        tmp[n++] = data[j++];
                    } else {
                        tmp[n++] = data[i++];
                    }
                }
            }
            while (i <= mid) {
                tmp[n++] = data[i++];
            }
            while (j <= right) {
                tmp[n++] = data[j++];
            }
            for (int k = 0; k < tmp.length; k++) {
                data[left + k] = tmp[k];
            }
    }
    5.    快递排序
    基本思想:选择一个基准元素,通常选择第一个元素或者最后一个元素,通过一趟扫描,将待排序列分成两部分,一部分比基准元素小,一部分大于等于基准元素,此时基准元素在其排好序后的正确位置,然后再用同样的方法递归地排序划分的两部分
    /**
         * 快速排序
         * @param data
         * @param left
         * @param right
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, int left, int right, boolean reverse) {
            if (left >= right) {
                return;
            }
            int index = getIndex(data, left, right, reverse);
            sort(data, left, index - 1, reverse);
            sort(data, index + 1, right, reverse);
        }
    
        /**
         * 将待排序片段调整顺序,获得"中间数据"的正确索引
         * @param data
         * @param left
         * @param right
         * @param reverse 从大到小(false)还是从小到大(ture)
         * @return
         */
        public static int getIndex(int[] data, int left, int right, boolean reverse) {
            int cur = data[left];
            int i = left;
            int j = right;
            while (i < j) {
                if (reverse) {  //从小到大
                    while (data[j] > cur && i < j) {
                        --j;
                    }
                    data[i] = data[j];
                    while (data[i] <= cur && i < j) {
                        ++i;
                    }
                    data[j]=data[i];
                } else {    //从大到小
                    while (data[j] < cur && i < j) {
                        --j;
                    }
                    data[i]=data[j];
                    while (data[i] >= cur && i < j) {
                        ++i;
                    }
                    data[j]=data[i];
                }
            }
            data[i] = cur;
            return i;
    }
    6.    基数排序
    基本思想:将所有待比较数值(正整数)统一为同样的数位长度,数位较短的数前面补零。然后,从最低位开始,依次进行一次排序。这样从最低位排序一直到最高位排序完成以后,数列就变成一个有序序列。
    /**
         * 基数排序
         * @param data    要排序的数组
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, boolean reverse) {
            if (data.length == 1) {
                return;
            }
            int max = 0;
            for (int i = 0; i < data.length; i++) { //找出最大的数据
                if (max < data[i]) {
                    max = data[i];
                }
            }
            System.out.println("the max number is :" + max);
            int radix = 1;
            ArrayList<ArrayList<Integer>> numbers = new ArrayList<ArrayList<Integer>>(10);
            for (int i = 0; i < 10; i++) {
                numbers.add(i, new ArrayList<Integer>());
            }
            while (max > radix) {
                for (int i = 0; i < data.length; i++) {
                    int index = (data[i] / radix) % 10;
                    ArrayList<Integer> list = numbers.get(index);
                    list.add(data[i]);
                    numbers.set(index, list);
                }
                resetOrder(data, numbers, reverse);
                radix = radix * 10;
            }
        }
    
        /**
         * 重新调整数组顺序
         * @param data
         * @param numbers
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void resetOrder(int[] data, ArrayList<ArrayList<Integer>> numbers, boolean reverse) {
            int n = 0;
            if (reverse) {
                for (int i = 0; i < numbers.size(); i++) {
                    ArrayList<Integer> list = numbers.get(i);
                    while(list.size()>0){
                        data[n++] = list.get(0);
                        list.remove(0);
                    }
                }
            } else {
                for (int i = numbers.size() - 1; i >= 0; i--) {
                    ArrayList<Integer> list = numbers.get(i);
                    while(list.size()>0){
                        data[n++] = list.get(0);
                        list.remove(0);
                    }
                }
            }
    }
    7.    选择排序
    基本思想:在要排序的一组数中,选出最小的一个数与第一个位置的数交换;然后在剩下的数当中再找最小的与第二个位置的数交换,如此循环到倒数第二个数和最后一个数比较为止。
    /**
         * 选择排序
         * @param data  要排序的数组
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, boolean reverse) {
            if (data.length == 1) {
                return;
            }
            for(int i=0;i<data.length-1;i++){
                int tmp=data[i];    //要初始化
                int index = i;      //要初始化
                for(int j=i;j<data.length;j++){
                    if(reverse) {   //从小到大(ture)
                        if (tmp>=data[j]){
                            tmp = data[j];  //最小值
                            index = j;
                        }
                    }else {
                        if (tmp<=data[j]){
                            tmp = data[j];  //最大值
                            index = j;
                        }
                    }
                }
                data[index] = data[i];
                data[i] = tmp;
            }
    }
    8.    希尔排序
    基本思想:算法先将要排序的一组数按某个增量d(n/2,n为要排序数的个数)分成若干组,每组中记录的下标相差d.对每组中全部元素进行直接插入排序,然后再用一个较小的增量(d/2)对它进行分组,在每组中再进行直接插入排序。当增量减到1时,进行直接插入排序后,排序完成。
    /**
         * 希尔排序
         * @param data    要排序的数组
         * @param reverse 从大到小(false)还是从小到大(ture)
         */
        public static void sort(int[] data, boolean reverse) {
            if (data.length == 1) {
                return;
            }
            for (int d = data.length / 2; d >= 1; d = d / 2) {  //组大小
                for (int k = 0; k < d; k++) {   //多少组
                    for (int n = d + k; n < data.length; n = n + d) {   //同一组
                        int cur = n;
                        while (cur - d >= 0) {  //插入排序
                            int tmp = 0;
                            if (reverse) {  //小到大(ture)
                                if (data[cur] <= data[cur - d]) {
                                    tmp = data[cur];
                                    data[cur] = data[cur - d];
                                    data[cur - d] = tmp;
                                }
                            } else {         //从大到小(false)
                                if (data[cur] >= data[cur - d]) {
                                    tmp = data[cur];
                                    data[cur] = data[cur - d];
                                    data[cur - d] = tmp;
                                }
                            }
                            cur = cur - d;
                        }
                    }
                }
            }
    
    }

    3.一致性Hash算法,一致性Hash算法的应用

  • 相关阅读:
    C#缓存初步学习
    C#性能优化杂七杂八的总结
    C#常用的加密算法:MD5、Base64、SHA1、SHA256、HmacSHA256、DES、AES、RSA
    C# RAS生成.NET公钥与私钥以及.NET公钥与私钥转Java公钥私钥类
    SqlSever查询某个表或某个库的列名称、说明、备注、注释,类型等
    常用Git命令清单
    .NET Core 3.0X及以上版本的EFCore连接MySql 8.022
    ASP.NET Core 3.X后出现 [The JSON value could not be converted to System.Nullable] 错误
    从遥感影像到土地利用转移矩阵
    利用Jenkins+SVN+Windows服务对NetCore项目实行持续集成、自动化部署CI/CD
  • 原文地址:https://www.cnblogs.com/dsgerblog/p/10563439.html
Copyright © 2011-2022 走看看