C++版 - 剑指offer面试题38:数字在已排序数组中出现的次数

数字在已排序数组中出现的次数


参与人数:2597    时间限制:1秒   空间限制:32768K
本题知识点:  数组


题目描述

统计一个数字在已排序数组中出现的次数。

样例输入:
2 3 3 3 3 4 51
3

6,5,3,3,1,0
3

样例输出:
4
2


分析:
      数字在排序数组中出现的次数,首先想到的方法应该是用hash表,计算出数组中所有数据出现的次数,然后直接查找,时间复杂度O(n),空间复杂度O(n)。但这种方法未能利用该数组是已排序的特点,所以如果输入是已排好序的题目,要及时联想到二分查找。


具体步骤:先用二分法找到某个目标值k出现的位置,然后统计前面一半中k出现的次数sum1,后面一半中k出现的次数sum2,最后sum=sum1+1+sum2。二分查找时间复杂度是O(logn)。


AC代码:

#include<cstdio>
#include<vector>
using namespace std;
class Solution {
public:
    int GetNumberOfK(vector<int> data, int k) {
        int idx = biSearch(data,0, (int)data.size(), k);   // 某个k的位置为idx 
        if(idx == -1) return 0;  // 未找到 
        int sum = 1; // 如果能找到1个k,sum初始化为1 
        for(int j = idx - 1; j >= 0; j--){  // 统计之前的那个k所在位置idx的前面k出现的次数 
            if(data[j] == k) sum++;
            else break;
        }
        for(int j = idx + 1; j < (int)data.size(); j++){  // 统计之前的那个k所在位置idx的后面k出现的次数
            if(data[j] == k) sum++;
            else break;
        }
        return sum;
    }     
    int biSearch(vector<int> &data, int begin, int end, int k){
        if(begin >= end) return -1;
        int mid = (begin + end)/2;
        if(data[mid] == k) return mid;  // 如果在区间mid处找到,提前返回;否则递归地在前一半去找,否则在后一半去找,总能找到 
        int pos1, pos2 = -1;
        pos1 = biSearch(data,begin,mid,k); 
        if(pos1 != -1) return pos1;
        pos2 = biSearch(data,mid + 1, end,k); 
        if(pos2 != -1) return pos2;              
        return -1;
    }
};
// 以下为测试 
int main()
{
	Solution sol;
	vector<int> vec1={1,3,4,5};
	vector<int> vec2={6,5,3,3,1,0};	
	int res1 = sol.GetNumberOfK(vec1, 4);
	int res2 = sol.GetNumberOfK(vec2, 3);	
	printf("%d\n", res1);	
	printf("%d\n", res2);		
	return 0;
}




全部评论

相关推荐

bg:弱211本&nbsp;26届日常实习大二下的时候开始找了第一段实习,找的比较随意,随便找了个小厂🐶了三个月(4月到6月),最后以回学校准备期末考告终,开发流程不是特别规范,但是感觉那段时间是自己成长最快的阶段。7月份报名了移动的线上实习,做的是效益评估智慧管理平台,技术难点主要是业务实现,没什么并发难度,所以就抱着边写代码边啃算法和小林的计划,自己在图书馆学习了。此外,自己还有一段开源经历,研究的是JDK协程在completeableFuture下的自旋优化。自八月底,开始海投简历,项目用马哥的牛券和短链接。百度快手bilibili(简历挂)腾讯(二面挂)小红书(二面挂,现在又被一个鸡架平台捞起来了,看看有没有机会)美团(没人回,后面发现官网挺多,Boss上面岗位不多,手痒痒但还是之后再说吧)字节(一面挂了一次,又被捞,第二次二面挂,暂时不投了,心累)京东(简历挂,师姐推不过去啊)momenta(一面过了,二面结束后聊了聊感觉用cpp和py用的比较多,主动终止了)货拉拉(流程很快,一面二面就隔了一天,很快出offer,打算接)虎牙(等待一面ing,如果能拿到offer,打算看看业务怎么样,考虑)电信亿迅(oc,拒绝)360(oc,base地北京,太远了,已拒)用图科技(oc,已拒)光魔科技(oc,已拒)用友网络(oc,已拒)爱奇艺(一面后没有消息了)集度汽车(oc,已拒)开元维度(oc,已拒)moka(oc,已拒)元行科技(oc,已拒)汉朔科技(oc,已拒)。。。还有好多好多,就不一一列举了全都是在Boss上面投的,发现上面岗位不多,感觉后面还是去官网看看吧。权衡了别的offer,想到大三上学校可能还会有一些事(虽然打算全把课翘了),还有不想谈异地恋,最后还是选择base地离广州近一点的深圳。货拉拉的业务也比较感兴趣,已经约好27号入职,干到11月,满打满算简历上写3个月,沉淀一下这段实习再找下一段大厂日常实习了。个人感受:感觉现在八股问的不是特别多,很多喜欢对着实习经历和项目来问,好几场都是全程几乎没八股,都是实习、项目和场景题。或者是从简历的实习经历开始发散问一些八股,可能死磕八股现在不太行了。或者感觉,也不需要硬记八股,多看一些专栏,把那些JUC,JVM的知识点在不同的地方多见几次,可能就真的能熟悉,面试的时候按着自己的想法说出来了吧。力扣自己一直也在刷,来来回回刷了300来题,现在不少小厂都开始问算法了,难顶,看来还是继续刷。大家也一起好好加油!!!
点赞 评论 收藏
分享
点赞 收藏 评论
分享
牛客网
牛客企业服务