美文网首页
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 设计,数学

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