day21 algo_116、adv_105|、108

任务量: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
(1
2+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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值