2022-01-11每日刷题打卡
AcWing——y总算法课
846. 树的重心 - AcWing题库
给定一颗树,树中包含 n 个结点(编号 1∼n)和 n−1 条无向边。
请你找到树的重心,并输出将重心删除后,剩余各个连通块中点数的最大值。
重心定义:重心是指树中的一个结点,如果将这个点删除后,剩余各个连通块中点数的最大值最小,那么这个节点被称为树的重心。
输入格式
第一行包含整数 n,表示树的结点数。
接下来 n−1 行,每行包含两个整数 a 和 b,表示点 a 和点 b 之间存在一条边。
输出格式
输出一个整数 m,表示将重心删除后,剩余各个连通块中点数的最大值。
数据范围
1≤n≤10^5
输入样例
9
1 2
1 7
1 4
2 8
2 5
4 3
3 9
4 6
输出样例:
4
#include<iostream>
#include<algorithm>
#include<string.h>
using namespace std;
const int N=100010;
const int M=N*2;
int idx,e[M],ne[M],h[N],ans=N,n;
bool flag[N];
void add(int a,int b)
{
e[idx]=b,ne[idx]=h[a],h[a]=idx++;
}
int dfs(int u)
{
flag[u]=true;
int sum=1,res=0;
for(int i=h[u];i!=-1;i=ne[i])
{
int j=e[i];
if(!flag[j])
{
int s=dfs(j);
res=max(res,s);
sum+=s;
}
}
res=max(res,n-sum);
ans=min(res,ans);
return sum;
}
int main()
{
memset(h,-1,sizeof(h));
int a,b;
cin>>n;
for(int i=0;i<n-1;i++)
{
cin>>a>>b;
add(a,b);
add(b,a);
}
dfs(1);
cout<<ans<<endl;
return 0;
}
847. 图中点的层次 - AcWing题库
给定一个 n 个点 m 条边的有向图,图中可能存在重边和自环。
所有边的长度都是 1,点的编号为 1∼n。
请你求出 11 号点到 n 号点的最短距离,如果从 1 号点无法走到 n 号点,输出 −1。
输入格式
第一行包含两个整数 n 和 m。
接下来 mm 行,每行包含两个整数 a 和 b,表示存在一条从 a 走到 b 的长度为 11 的边。
输出格式
输出一个整数,表示 1 号点到 n 号点的最短距离。
数据范围
1≤n,m≤10^5
输入样例:
4 5
1 2
2 3
3 4
1 3
1 4
输出样例:
1
#include<iostream>
using namespace std;
#include<string.h>
const int N=100010;
int idx,h[N],e[N],ne[N],n,m;
int q[N],d[N];
void add(int a,int b)
{
e[idx]=b,ne[idx]=h[a],h[a]=idx++;
}
int bfs()
{
int tt=0,hh=0;
q[0]=1;
memset(d,-1,sizeof d);
d[1]=0;
while(hh<=tt)
{
int t=q[hh++];
for(int i=h[t];i!=-1;i=ne[i])
{
int j=e[i];
if(d[j]==-1)
{
d[j]=d[t]+1;
q[++tt]=j;
}
}
}
return d[n];
}
int main()
{
cin>>n>>m;
memset(h,-1,sizeof h);
for(int i=0;i<m;i++)
{
int a,b;
cin>>a>>b;
add(a,b);
}
cout<<bfs()<<endl;
return 0;
}
逆序对
问题描述
给定一个长度为n的排列,求其逆序对数
输入格式
第1行输入一个数n表示排列长度
接下来一行n个用空格隔开的数,表示这个排列
输出格式
输出逆序对总数
样例输入
4
1 2 4 3
样例输出
1
#include<iostream>
using namespace std;
const int N=1e6+5;
int n;
long long res=0;
int arr[N],tmp[N];
void merge_sort(int l,int r)
{
if (l >= r)return;
int mid = (l + r) / 2;
merge_sort(l, mid);
merge_sort(mid + 1, r);
int i = l, j = mid + 1, k = 0;
while (i <= mid && j <= r)
{
if (arr[i] <= arr[j])tmp[k++] = arr[i++];
else
{
tmp[k++] = arr[j++];
res += mid - i + 1;
}
}
while (i <= mid)tmp[k++] = arr[i++];
while (j <= r)tmp[k++] = arr[j++];
for (int i = l, j = 0; i <= r; i++, j++)
arr[i] = tmp[j];
}
int main()
{
scanf("%d",&n);
for(int i=0;i<n;i++)
scanf("%d",&arr[i]);
merge_sort(0,n-1);
cout<<res;
return 0;
}