美文网首页
2020-03-07 刷题4 设计,数学

2020-03-07 刷题4 设计,数学

作者: nowherespyfly | 来源:发表于2020-03-07 19:30 被阅读0次

384 打乱数组

标签:洗牌算法,数组
本题看得我一脸懵bo,后来看了评论区才知道是要用洗牌算法,也就是保证shuffle过后,每个元素出现在每个位置的概率是相同的,也就是1/n。

1)最naive的做法是,开辟一片空间来存放shuffle后的数组,每次随机从剩下未就位的元素中取出一个元素放好,然后删除刚取出的元素。这种做法的时间复杂度是O(n^2),因为数组删除元素代价是O(n),空间复杂度是O(n).
证明:每个元素出现在第k个位置的概率是
n-1/n * n-2/n-1 * ... 1/n-k+1 = 1/n,也就是前k-1轮没有取到它,第k轮取到它的概率。

2)第二种做法是,假设从末尾数k个元素已经就位,每次从前n-k个元素中随机选一个与第n-k个元素交换位置。因为原地操作,所以这种做法的空间复杂度是O(1),且由于每次交换都是常数时间,时间复杂度也是O(n)。不过这种做法需要提前知道数组的长度。
证明:
洗牌后,每个元素出现在第n-1个位置的概率是:1/n,也就是第一轮就被选中,出现在n-2的概率是:n-1/n * 1/n-1,即第一轮没选中第二轮选中,其他位置也可以类似推出。

代码:

class Solution {
public:
    Solution(vector<int>& nums) {
        Elements = nums;
    }
    
    /** Resets the array to its original configuration and return it. */
    vector<int> reset() {
        return Elements;
    }
    
    /** Returns a random shuffling of the array. */
    vector<int> shuffle() {
        vector<int> vShuffle = Elements;
        for(int i = Elements.size() - 1; i > 0; i--){
            int idx = rand() % (i + 1);
            if(idx != i)
                swap(vShuffle[i], vShuffle[idx]);
        }
        return vShuffle;;
    }
private: vector<int> Elements;
};

/**
 * Your Solution object will be instantiated and called as such:
 * Solution* obj = new Solution(nums);
 * vector<int> param_1 = obj->reset();
 * vector<int> param_2 = obj->shuffle();
 */

这里我犯了个错误,因为要从0到i中随机选一个数,所以应该是rand() % (i+1),开始写成了rand() % i,这样的话,i是无论如何也出不来的。
3)第三种方法适用于长度未知的数组,从左向右扫描数组,到第i个位置的时候,随机将其与前i个元素中的一个交换(包括它自己)。
证明:
对于第i-1个元素,其出现在前i个位置上任意一个的概率都是1/i * i/i+1 * ...n-1/n=1/n,即后面的n-i次每次交换都没有选到它。
出现在i后位置k-1的概率是:1/k * k/k+1 * ...* n-1/n =1/n.


204 计算质数

标签:质数,数学
最朴实的办法就是依次判断2到n-1之间的每个数i是不是质数,判断的时候,依次对于每个比i小的数,看能否被i整除。
解法1:
优化1: 将用于测试整除的除数上界改为sqrt(n),判断每个数是不是质数就可以从O(n)降到O(n^1/2).
优化2: 质数一般都出现在6的倍数左右,比如5和7,比如17和19. 很好证明,6x-3, 6x-2, 6x-1, 6x, 6x+1, 6x+2, 这一轮数中,除了6x-1和6x+1,其他的肯定都不是质数。所以,可以将循环的步长设为6,每次判断6x-1和6x+1就可以。这样复杂度降低到原来的1/3.
代码:

class Solution {
public:
    bool check_prime(int n){
        for(int i = 2; i * i <= n; i++){
            if(n % i == 0) return false;
        }
        return true;
    }
    int countPrimes(int n) {
        int cnt = 0;
        for(int i = 2; i <= 4 && i < n; i++){
            if(check_prime(i)) cnt++;
        } 
        for(int i = 5; i < n; i += 6){
            if(check_prime(i)) cnt++;
            if(i+2 < n)
                if(check_prime(i+2)) cnt++;
        }
        return cnt;
    }
};

我就说这个运行时间不靠谱,同一份代码我两次跑差了二百多毫秒。

解法2
更快的做法是可以搞一个数组,每扫描到一个数时,就把它的倍数都改为false;当然这样有重复操作的嫌疑,比如扫描到2的时候就会把6置为false了,但是到3的时候,又搞了一遍。比较好的做法就是从这个数的平方开始,哇真的很机智啊,运行时间一下降了好多。这样的复杂度是O(nloglogn).

class Solution {
public:
    int countPrimes(int n) {
        int cnt = 0;
        bool *is_prime = new bool[n];
        for(int i = 0; i < n; i++) *(is_prime+i) = true;
        for(int i = 2; i < n; i++){
            if(*(is_prime+i)){
                cnt++;
                for(int j = i; j < ceil(float(n) / i); j++) *(is_prime+j*i) = false;
            }
        }
        return cnt;
    }
};

326 3的幂

这个题是真的搞笑,不过也很机智。循环或者递归是很好做,但是由于输入就是32位整数,这个范围内最大的3的幂就是3^19, 又因为3是质数,所以能整除3^19的一定是3的幂。一行搞定。。。。

class Solution {
public:
    bool isPowerOfThree(int n) {
        return (n > 0 && (1162261467 % n == 0));
    }
};

13 罗马数字转整数

这个题目比较难处理的就是,小的数字可能出现在大的之前。我开始写了很多条件判读,后来发现除了给定的六种情况,小数字是不会出现在大数字之前的,所以直接比较前后的数字就可以了。发现出现特殊情况,就减去两倍的小数字。

class Solution {
public:
    int romanToInt(string s) {
        map<char, int> m = {{'I', 1}, {'V', 5}, {'X', 10}, {'L', 50}, {'C', 100}, {'D', 500}, {'M', 1000}};
        int sum = 0;
        for(int i = 0; i < s.size(); i++){
            sum = sum + m[s[i]];
            if(i > 0){
                if(m[s[i-1]] < m[s[i]]) sum = sum - 2 * m[s[i-1]];
            }
        }
        return sum;
    }
};

相关文章

  • 2020-03-07 刷题4 设计,数学

    384 打乱数组 标签:洗牌算法,数组本题看得我一脸懵bo,后来看了评论区才知道是要用洗牌算法,也就是保证shuf...

  • PTA刷题总结-Part 3 数据结构与算法

    PTA刷题总结-Part 1 基础部分PTA刷题总结-Part 2 模拟与数学问题PTA刷题总结-Part 3 数...

  • PTA刷题总结-Part 2 模拟与数学问题

    PTA刷题总结-Part 1 基础部分PTA刷题总结-Part 2 模拟与数学问题PTA刷题总结-Part 3 数...

  • 2019-08-01

    上午 阅读水浒传五回 数学培训班上课2小时 下午 语文上课2小时 完成数学语文作业 晚上 听精英数学4个压轴题,刷...

  • 获得感比兴趣更重要

    最近看到学生疯狂的喜欢上了做数学。 私下里问缘由,因为数学可以刷题啊! 刷题?这是什么答案? 当时的我有点蒙住了,...

  • DAY4

    我给孩子的美言录: 你做数学题简直太让我Amazing,超神速啊。数学课前居然刷了4张小超市,你怎么做到的?数学课...

  • 数学┃练习还是刷题

    作为小学生家长,这几年颇有点感受,各个群都在进行五花八门的讨论,核心虽然是如何打造一个拥有素质教育综合性的孩子(快...

  • 数学刷题方法,biubiubiu~

    做数学题,从什么都想弄懂→什么都弄不懂 这是心态的问题,除非大牛,接触新的或遗忘的知识,总得熟悉一下,急不得 遇见...

  • 学习数学,到底刷不刷题?

    一,跳绳的故事 四年级男孩一分钟跳绳252个,怎么做到的?练习呗! 想突破关口就得加量不加价,跳1000个一天不能...

  • 高中数学:近三年高考真题“刷题库(含解析)”高三生人手一份,建议

    各位同学、家长大家好,今天给大家总结的是高中数学经典题的合集。 我们知道高中数学的学习是离不开刷题的,那么刷哪些题...

网友评论

      本文标题:2020-03-07 刷题4 设计,数学

      本文链接:https://www.haomeiwen.com/subject/crxhdhtx.html