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

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;
    }
};
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容

  • 在C语言中,五种基本数据类型存储空间长度的排列顺序是: A)char B)char=int<=float C)ch...
    夏天再来阅读 3,339评论 0 2
  • 计算机二级C语言上机题库(南开版) 1.m个人的成绩存放在score数组中,请编写函数fun,它的功能是:将低于平...
    MrSunbeam阅读 6,336评论 1 42
  • 说明:该系列博客整理自《算法导论(原书第二版)》,但更偏重于实用,所以晦涩偏理论的内容未整理,请见谅。另外本人能力...
    黑夜0411阅读 1,467评论 0 2
  • 1、用C语言实现一个revert函数,它的功能是将输入的字符串在原串上倒序后返回。 2、用C语言实现函数void ...
    希崽家的小哲阅读 6,255评论 0 12
  • 1. 找出数组中重复的数字 题目:在一个长度为n的数组里的所有数字都在0到n-1的范围内。数组中某些数字是重复的,...
    BookThief阅读 1,748评论 0 2