• 时间复杂度课后习题


    目录

    选择题

    编程题

    1.消失的数字

    方法1:辅助数组法

     方法2:位运算法

    方法3:求和相减法

    2.轮转数组

    方法1:额外数组法 

    方法2:数组翻转法


    选择题

    1.给定一个整数sum,从有N个有序元素的数组中寻找元素a,b,使得a+b的结果最接近sum,最快的平均时间复杂度是(   )

    A.O(n)

    B.O(n^2)

    C.O(nlog2(n))

    D.O(log2(n))

     解析:

      此题目中,数组元素有序,所以a,b两个数可以分别从开始和结尾处开始搜,根据首尾元素的和是否大于sum,决定搜索的移动,整个数组被搜索一遍,就可以得到结果,所以最好时间复杂度为n


    2.如果一个函数的内部中只定义了一个二维数组a[3][6],请问这个函数的空间复杂度为(   )

    A.O(n)

    B.O(n^2)

    C.O( 1 )

    D.O(m*n)

    答案:C

      解析:函数内部数组的大小是固定的,不论函数运行多少次,所需空间都是固定大小的,因此空间复杂度为O(1)


    3.分析以下函数的空间复杂度

    1. public static int[][] get2Array(int n){
    2. int[][] array = new int[n][];
    3. for(int i = 0; i < n; i++) {
    4. array[i] = new int[n-i];
    5. n--;
    6. }
    7. return array;
    8. }

    解析:

    第0行n个int的空间

    第1行n-1个int的空间

    第2行n-2个int的空间

    ...

    第n-1行1个元素的空间

    空间总的个数为:1+2+3+...+N-1 + N + N = (1+N)*N/2 + N = N^2/2 + 3N/2

    采用大O渐进发表示就是:O(N^2)

    故选择C


    编程题(小练一手)

    1.消失的数字

    面试题 17.04. 消失的数字icon-default.png?t=N7T8https://leetcode.cn/problems/missing-number-lcci/

    方法1:辅助数组法
    1. public int missingNumber(int[] nums) {
    2. int[] arr = new int[nums.length + 1];
    3. for (int i = 0; i < nums.length; i++) {
    4. arr[nums[i]] = 1;
    5. }
    6. int i = 0;
    7. for (; i < arr.length; i++) {
    8. if (arr[i] == 0) {
    9. break;
    10. }
    11. }
    12. return i;
    13. }

    解析:

    (1)首先遍历数组 nums,将数组中的每个元素作为辅助数组arr的下标,存放整数1。

    (2)然后依次检查从 0到 n 的每个整数是否在辅助数组中,不在辅助数组中的数字即为消失的数字。

     方法2:位运算法
    1. public int missingNumber(int[] nums) {
    2. int flag = 0;
    3. for (int i = 0; i < nums.length; i++) {
    4. flag^=nums[i];
    5. }
    6. for (int i = 0; i <=nums.length; i++) {
    7. flag^=i;
    8. }
    9. return flag;
    10. }

    解析:

    利用异或运算两次遍历可以算出消失的数字

    方法3:求和相减法
    1. public int missingNumber(int[] nums) {
    2. int sum1=0,sum2=0;
    3. for (int i = 0; i < nums.length; i++) {
    4. sum1+=nums[i];
    5. }
    6. for (int i = 0; i <=nums.length; i++) {
    7. sum2+=i;
    8. }
    9. return sum2-sum1;
    10. }

    解析:

    将数组 nums 的元素之和记为 sum1,1到nums.length之和记为 sum2。sum1 比sum2  消失的一个数字,因此消失的数字即为sum2-sum1 之差。


    2.轮转数组

    189. 轮转数组icon-default.png?t=N7T8https://leetcode.cn/problems/rotate-array/

    方法1:额外数组法 
    1. class Solution {
    2. public void rotate(int[] nums, int k) {
    3. int sz=nums.length;
    4. int[] arr = new int[sz];
    5. for (int i = 0; i < sz; i++) {
    6. arr[(i+k)%sz]=nums[i];
    7. }
    8. System.arraycopy( arr, 0, nums, 0, sz);
    9. }
    10. }

    解析: 

      我们可以使用额外的数组来将每个元素放至正确的位置。用 size 表示数组的长度,我们遍历原数组,将原数组下标为 i 的元素放至新数组下标为 (i+k)%size 的位置,最后将新数组拷贝至原数组即可。

    方法2:数组翻转法

    思路如下:

    1. 首先对整个数组实行翻转。
    2. 这时候,从 k 处分隔数组,左右两数组,各自进行翻转即可。
    1. class Solution {
    2. public void rotate(int[] nums, int k) {
    3. int sz=nums.length;
    4. k%=sz;
    5. reverse(nums,0,sz-1);
    6. reverse(nums,0,k-1);
    7. reverse(nums,k,sz-1);
    8. }
    9. public void reverse(int[] nums, int start, int end) {
    10. while (start
    11. int tmp=nums[start];
    12. nums[start]=nums[end];
    13. nums[end]=tmp;
    14. start++;
    15. end--;
    16. }
    17. }
    18. }

    以上为我个人的小分享,如有问题,欢迎讨论!!! 

    都看到这了,不如关注一下,给个免费的赞 

     

  • 相关阅读:
    抽取泛微和建云的销售合同定时任务(要求记录翻译不成功的字段)
    【javascript】Array 常用方法
    Spring中有哪几种方法获取HttpSession对象
    Spring之Gateway网关
    wget 下载盯盘文件
    《复盘网飞》整理
    大数据与人工智能的未来已来
    SDP最佳实践丨为汽车品牌 L 铸造「数字化营销+管控」
    [面试]我们常说的负载均衡是什么东西?
    u-boot中的&&
  • 原文地址:https://blog.csdn.net/WHabc2002/article/details/132890349