A. Replacing Elements
有大于 d 的数的话,只要看有没有两个数他们的和不大于 d 就行了。
没大于 d 的数的话,那肯定就没问题了。
#include <cstdio>
#include <algorithm>
using namespace std;
int main()
{
int n,d,T;
int a[200];
scanf("%d",&T);
while (T--) {
scanf("%d%d",&n,&d);
for (int i=1; i<=n; i++) {
scanf("%d",&a[i]);
}
sort(a+1,a+n+1);
printf((a[1]+a[2]<=d || a[n]<=d) ? "YES\n" : "NO\n");
}
return 0;
}
B. String LCM
要是存在这样的串的话,说明 s 和 t 都是某个串的重复,这个数据范围,暴力枚举一下重复的串即可。
#include <cstdio>
#include <string>
#include <iostream>
using namespace std;
int check(const string &x, const string &y) { // y/x ?
int n=x.length(),m=y.length();
int i,j;
int ret=0;
for (i=-1; i<m-1; ret++) {
for (j=0; j<n; j++) {
i++;
if (i>=m || x[j]!=y[i]) {
return 0;
}
}
}
return ret;
}
int gcd(int x, int y) {return y ? gcd(y,x%y) : x;}
int main()
{
int T;
cin >>T;
string A,B,tmp;
int l;
while (T--) {
cin >>A >>B;
if (A.length()<B.length()) {
swap(A,B);
}
if (A.length()==B.length()) {
if (A==B) {
cout <<A <<endl;
}
else {
cout <<-1 <<endl;
}
}
else {
tmp="";
int l=B.length(),i,n1,n2,nn;
for (i=0; i<l; i++) {
tmp+=B[i];
n1=check(tmp,A);
n2=check(tmp,B);
if (n1 && n2) {
break;
}
}
if (i==l) {
cout <<-1 <<endl;
}
else {
nn=gcd(n1,n2);
nn=n1*n2/nn;
for (int k=1; k<=nn; k++) {
cout <<tmp;
}
cout <<endl;
}
}
}
return 0;
}
C. No More Inversions
找规律,把顺序排列的后面连续几个倒序即可。
#include <cstdio>
int main()
{
int T,n,k,m;
scanf("%d",&T);
while (T--) {
scanf("%d%d",&n,&k);
m=k+k-n;
for (int i=1; i<m; i++) {
printf("%d ",i);
}
for (int i=k; i>=m; i--) {
printf("%d ",i);
}
puts("");
}
return 0;
}
D. Program
因为只会删掉中间连续的一段,所以可以相当于从左开始连续一段加上从右开始连续一段。分别求出往左往右的最大最小值即可。
(从右边开始的那段,在处理答案的时候补上个区间差值就行)
#include <cstdio>
#define N 200005
int T,n,m,L,R,Max,Min;
char ch[N];
int x1[N],x2[N],max1[N],max2[N],min1[N],min2[N];
int max(int x,int y) {return x>y ? x : y;}
int min(int x,int y) {return x<y ? x : y;}
int main()
{
scanf("%d",&T);
while (T--) {
scanf("%d%d%s",&n,&m,(ch+1));
x1[0]=x2[n+1]=max1[0]=max2[n+1]=min1[0]=min2[n+1]=0;
for (int i=1; i<=n; i++) {
x1[i]=x1[i-1]+(ch[i]=='-' ? -1 : 1);
max1[i]=max(max1[i-1],x1[i]);
min1[i]=min(min1[i-1],x1[i]);
}
for (int i=n; i; i--) {
x2[i]=x2[i+1]+(ch[i]=='-' ? 1 : -1);
max2[i]=max(max2[i+1],x2[i]);
min2[i]=min(min2[i+1],x2[i]);
}
while (m--) {
scanf("%d%d",&L,&R);
Max=max(max1[L-1],max2[R+1]+x1[L-1]-x2[R+1]);
Min=min(min1[L-1],min2[R+1]+x1[L-1]-x2[R+1]);
printf("%d\n",Max-Min+1);
}
}
return 0;
}
E. Minimum Path
见我这篇博客。
https://blog.csdn.net/jackypigpig/article/details/112730352
#include <queue>
#include <vector>
#include <cstring>
#include <cstdio>
using namespace std;
#define ID(x,y,z) (((x)<<3)+((y)<<1)+z)
const int N=200015;
vector<pair<int,int>> graph[N];
long long d[N][2][2];
bool vis[N][2][2];
int n,m;
struct cmp {
bool operator() (pair<long long,long long> x, pair<long long,long long> y) {
return x.first > y.first;
}
};
int main()
{
scanf("%d%d",&n,&m);
for (int i=1,u,v,w; i<=m; i++) {
scanf("%d%d%d",&u,&v,&w);
u--;
v--;
graph[u].emplace_back(make_pair(v,w));
graph[v].emplace_back(make_pair(u,w));
}
for (int i=0; i<n; i++) {
for (int j=0; j<2; j++) {
for (int k=0; k<2; k++) {
d[i][j][k]=10000000000000000ll;
vis[i][j][k]=0;
}
}
}
d[0][0][0]=0;
priority_queue<pair<long long,long long>,vector<pair<long long,long long>>,cmp> q;
q.push({0,ID(0,0,0)});
while (!q.empty()) {
int u,mx,mn;
u=(q.top().second>>3);
mx=(q.top().second>>1)&1;
mn=q.top().second&1;
q.pop();
if (vis[u][mx][mn])
continue;
vis[u][mx][mn]=1;
for (pair<int,int> I: graph[u]) {
int v=I.first,w=I.second;
for (int i=0; i<=(mx^1); i++) {
for (int j=0; j<=(mn^1); j++) {
if (d[v][mx|i][mn|j]>d[u][mx][mn]+1ll*(1-i+j)*w) {
d[v][mx|i][mn|j]=d[u][mx][mn]+1ll*(1-i+j)*w;
q.push(make_pair(d[v][mx|i][mn|j],ID(v,(mx|i),(mn|j))));
}
}
}
}
}
for (int i=1; i<n; i++) {
printf("%lld ",d[i][1][1]);
}
puts("");
return 0;
}