算法笔记

特征

  • 有穷性
  • 确切性
  • 输入项
  • 输出项
  • 可行性

算法优劣评定

  • 时间复杂度
  • 空间复杂度
  • 正确性
  • 可读性
  • 健壮性

时间复杂度

  • O(N^3)
  • O(N^2)
  • O(N)
  • O(NlogN) 查找二叉树
  • O(logN)
  • O(1)
排序算法 平均时间复杂度
冒泡排序 O(n2)
选择排序 O(n2)
插入排序 O(n2)
希尔排序 O(n1.5)
快速排序 O(N*logN)
归并排序 O(N*logN)
堆排序 O(N*logN)
基数排序 O(d(n+r))

算法分析方法

  • 递归法 汉诺塔
  • 穷举法 暴力密码破解法
  • 贪心算法 加勒比海盗偷宝藏
  • 分治法 乐毅连下齐72城二分搜索
  • 动态规划法 导弹拦截
  • 迭代法 超能生的兔子
  • 回溯法 八皇后

排序算法

1、插入排序

    在要排序的一组数中,假定前n-1个数已经排好序,
现在将第n个数插到前面的有序数列中,
使得这n个数也是排好顺序的。如此反复循环,直到全部排好顺序。
  • 直接插入排序
public static void insert(int[] arr) {
    for (int i = 1; i < arr.length; i++) {
        for (int j = i; j > 0 && (arr[j]<arr[j-1]); j--) {
            int tem = arr[j-1];
            arr[j-1] = arr[j];
            arr[j] = tem;
        }
    }
}
  • 二分插入排序
public static void binaryInsert(int[] arr) {
    for (int i = 1; i < arr.length; i++) {
        int left = 0;
        int right = i-1;
        int key = arr[i];//待插入的数据
        while(left <= right){
            int mid = (left+right)/2;
            if (key < arr[mid]) {
                right = mid -1;
            }else{
                left = mid +1;
            }
        }
        for (int j = i; j > left; j--) {
            arr[j] = arr[j-1];
        }
        arr[left] = key;
    }
}
  • 希尔排序
希尔排序是非稳定排序算法.
在要排序的一组数中,根据某一增量分为若干子序列,并对子序列分别进行插入排序。
然后逐渐将增量减小,并重复上述过程。
直至增量为1,此时数据序列基本有序,最后进行插入排序。
public static void shell(int[] arr) {
    int n = arr.length;
    int h = 1;
    while(h < n/3){
        h = h*3+1;
    }
    while(h >= 1){
        for (int i = h; i < n; i++) {
            for (int j = i; j >= h && arr[j] < arr[j-h]; j-=h) {
                int tem = arr[j];
                arr[j] = arr[j-h];
                arr[j-h] = tem;
            }
        }
        h /= 3;
    }
}

2、选择排序

在长度为N的无序数组中,第一次遍历n-1个数,找到最小的数值与第一个元素交换;
第二次遍历n-2个数,找到最小的数值与第二个元素交换;
。。。
第n-1次遍历,找到最小的数值与第n-1个元素交换,排序完成。
  • 简单选择排序
public static void choose(int[] arr){
    for (int i = 0; i < arr.length; i++) {
        for (int j = i; j < arr.length; j++) {
            if (arr[j] < arr[i]) {
                int tem = arr[i];
                arr[i] = arr[j];
                arr[j] = tem;
            }
        }
    }
}
  • 堆排序
    不稳定排序
    将待排序序列构造成一个大顶堆,此时,整个序列的最大值就是堆顶的根节点。
将其与末尾元素进行交换,此时末尾就为最大值。
然后将剩余n-1个元素重新构造成一个堆,这样会得到n个元素的次小值。
如此反复执行,便能得到一个有序序列了
public static void heap(int[] arr) {
    int n = arr.length;
    for (int i = n/2-1; i >= 0; i--) {
        //这一步不是单单地找出最大的节点,而是形成大根堆,对后面有帮助
        adjustHeap(arr,i,n);  
    }
    for (int j = n-1; j > 0; j--) {
        int tem = arr[0];
        arr[0] = arr[j];
        arr[j] = tem;

        adjustHeap(arr, 0, j);
    }
}

private static void adjustHeap(int[] arr, int parent, int length) {
    int tem = arr[parent];
    int child = 2*parent+1;  //左孩子
    while(child < length){
        if (child+1 < length && arr[child] < arr[child+1]) {//右孩子存在,并且右孩子大于左孩子
            child++;  //从左孩子切换到右孩子
        }
        if (tem > arr[child]) {
            break;  //如果当前元素大于左右孩子节点,则跳出
        }
        arr[parent] = arr[child]; //从左右孩子中选出较大的给父节点
        //把小元素向下调整
        parent = child;
        child = 2*child + 1;
    }       
    arr[parent] = tem;
}

3、交换排序

交换排序的基本方法是:两两比较待排序记录的排序码,
交换不满足顺序要求的偶对,直到全部满足位置。
常见的冒泡排序和快速排序就属于交换类排序。
  • 冒泡排序
从数组中第一个数开始,依次遍历数组中的每一个数,通过相邻比较交换,每一轮循环下来找出剩余未排序数的中的最大数并”冒泡”至数列的顶端。
public static void Bubble(int[] arr) {
    for (int i = 0; i < arr.length; i++) {
        for (int j = 0; j < arr.length-i-1; j++) {
            if (arr[j]>arr[j+1]) {
                int tem = arr[j];
                arr[j] = arr[j+1];
                arr[j+1] = tem;
            }
        }
    }
}
  • 快速排序
1)从待排序的n个记录中任意选取一个记录(通常选取第一个记录)为分区标准;
(2)把所有小于该排序列的记录移动到左边,把所有大于该排序码的记录移动到右边,
中间放所选记录,称之为第一趟排序;
(3)然后对前后两个子序列分别重复上述过程,直到所有记录都排好序。
public static void quick(int[] arr) {
    if (arr.length > 0) {
        quick(arr,0,arr.length-1);
    }
}

private static void quick(int[] arr, int lo, int hi) {
    if (lo >= hi) {
        return;
    }
    int first = lo;
    int  last = hi;
    int key = arr[first];
    while(first < last){
        while(first < last && arr[last] >= key){
            last--;
        }
        arr[first] = arr[last];
        while(first < last && arr[first] <= key){
            first++;
        }
        arr[last] = arr[first];   
    }
    arr[first] = key;
    quick(arr, lo, first-1);
    quick(arr, first+1, hi);        
}

4、归并排序

归并排序是利用归并的思想实现的排序方法,
该算法采用经典的分治策略(分治法将问题分(divide)成一些小的问题然后递归求解,
而治(conquer)的阶段则将分的阶段得到的各答案"修补"在一起,即分而治之)。
public static void merge(int[] arr) {
    int tem[] = new int[arr.length]; 
    merge(arr,0,arr.length-1,tem);
}

private static void merge(int[] arr, int left, int right, int[] tem) {
    if (left < right) {
        int mid = (left+right)/2;
        merge(arr,left,mid,tem);
        merge(arr,mid+1,right,tem);
        mergeSub(arr,left,mid,right,tem);
    }
}

private static void mergeSub(int[] arr, int left, int mid, int right,
                             int[] tem) {
    int i = left;
    int j = mid +1;
    int third = 0;
    while(i <= mid && j <= right){
        if (arr[i] <= arr[j]) {
            tem[third++] = arr[i++];
        }else{
            tem[third++] = arr[j++];
        }           
    }
    while(i <= mid){
        tem[third++] = arr[i++];
    }
    while(j <= right){
        tem[third++] = arr[j++];
    }
    //将tem中的元素全部拷贝到原数组中
    third = 0;
    while(left <= right){
        arr[left++] = tem[third++];
    }
}

5、基数排序

基数排序是一种借助多关键字排序的思想对单逻辑关键字进行排序的方法。
它是一种稳定的排序算法。
多关键字排序中有两种方法:最高位优先法(MSD)和最低位优先法(LSD)。
public static void radix(int[] arr) {
    if (arr.length <= 0) {
        return;
    }
    int times = 0,max=arr[0]; //times为数组中最大位数
    for (int i = 0; i < arr.length; i++) {
        if (max < arr[i]) {
            max = arr[i];
        }
    }
    while(max > 0){
        max = max/10;
        times++;
    }
    //-------------------------
    int[][]temp = new int[10][arr.length]; //数组的第一维表示可能的余数0-9
    int[]num = new int[10]; //数组num[i]用来表示该位是i的数的个数
    for (int i = 0; i < times; i++) {
        for (int j = 0; j < arr.length; j++) {
            int x = arr[j]%(int)Math.pow(10, i+1)/(int)Math.pow(10, i);
            //x 为当前数所在的当前位 0-9
            temp[x][num[x]++] = arr[j];
        }
        int t = 0;
        for (int j = 0; j < 10; j++) {
            if (num[j] != 0) {
                for(int k = 0; k < num[j]; k++){
                    arr[t++] = temp[j][k];
                }
            }
            num[j] = 0;
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容

  • 课程介绍 先修课:概率统计,程序设计实习,集合论与图论 后续课:算法分析与设计,编译原理,操作系统,数据库概论,人...
    ShellyWhen阅读 2,270评论 0 3
  • 概述:排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    每天刷两次牙阅读 3,729评论 0 15
  • 概述排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的...
    Luc_阅读 2,259评论 0 35
  • 一些概念 数据结构就是研究数据的逻辑结构和物理结构以及它们之间相互关系,并对这种结构定义相应的运算,而且确保经过这...
    Winterfell_Z阅读 5,712评论 0 13
  • 外面开始嘈杂起来,有机械铠甲走动的声音。 “报告!“有个士兵高声喊道。 “进来!“梅特森说。 “有什么事吗?“ “...
    活捉一只胖墩阅读 274评论 0 1