【精品计划 附录2】- 算法分析

  • 数学模型
    • 1. 近似
    • 2. 增长数量级
    • 3. 内循环
    • 4. 成本模型
  • 注意事项
    • 1. 大常数
    • 2. 缓存
    • 3. 对最坏情况下的性能的保证
    • 4. 随机化算法
    • 5. 均摊分析
  • ThreeSum
    • 1. ThreeSumSlow
    • 2. ThreeSumBinarySearch
    • 3. ThreeSumTwoPointer
  • 倍率实验

数学模型

1. 近似

N3/6-N2/2+N/3 ~ N3/6。使用 ~f(N) 来表示所有随着 N 的增大除以 f(N) 的结果趋近于 1 的函数。

2. 增长数量级

N3/6-N2/2+N/3 的增长数量级为 O(N3)。增长数量级将算法与它的具体实现隔离开来,一个算法的增长数量级为 O(N3) 与它是否用 Java 实现,是否运行于特定计算机上无关。

3. 内循环

执行最频繁的指令决定了程序执行的总时间,把这些指令称为程序的内循环。

4. 成本模型

使用成本模型来评估算法,例如数组的访问次数就是一种成本模型。

注意事项

1. 大常数

在求近似时,如果低级项的常数系数很大,那么近似的结果是错误的。

2. 缓存

计算机系统会使用缓存技术来组织内存,访问数组相邻的元素会比访问不相邻的元素快很多。

3. 对最坏情况下的性能的保证

在核反应堆、心脏起搏器或者刹车控制器中的软件,最坏情况下的性能是十分重要的。

4. 随机化算法

通过打乱输入,去除算法对输入的依赖。

5. 均摊分析

将所有操作的总成本除于操作总数来将成本均摊。例如对一个空栈进行 N 次连续的 push() 调用需要访问数组的次数为 N+4+8+16+…+2N=5N-4(N 是向数组写入元素的次数,其余都是调整数组大小时进行复制需要的访问数组次数),均摊后访问数组的平均次数为常数。

ThreeSum

ThreeSum 用于统计一个数组中和为 0 的三元组数量。

public interface ThreeSum {int count(int[] nums);
}

1. ThreeSumSlow

该算法的内循环为 if (nums[i] + nums[j] + nums[k] == 0) 语句,总共执行的次数为 N(N-1)(N-2) = N3/6-N2/2+N/3,因此它的近似执行次数为 ~N3/6,增长数量级为 O(N3)。

public class ThreeSumSlow implements ThreeSum {@Overridepublic int count(int[] nums) {int N = nums.length;int cnt = 0;for (int i = 0; i < N; i++) {for (int j = i + 1; j < N; j++) {for (int k = j + 1; k < N; k++) {if (nums[i] + nums[j] + nums[k] == 0) {cnt++;}}}}return cnt;}
}

2. ThreeSumBinarySearch

将数组进行排序,对两个元素求和,并用二分查找方法查找是否存在该和的相反数,如果存在,就说明存在和为 0 的三元组。

应该注意的是,只有数组不含有相同元素才能使用这种解法,否则二分查找的结果会出错。

该方法可以将 ThreeSum 算法增长数量级降低为 O(N2logN)。

public class ThreeSumBinarySearch implements ThreeSum {@Overridepublic int count(int[] nums) {Arrays.sort(nums);int N = nums.length;int cnt = 0;for (int i = 0; i < N; i++) {for (int j = i + 1; j < N; j++) {int target = -nums[i] - nums[j];int index = BinarySearch.search(nums, target);// 应该注意这里的下标必须大于 j,否则会重复统计。if (index > j) {cnt++;}}}return cnt;}
}
public class BinarySearch {public static int search(int[] nums, int target) {int l = 0, h = nums.length - 1;while (l <= h) {int m = l + (h - l) / 2;if (target == nums[m]) {return m;} else if (target > nums[m]) {l = m + 1;} else {h = m - 1;}}return -1;}
}

3. ThreeSumTwoPointer

更有效的方法是先将数组排序,然后使用双指针进行查找,时间复杂度为 O(N2)。

同样不适用与数组存在重复元素的情况。

public class ThreeSumTwoPointer implements ThreeSum {@Overridepublic int count(int[] nums) {int N = nums.length;int cnt = 0;Arrays.sort(nums);for (int i = 0; i < N - 2; i++) {int l = i + 1, h = N - 1, target = -nums[i];while (l < h) {int sum = nums[l] + nums[h];if (sum == target) {cnt++;l++;h--;} else if (sum < target) {l++;} else {h--;}}}return cnt;}
}

倍率实验

如果 T(N) ~ aNblogN,那么 T(2N)/T(N) ~ 2b

例如对于暴力的 ThreeSum 算法,近似时间为 ~N3/6。进行如下实验:多次运行该算法,每次取的 N 值为前一次的两倍,统计每次执行的时间,并统计本次运行时间与前一次运行时间的比值,得到如下结果:

NTime(ms)Ratio
50048/
10003206.7
20005551.7
400041057.4
8000335758.2
160002689098.0

可以看到,T(2N)/T(N) ~ 23,因此可以确定 T(N) ~ aN3logN。

public class RatioTest {public static void main(String[] args) {int N = 500;int loopTimes = 7;double preTime = -1;while (loopTimes-- > 0) {int[] nums = new int[N];StopWatch.start();ThreeSum threeSum = new ThreeSumSlow();int cnt = threeSum.count(nums);System.out.println(cnt);double elapsedTime = StopWatch.elapsedTime();double ratio = preTime == -1 ? 0 : elapsedTime / preTime;System.out.println(N + "  " + elapsedTime + "  " + ratio);preTime = elapsedTime;N *= 2;}}
}
public class StopWatch {private static long start;public static void start() {start = System.currentTimeMillis();}public static double elapsedTime() {long now = System.currentTimeMillis();return (now - start) / 1000.0;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/444035.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

俄罗斯方块(C++)

#include<iostream> #include<stdlib.h> #include<windows.h> #include<time.h> #include<conio.h> using namespace std;#define A1 0//A代表长条型&#xff0c;B为方块&#xff0c;C为L型&#xff0c;D为闪电型&#xff08;实在无法描述那个形…

leetcode445. 两数相加 II

给你两个 非空 链表来代表两个非负整数。数字最高位位于链表开始位置。它们的每个节点只存储一位数字。将这两数相加会返回一个新的链表。 你可以假设除了数字 0 之外&#xff0c;这两个数字都不会以零开头。 进阶&#xff1a; 如果输入链表不能修改该如何处理&#xff1f;换…

Java中如何实现每天定时对数据库的操作

现在有一个很棘手的问题&#xff1a;客户要赠加一个功能&#xff0c;就是每天晚上11点要统计一下数据&#xff0c;并存到一个文件中&#xff0c;我试着用线程&#xff0c;但是总达不到理想的效果。请给点思路&#xff0c;多谢了。 我们的开发环境是tomcat和servlet&#xff0c;…

leetcode面试题 02.08. 环路检测

给定一个有环链表&#xff0c;实现一个算法返回环路的开头节点。 有环链表的定义&#xff1a;在链表中某个节点的next元素指向在它前面出现过的节点&#xff0c;则表明该链表存在环路。 示例 1&#xff1a; 输入&#xff1a;head [3,2,0,-4], pos 1 输出&#xff1a;tail co…

leetcode485. 最大连续1的个数 *py:“又是一行就解决了,没意思”

给定一个二进制数组&#xff0c; 计算其中最大连续1的个数。 示例 1: 输入: [1,1,0,1,1,1] 输出: 3 解释: 开头的两位和最后的三位都是连续1&#xff0c;所以最大连续1的个数是 3. 注意&#xff1a; 输入的数组只包含 0 和1。 输入数组的长度是正整数&#xff0c;且不超过 1…

leetcode645. 错误的集合

645. 错误的集合 难度简单98 集合 S 包含从1到 n 的整数。不幸的是&#xff0c;因为数据错误&#xff0c;导致集合里面某一个元素复制了成了集合里面的另外一个元素的值&#xff0c;导致集合丢失了一个整数并且有一个元素重复。 给定一个数组 nums 代表了集合 S 发生错误后的…

一篇文章揭穿创业公司的套路

初衷 每个初入社会的求职者&#xff0c;都曾经有过找工作被坑的经历。总结了以下潜台词&#xff0c;如果你能全部GET到&#xff0c;那么恭喜你&#xff0c;已被老板们拉入黑名单。 核心 「工资4k - 8k」——那工资就是4k 「工资上不封顶」——就是说说而已&#xff0c;没人会给…

java获取IP地址:

public class IPDemo {public static void main(String [] args) throws UnknownHostException{//获取本地主机InetAddress localHost InetAddress.getLocalHost();System.out.println(localHost);//Lenovo-sxg/192.168.1.106//获取本地主机的IP地址String ipaddress localHo…

《三天给你聊清楚redis》第1天先唠唠redis是个啥(18629字)

后端需要知道的关于redis的事&#xff0c;基本都在这里了。 此文后续会改为粉丝可见&#xff0c;所以喜欢的请提前关注。 你的点赞和评论是我创作的最大动力&#xff0c;谢谢。 1、入门 Redis是一款基于键值对的NoSQL数据库&#xff0c;它的值支持多种数据结构&#xff1a;…

获取IP地址:

public class IPDemo {public static void main(String [] args) throws UnknownHostException{//获取本地主机InetAddress localHost InetAddress.getLocalHost();System.out.println(localHost);//Lenovo-sxg/192.168.1.106//获取本地主机的IP地址String ipaddress localHo…

使用github+jsdelivr作为视频床

感谢JefferyIF大佬提供的神奇方法。 1. 配置FFmpeg 注&#xff1a;IOS因为不支持HLS&#xff0c;所以对IOS上无法正常播放视频&#xff0c;其他端都可以正常播放。 因为脚本要使用到FFmeg对源视频文件切分成m3u8格式&#xff0c;所以在使用脚本之前&#xff0c;请配置好 FFm…

使用PicGo+github+jsdelivr作为图床

1.什么是图床&#xff1f; 所谓图床工具&#xff0c;就是自动把本地图片转换成链接的一款工具&#xff0c;网络上有很多图床工具&#xff0c;就目前使用种类而言&#xff0c;PicGo 算得上一款比较优秀的图床工具。它是一款用 Electron-vue 开发的软件&#xff0c;可以支持微博…

万字干货:教新手从0到1搭建完整的增长数据体系

在实际的业务中&#xff0c;大多数人可能只会遇到以下一种或几种常见的场景&#xff0c;并且对于各个细分场景&#xff0c;所需要解决的问题和关注重点都是不一样的。 场景一&#xff1a;你刚加入一个成熟产品的用户增长部门&#xff0c;会发现业务当前有非常全面详实的用户和业…

《三天给你聊清楚redis》第2天看看redis怎么被搞出来的(22036字)

后端需要知道的关于redis的事&#xff0c;基本都在这里了。 此文后续会改为粉丝可见&#xff0c;所以喜欢的请提前关注。 你的点赞和评论是我创作的最大动力&#xff0c;谢谢。 3、单机实现 3.1、数据库概述 redis服务器将所有数据库都保存在redis/redisServer中&#xff…

JAVA中对象的序列化的作用?

1、序列化是干什么的&#xff1f; 简单说就是为了保存在内存中的各种对象的状态&#xff0c;并且可以把保存的对象状态再读出来。虽然你可以用你自己的各种各样的方法来保存Object States&#xff0c;但是Java给你提供一种应该比你自己好的保存对象状态的机制,那就是序列化。 2…

leetcode559. N叉树的最大深度

给定一个 N 叉树&#xff0c;找到其最大深度。 最大深度是指从根节点到最远叶子节点的最长路径上的节点总数。 例如&#xff0c;给定一个 3叉树 : 我们应返回其最大深度&#xff0c;3。 说明: 树的深度不会超过 1000。 树的节点总不会超过 5000。 思路见代码 /* // De…

leetcode1491. 工资平均值 这也叫题?也太简单了吧

给你一个整数数组 salary &#xff0c;数组里每个数都是 唯一 的&#xff0c;其中 salary[i] 是第 i 个员工的工资。 请你返回去掉最低工资和最高工资以后&#xff0c;剩下员工工资的平均值。 示例 1&#xff1a; 输入&#xff1a;salary [4000,3000,1000,2000] 输出&#x…

leetcode1207. 独一无二的出现次数

给你一个整数数组 arr&#xff0c;请你帮忙统计数组中每个数的出现次数。 如果每个数的出现次数都是独一无二的&#xff0c;就返回 true&#xff1b;否则返回 false。 示例 1&#xff1a; 输入&#xff1a;arr [1,2,2,1,1,3] 输出&#xff1a;true 解释&#xff1a;在该数组…

leetcode1085. 最小元素各数位之和 py不止是字符串的神!

给你一个正整数的数组 A。 然后计算 S&#xff0c;使其等于数组 A 当中最小的那个元素各个数位上数字之和。 最后&#xff0c;假如 S 所得计算结果是 奇数 的请你返回 0&#xff0c;否则请返回 1。 示例 1: 输入&#xff1a;[34,23,1,24,75,33,54,8] 输出&#xff1a;0 解释…

leetcode709. 转换成小写字母 最秀狼人做法(比狠多一点)

实现函数 ToLowerCase()&#xff0c;该函数接收一个字符串参数 str&#xff0c;并将该字符串中的大写字母转换成小写字母&#xff0c;之后返回新的字符串。 示例 1&#xff1a; 输入: "Hello" 输出: "hello" 示例 2&#xff1a; 输入: "here"…