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

2020-03-07  本文已影响0人  nowherespyfly

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;
    }
};
上一篇下一篇

猜你喜欢

热点阅读