文章目录
- 造海船
- 描述
- 输入描述
- 输出描述
- 样例输入 1
- 样例输出 1
- 提示
- 题解
- 寻找第一个1
- 题目描述
- 输入描述
- 输出描述
- 测试用例
- 题解
- 查找数字是否出现
- 描述
- 输入描述
- 输出描述
- 样例输入 1
- 样例输出 1
- 题解
- 字典找数
- 描述
- 输入描述
- 输出描述
- 样例输入 1
- 样例输出 1
- 题解
- 寻找第一个偶数
- 题目描述
- 输入描述
- 输出描述
- 题解
造海船
描述
明朝郑和下西洋,需要建造庞大的海船,需要足够的木料,因为那时候没有钢铁制造的船,现在有 n 根原木,现在想把这些木头切割成 k 段长度均为 l 的小段木头(木头有可能有剩余),用来制造船的部件。
当然,工匠希望得到的小段木头越长越好,这样可以让船更大一些不浪费木料,请求出 l 的最大值。
原木的长度都是正整数,我们要求切割得到的小段木头的长度也是正整数。
例如有两根原木长度分别为 11 和 21,要求切割成等长的 6 段,很明显能切割出来的小段木头长度最长为 5。
现在希望你能用现代科技可以帮助他们计算出来。
输入描述
第一行是两个正整数 n,k,分别表示原木的数量,需要得到的小段的数量。
接下来 n 行,每行一个正整数 Li,表示一根原木的长度。
输出描述
仅一行,即 l 的最大值。
如果连 1cm 长的小段都切不出来,输出 0。
样例输入 1
3 7
232
124
456
样例输出 1
114
提示
数据规模与约定
对于 100% 的数据,有 1≤n≤105,1≤k≤108,1≤Li≤10^8(i∈[1,n])。
题解
#include<iostream>
using namespace std;
#define LL long long
int n,k,l,r,mid,a[100860];
LL sum=0;bool yvn (int x){int ans=0;for(int i=0;i<n;i++){ans+=a[i]/x;}return ans>=k;
}
int main(){cin>>n>>k;for(int i=0;i<n;i++) cin>>a[i],sum+=a[i];if(sum<k) cout<<"0"<<endl;else if (sum==k) cout<<"1"<<endl;else {l=0;r=sum/k;while(l<r){mid=(l+r+1)>>1;if(yvn(mid)) l=mid;else r=mid-1;}cout<<l<<endl;}return 0;
}
寻找第一个1
题目描述
给定一个01序列,序列从左至右一开始是若干个0,接下来直到末尾是若干个1。求第一个1的下标;如果序列中没有1,那么输出-1。
注:使用二分法实现。
输入描述
输出描述
输出第一个1的下标;如果没有1,那么输出-1。
测试用例
样例1
输入
5
0 0 0 1 1
输出
3
解释
从左到右第一个1的下标是3
样例2
输入
5
0 0 0 0 0
输出
-1
解释
序列中没有1,因此输出-1
题解
只需要在二分模板的基础上稍加修改,当找到1的时候不让他结束,继续往左找,因为要找第一个1,不确定左边还有没有1,这里还需要定义一个标志位flag,如果找到1那么变为true,最后while循环结束只要看一下是不是true,不是的话说明没找到,输出-1就ok了。
#include<bits/stdc++.h>
using namespace std;
bool flag = false;
const int N = 1e6;
int A[N];int BinarySearch(int *a,int len){int low = 0,high = len-1,mid;while(low<=high){mid = (low+high)/2;if(a[mid] == 1){high = mid-1;flag = true;}else{low = mid+1;}}return flag == true?high+1:-1;
}int main(){int n;cin >> n;for(int i=0;i<n;i++){cin >> A[i];}cout << BinarySearch(A,n);
}
查找数字是否出现
描述
在一个升序序列中,查找给定值是否存在,如果存在输出YES,否则输出NO。
输入描述
第一行一个整数n,表示升序序列的长度(1<n≤107) 。
第二行包含n个整数,升序序列各元素。所有元素取值区间(0~1010)。
第三行一个整数m,表示要查询数值的个数(1<m<105)。
接下来m行,每行一个整数,表示要查找数字。查找数字取值( 0~1010 )。
输出描述
m行,每行YES或NO,如果存在输出YES,否则输出NO。保持输入顺序。
样例输入 1
5
1 1 3 6 1024
2
1000
6
样例输出 1
NO
YES
题解
#include <bits/stdc++.h>
using namespace std;
const int N = 1e8;
int A[N];int main() {int key,n,m,mid;cin >> n;for(int i = 0;i < n;i ++){cin >> A[i];}cin >> m;while(m--){cin >> key;int low = 0,high = n-1,flag;while(low <= high){mid = (low+high)/2,flag = 0;if(A[mid] == key) {cout << "YES" << endl;flag = 1;break;}else if(A[mid] > key) high =mid - 1;else low = mid+1;}if(!flag) cout << "NO" << endl;}return 0;
}
字典找数
描述
现有一字典,查找字典15到45页中,某一页的页码,用二分法快速翻至所需要页码,需要翻多少次?(中间值 mid = (最大值+最小值)/2)
输入描述
字典中某一页的页码n。(15≤n≤45)
输出描述
一个整数,表示查找的次数。
样例输入 1
18
样例输出 1
3
题解
#include<bits/stdc++.h>
using namespace std;
int main() {int low = 15,high = 45,mid,n,count = 0;cin >> n;while(low<=high){mid = (low+high)/2;if(mid == n) {count++;break;}else if(mid < n){low = mid + 1;count++;}else{high = mid - 1;count++;}}cout << count;return 0;
}
寻找第一个偶数
题目描述
给定一个正整数序列,序列从从只有一开始是若干个奇数,接下来直到末尾是若干个偶数。求第一个偶数的下标;如果序列中没有偶数,那么输出-1。
注:使用二分法实现。
输入描述
输出描述
输出第一个偶数的下标;如果没有偶数,那么输出-1。
样例1
输入
5
3 5 1 8 6
输出
3
解释
从左到右第一个偶数是8,下标是3
样例2
输入
5
3 5 1 7 3
输出
-1
解释
序列中没有偶数,因此输出-1
题解
#include<bits/stdc++.h>
using namespace std;
bool flag = false;
const int N = 1e6;
int A[N];int BinarySearch(int *a,int len){int low = 0,high = len-1,mid;while(low<=high){mid = (low+high)/2;if(a[mid] % 2 == 0){high = mid-1;flag = true;}else{low = mid+1;}}return flag == true?high+1:-1;
}int main(){int n;cin >> n;for(int i=0;i<n;i++){cin >> A[i];}cout << BinarySearch(A,n);
}