lintcode 1-4

1 .A + B 问题

给出两个整数a和b, 求他们的和, 但不能使用 + 等数学运算符。

加减法在底层是使用二进制来进行运算的。
加法,

位运算

  1. 按位求反~
    运来为1,变为0,0变为1;~1010=0101
  2. 与运算&
    两个对应位置都为1,则为1,否则为0。1010 & 1011=·1010
  3. 或运算|
    两个对应位置都为0,则为0,否则为1。1010 | 1011 = 1011
  4. 异或运算^
    两个对应位置只有一个为1,则为1,否则为0。1010 ^ 1011 = 0001

加法分为两部分,某一位相加,和产生进位。
不考虑进位的情况下,二进制:1+1=00+1=1。这符合异或运算。
然后再加上这一位产生的进位。如果两个对应为上都为1,那么应该产生进位1 & 1 = 11 & 0 = 0,这符合与运算。
如果当前应该进位,那么下一位应该加1,也就是本位进位,下一位加1,所以应该使用移位运算,进行左移一位1&1<<1

所以加法是:异或运算进行相加,而与运算进行进位。

a + b = a^b + (a&b)<<1

题目要求不能使用加法,所以,是个递归a+b = add(a^b,(a&b)<<1)
当,其中任意一个为0时(通常是不产生进位,第二个参数为0),那么结束,第一个参数也就是最终的值。

代码

int add(a,b)
{
  if(a == 0)
  {
    return b;//通常因为起始输出的a就是0
  }
  if(b == 0)
  {
    return a;//通常因为,在递归中不产生进位。
  }
  return add(a^b,(a&b)<<1);
}

拓展

如果是减法那?
相减使用的是还是异或运算,借位使用的仍然是与运算。但是,1-00-1,并不一定是否要借位啊。
减去一个数等于加上这个数的相反数。而按位求反,正式求一个数的相反数(不考虑进位)。
但是这又涉及到第一位的问题。
暂时无解。

2. 阶乘结果尾部的0

设计一个算法,计算出n阶乘中尾部零的个数

先计算出阶乘结果,然后再不断的%10,记录次数,当不是0的时候返回。
这种肯定不行。

数学的方法:

这些相乘的数中,含有多少个因数5,就有多少个0,因为偶数比因数5多的多。比如25,含2个。125含3个。

b=n/10结果为,有多少个5的倍数:1,5,、、25、、、100、
b/5结果为,5的倍数中,多出的因数5:25,125
在继续,然后把数值相加。

    long long trailingZeros(long long n) {    
        long long count{0};
        long long  tmp=n;
         while(tmp != 0)
         {             
             tmp=tmp/5;
             count+=tmp;
         }
         return count;
    }

3. 统计数字

计算数字k在0到n中的出现的次数,k可能是0~9的一个值

貌似目前只能挨个比较。

int digitCounts(int k, int n) {
        int count = 0;  
        for(int i = 0;i<=n;i++)  
        {  
            int number = i;  
            while(number/10)  
            {  
                if(number % 10 == k)  
                {  
                    count++;  
                }  
                number = number/10;  
            }  
            if(number == k)  
                count++;  
        }  
        return count;  
    }  

目前知道的只有这种。
还有一种是按照位来判断的。
看原理应该是除了第一位以外的其他都相同。
按位分析

4. 丑数 II

设计一个算法,找出只含素因子2,3,5 的第 n 大的数。
符合条件的数如:1, 2, 3, 4, 5, 6, 8, 9, 10, 12...

这个丑数也就是,因数是如上几个的都是丑数。

按照惯例,

  1. 从1到n,暴力的判断是否是丑数,记录是第几个,然后返回。这种显然不行。
    明显不行啊。应该是大于O(3n)??中间空的多了可能更大。
    好像也可以。
  2. 从1开始构造整个丑数队列。

构造这个丑数数列:

  1. 暴力的构造
    就是从头到尾挨个构造
  2. 将构造好的丑数存起来,然后从中取最小的。

如上的丑数都有
2^a * 3^b * 5^c的形式。
所以,我们每次讲讲一个数,乘2,3,5分别存在三个容器中。再从中取最小的插入到构造的丑数队列。

第一次我们将,1*2,1*3,1*5放入q2,q3,q5中,然后再从中取最小的,放入构造好的容器中。
取出的元素假设是m,那么将m*2,m*3,m*5,放入到容器中。

这里存在一个问题:就是会重复
如果m从q2中取出,那么分别乘放入三个容器中
如果从q3中取出,那么只相乘放入q3,q5中即可。
如果从q5中取出,那么只相乘放入q5
原因在于:
如果我们从q5中取出元素y他应该是由5x得来的。也就是此刻最小的元素是5x,那么之前,我们一定遇到过2x,也就是一定添加过2x*5,此时如果在,添加y*2 = 5x*2,便会重复。

#include <iostream>
#include <deque>
#include <algorithm>
using namespace std;
int64_t uglyNum(int64_t n)
{
    deque<int64_t> two;
    deque<int64_t> three;
    deque<int64_t> five;
    two.push_back(1);
    
    int64_t i = 0;
    int64_t tmp;
    INT
    int64_t twoMin;
    int64_t threeMin;
    int64_t fiveMin;
    
    int64_t ret;
    while(i != n)
    {
        i++;
        twoMin = two.empty()?INT64_T:two.front();
        threeMin = three.empty()?INT64_T:three.front();
        fiveMin = five.empty()?INT64_T:five.front();
        tmp =  min(min(twoMin,threeMin),fiveMin);
        
        if(tmp==twoMin)
        {
            ret=twoMin;
            two.push_back(2*ret);
            three.push_back(3*ret);
            five.push_back(5*ret);
            two.pop_front();
            continue;
        }
        if(tmp == threeMin)
        {
            ret=three.front();
            three.push_back(3*ret);            
            five.push_back(5*ret);
            three.pop_front();            
            continue;
        }
        if(tmp == fiveMin)
        {
            ret=five.front();
            five.push_back(5*ret);
            five.pop_front();
            continue;            
        }
    }
    return ret;   
}
int64_t main()
{
    int64_t i;
    while(cin>>i)
    {
        cout<<"ugly :"<<uglyNum(i)<<endl;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容