题意
题解
代码
#include
using namespace std;
int main() {
double stand[4][5]=//标准分
{1,1,0.8,0.5,0,
2,2,1.6,1.0,0,
3,3,2.4,1.5,0,
5,5,2.5,2,0};
double extra[5]={1,0.5,0.4,0.3,0};//额外分
double A=0,A0=0,B=0,B0=0,x;
for(int i=0;i<4;i++)
for(int j=0;j<5;j++) {
cin>>ghaqx;
A0+=stand[i][j]*x;//实际分
A+=stand[i][0]*x;//满分
if(i==3) {
B0+=extra[j]*x;//额外实际分
B+=extra[0]*x;//额外满分
}
}
printf("%.9lf\n",A0/A*100+B0/B);//输出百分比(除去百分号)
return 0;
}
题意
题解
代码
#include
using namespace std;
const int N=1010;
int n,p,a[N][N];
int main() {
cin>>n>>p;
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
cin>>a[i][j];
for(int i=1;i<=n;i++)
if(i>(n%p)) cout<<"0 ";
else cout<<a[i][i]<<" ";
return 0;
}
题意
题解
f2[u]=cnt2[u]*sz[u]
。当转移到点2时,即u=2时,2的子树下点与2的lca变为2,其他点与2的lca不变依然为1,所以有转移方程为f2[u]=f2[fa]-sz[u]*cnt2[fa]+sz[u]*cnt2[u]
代码
#include
#include
using namespace std;
const int N=1e5+10;
int n,q;
int h[N],e[2*N],ne[2*N],idx;//建树
int f2[N],f5[N],cnt2[N],cnt5[N],sz[N];//状态,因子数,子树大小
void add(int a,int b) {//加边
e[idx]=b,ne[idx]=h[a],h[a]=idx++;
}
int get(int x,int cot) {//计算x中cot的因子数量
int res=0;
while(x%cot==0) {
res++;
x/=cot;
}
return res;
}
void dfs(int u,int fa) {//dfs树预处理sz数组以及cnt数组
sz[u]=1;
cnt2[u]=get(u,2);
cnt5[u]=get(u,5);
for(int i=h[u];~i;i=ne[i]) {
int v=e[i];
if(v==fa) continue;
dfs(v,u);
sz[u]+=sz[v];
}
}
void dfs1(int u,int fa) {//树形dp转移
f2[u]=f2[fa]+(cnt2[u]-cnt2[fa])*sz[u];
f5[u]=f5[fa]+(cnt5[u]-cnt5[fa])*sz[u];
for(int i=h[u];~i;i=ne[i]) {
int v=e[i];
if(v==fa) continue;
dfs1(v,u);
}
}
int main() {
cin.tie(0)->sync_with_stdio(false);
cin>>n>>q;
int u,v;
memset(h,-1,sizeof h);//初始化
for(int i=1;i<n;i++) {
cin>>u>>v;
add(u,v);add(v,u);
}
dfs(1,0);
dfs1(1,0);//预处理好因子数量
while(q--) {
int x;
cin>>x;
cout<<min(f2[x],f5[x])<<'\n';//直接输出
}
return 0;
}
题意
题解
一般序列变化为邻位加法时,考虑差分
令差分序列b[i]=(a[(i+1)%3]-a[i]+3)%3
对于(2,1)这种差分,比如2 1 2,可以改变中间数字变成(0,0)
对于(1,1)这种差分,比如0 1 2,可以改变中间数字变成(2,0)
对于(0,1)这种差分,比如0 0 1,可以改变中间数字变成(1,0)
对于(1,0)这种差分,比如0 1 1,可以改变首位数字变成(0,0)
故可以发现几个性质:
(2,1),(1,0)这两种差分可以直接变成(0,0)
(0,1)可以转化为(1,0),实现了1的移动
(1,1)可以转化为(2,0),实现1->2转换
目标为:把所有数字变成一样,所以差分数组应该全变为0
差分序列中的1可以有两种方式变为0,一个是直接(1,0)变为0,另一个是(1,1)->(2,0)+(1,0)->(0,1)=(2,1)->(0,0)
差分序列中的2必须由后一位1变成0,即(2,0)+(0,1)->(1,0)=(0,0)
所以得出结论:差分序列中必须要有足够的1把2消除,其余多的1没关系可以直接变0
因此,只要差分序列中的1个数不小于2的即可
代码
#include
#include
using namespace std;
int main() {
int t,n;
cin>>t;
while(t--) {
cin>>n;
vector<int> a(n);
for(int i=0;i<n;i++) cin>>a[i];
int cnt=0;//记录序列中1,2的个数
for(int i=0;i<n;i++)
if((a[(i+1)%n]-a[i]+3)%3==1) cnt++;
else if((a[(i+1)%n]-a[i]+3)%3==2) cnt--;
cout<<(cnt<0 ? "No":"Yes")<<'\n';
}
return 0;
}