任务量:20
ALGO-116. 最⼤大的算式!!
问题描述
题⽬目很简单,给出N个数字,不不改变它们的相对位置,在中间加⼊入K个乘号和N-K-1个加号,(括号随
便便加)使最终结果尽量量⼤大。因为乘号和加号⼀一共就是N-1个了了,所以恰好每两个相邻数字之间都有⼀一
个符号。例例如:
N=5,K=2,5个数字分别为1、2、3、4、5,可以加成:
12(3+4+5)=24
1*(2+3)(4+5)=45
(12+3)(4+5)=45
……
输⼊入格式
输⼊入⽂文件共有⼆二⾏行行,第⼀一⾏行行为两个有空格隔开的整数,表示N和K,其中(2<=N<=15, 0<=K<=N-
1)。第⼆二⾏行行为 N个⽤用空格隔开的数字(每个数字在0到9之间)。
输出格式
输出⽂文件仅⼀一⾏行行包含⼀一个整数,表示要求的最⼤大的结果
样例例输⼊入
5 2
1 2 3 4 5
样例例输出
120
样例例说明
(1+2+3)45=120
思考:
最大的结果-看出他是通过动态规划进行
dp[i][j]表示i是前位数,j是有多少个号,s[i是总和]
则要想知道dp[i][j],可以尝试从第⼆二个数的前⾯面⼀一直到最后⼀一
个数的前⾯面依次添加乘号,将最⼤大的结果保存⾄至dp[i][j]中。就可以得到状态转移⽅方程为:dp[i][j] =
max(dp[i][j], dp[l-1][j-1] * (sum[i] – sum[l-1]));l为插⼊入相乘的两个数的后⼀一个数字的坐标。
可以二次做
#include<iostream>
using namespace std;
int main(){
int n,k;
cin>>n>>k;
int s[n];
int dp[16][16];//i是前位数,j是有多少个*号
for(int i=1;i<=n;i++)
{
cin>>s[i];
if(i>=2)
s[i]+=s[i-1];
}//s里面就是总和
for(int i=1;i<=n;i++)
dp[i][0]=s[i];
for(int i=2;i<=n;i++){
for(int j=1;(j<=k)&&(j<i);j++){
for(int l=2;l<=n;l++){
dp[i][j]=max(dp[i][j],dp[l-1][j-1]*(s[i]-s[l-1]));
}
}
}
cout<<fp[n][k];
return 0;
}
ADV-105. 不不同单词个数统计
问题描述
编写⼀一个程序,输⼊入⼀一个句句⼦子,然后统计出这个句句⼦子当中不不同的单词个数。例例如:对于句句⼦子“one
little two little three little boys”,总共有5个不不同的单词:one, little, two, three, boys。
说明:(1)由于句句⼦子当中包含有空格,所以应该⽤用gets函数来输⼊入这个句句⼦子;(2)输⼊入的句句⼦子
当中只包含英⽂文字符和空格,单词之间⽤用⼀一个空格隔开;(3)不不⽤用考虑单词的⼤大⼩小写,假设输⼊入的都
是⼩小写字符;(4)句句⼦子⻓长度不不超过100个字符。
输⼊入格式:输⼊入只有⼀一⾏行行,即⼀一个英⽂文句句⼦子。
输出格式:输出只有⼀一⾏行行,是⼀一个整数,表示句句⼦子中不不同单词的个数。
输⼊入输出样例例
样例例输⼊入
one little two little three little boys
样例例输出
5
思考:由于是要找不同的字符子串,所以可以充分利用set这个容量的特性(里面的元素是唯一存在的)
#include <iostream>
#include <set>
using namespace std;
int main() {
string t;
set<string> s;
while(cin >> t) s.insert(t);
cout << s.size();
return 0;
}
ADV-108. 分数统计
问题描述
2016.4.5已更更新此题,此前的程序需要重新提交。
问题描述
给定⼀一个百分制成绩T,将其划分为如下五个等级之⼀一:
90100为A,8089为B,7079为C,6069为D,0~59为E
现在给定⼀一个⽂文件inp,⽂文件中包含若⼲干百分制成绩(成绩个数不不超过100),请你统计五个等级
段的⼈人数,并找出⼈人数最多的那个等级段,按照从⼤大到⼩小的顺序输出该段中所有⼈人成绩(保证⼈人数最
多的等级只有⼀一个)。要求输出到指定⽂文件oup中。
输⼊入格式
若⼲干0~100的正整数,⽤用空格隔开
输出格式
第⼀一⾏行行为5个正整数,分别表示A,B,C,D,E五个等级段的⼈人数
第⼆二⾏行行⼀一个正整数,表示⼈人数最多的等级段中⼈人数
接下来⼀一⾏行行若⼲干个⽤用空格隔开的正整数,表示⼈人数最多的那个等级中所有⼈人的分数,按从⼤大到⼩小
的顺序输出。
样例例输⼊入
100 80 85 77 55 61 82 90 71 60
样例例输出
2 3 2 2 1
3
85 82 80
思考:
这里可以灵活运用二维的动态数组
#include<iostream>
#include<algorithm>
#include<vector>
using namespace std;
bool cmp(int a,int b){
return a>b;
}
int main(){
vector<vector<int >>v(5);
int n;
cin>>n;
for(int i=0;i<n;i++){
int t;
cin>>t;
if(t>=90&&t<=100)
v[0].push_back(t);
else if(t>=80&&t<=89)
v[1].push_back(t);
else if(t>=70&&t<=79)
v[2].push_back(t);
else if(t>=60&&t<=69)
v[3].push_back(t);
else
v[4].push_back(t);
}
for(int i=0;i<5;i++)
cout<<v[i].size()<<" ";
int max=0,b;
for(int i=0;i<5;i++){
if(max<v[i].size()){
max=v[i].size();
b=i;
}
}
cout<<endl<<max<<endl;
sort(v[b].begin(),v[b].end(),cmp);
for(int j=0;j<v[b].size();j++)
cout<<v[b][j]<<" ";
return 0;
}
ADV-111. Quadratic Equation
问题描述
求解⽅方程ax2+bx+c=0的根。要求a, b, c由⽤用户输⼊入,并且可以为任意实数。
输⼊入格式:输⼊入只有⼀一⾏行行,包括三个系数,之间⽤用空格格开。
输出格式:输出只有⼀一⾏行行,包括两个根,⼤大根在前,⼩小根在后,⽆无需考虑特殊情况,保留留⼩小数点后两
位。
输⼊入输出样例例
样例例输⼊入
2.5 7.5 1.0
样例例输出
-0.14 -2.86
分析:1.若 x1<x2,且f(x1)*f(x2)<0,则在(x1,x2)之间⼀一定有⼀一个根
二分法逼近解,不过这里分了大根小根,所以用-b/2a作为中间值
#include<iostream>
#include<cmath>
using namespace std;
double a,b,c;
double f(double x){
return a*pow(x,2)+b*x+c;
}
double find(double x,double y){
if(abs(x-y)<0.000001) return x;
double mid=(x+y)/2;
if(f(mid)==0) return mid;
if(f(mid)*f(x)>0) return find(mid,y);
else return find(x,mid);
}
int main(){
cin>>a>>b>>c;
double mid=-b/(2*a);
double ans1=find(-9999999,mid);
double ans2=find(mid,9999999);
if (abs(ans1 - 0) < 0.0001) ans1 = 0;
if (abs(ans2 - 0) < 0.0001) ans2 = 0;
printf("%.2f %.2f",ans2,ans1);
return 0;
}