2.15学习总结

2.15
1.聪明的质监员(二分+前缀和)
2.村村通(并查集)
3.玉蟾宫(悬线法DP)
4.随机排列(树状数组逆序对问题)
5.增进感情(DFS)

6.医院设置(floyd)

聪明的质监员https://www.luogu.com.cn/problem/P1314

题目描述

小T 是一名质量监督员,最近负责检验一批矿产的质量。这批矿产共有 �n 个矿石,从 11 到 �n 逐一编号,每个矿石都有自己的重量 ��wi​ 以及价值 ��vi​ 。检验矿产的流程是:

  1. 给定�m 个区间 [��,��][li​,ri​];

  2. 选出一个参数 �W;

  3. 对于一个区间 [��,��][li​,ri​],计算矿石在这个区间上的检验值 ��yi​:

��=∑�=����[��≥�]×∑�=����[��≥�]��yi​=j=li​∑ri​​[wj​≥W]×j=li​∑ri​​[wj​≥W]vj​

其中 �j 为矿石编号。

这批矿产的检验结果 �y 为各个区间的检验值之和。即:∑�=1���i=1∑m​yi​

若这批矿产的检验结果与所给标准值 �s 相差太多,就需要再去检验另一批矿产。小T 不想费时间去检验另一批矿产,所以他想通过调整参数 �W 的值,让检验结果尽可能的靠近标准值 �s,即使得 ∣�−�∣∣s−y∣ 最小。请你帮忙求出这个最小值。

输入格式

第一行包含三个整数 �,�,�n,m,s,分别表示矿石的个数、区间的个数和标准值。

接下来的 �n 行,每行两个整数,中间用空格隔开,第 �+1i+1 行表示 �i 号矿石的重量 ��wi​ 和价值 ��vi​。

接下来的 �m 行,表示区间,每行两个整数,中间用空格隔开,第 �+�+1i+n+1 行表示区间 [��,��][li​,ri​] 的两个端点 ��li​ 和 ��ri​。注意:不同区间可能重合或相互重叠。

输出格式

一个整数,表示所求的最小值。

输入输出样例

输入 #1复制

5 3 15
1 5
2 5
3 5
4 5
5 5
1 5
2 4
3 3 

输出 #1复制

10

说明/提示

【输入输出样例说明】

当 �W 选 44 的时候,三个区间上检验值分别为 20,5,020,5,0 ,这批矿产的检验结果为 2525,此时与标准值 �S 相差最小为 1010。

【数据范围】

对于 10%10% 的数据,有 1≤�,�≤101≤n,m≤10;

对于 30%30%的数据,有 1≤�,�≤5001≤n,m≤500 ;

对于 50%50% 的数据,有 1≤�,�≤5,0001≤n,m≤5,000;

对于 70%70% 的数据,有 1≤�,�≤10,0001≤n,m≤10,000 ;

对于 100%100% 的数据,有 1≤�,�≤200,0001≤n,m≤200,000,0<��,��≤1060

思路:每次找的时候用前缀和存,不然会TLE

#include 
using namespace std;
#define lowbit(x) (x& - (x))
#define int long long
#define INF 0x3f3f3f3f3f3f3f3f
const int N=2e5+5;

int n,m,s,min1=INF,maxn,minn=INF,sum,ans;	//矿石,区间,标准值

int w[N],v[N],l[N],r[N],pre_v[N],pre_n[N];


bool check(int k){
	ans=0,sum=0;
	memset(pre_v,0,sizeof(pre_v));
	memset(pre_n,0,sizeof(pre_n));
	for (int i=1;i<=n;++i){
		if (w[i]>=k){
			pre_v[i]=pre_v[i-1]+v[i];
			pre_n[i]=pre_n[i-1]+1;
		}else{
			pre_v[i]=pre_v[i-1];
			pre_n[i]=pre_n[i-1];
		}
	}
	for (int i=1;i<=m;++i){
		ans+=(pre_n[r[i]]-pre_n[l[i]-1])*(pre_v[r[i]]-pre_v[l[i]-1]);
	}
	sum=abs(ans-s);
	if (ans>s) return true;
	else return false;
}

signed main(){
	cin>>n>>m>>s;
	for (int i=1;i<=n;++i){
		cin>>w[i]>>v[i];
		maxn=max(maxn,w[i]);
		minn=min(minn,w[i]);
	} 
	for (int i=1;i<=m;++i){
		cin>>l[i]>>r[i];
	}
	minn=minn-1,maxn=maxn+2;
	while (minn<=maxn){
		int mid=minn+maxn>>1;
		if (check(mid)) minn=mid+1;
		else maxn=mid-1;
		if (sum
村村通https://www.luogu.com.cn/problem/P1536

题目描述

某市调查城镇交通状况,得到现有城镇道路统计表。表中列出了每条道路直接连通的城镇。市政府 "村村通工程" 的目标是使全市任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要相互之间可达即可)。请你计算出最少还需要建设多少条道路?

输入格式

输入包含若干组测试数据,每组测试数据的第一行给出两个用空格隔开的正整数,分别是城镇数目 �n 和道路数目 �m ;随后的 �m 行对应 �m 条道路,每行给出一对用空格隔开的正整数,分别是该条道路直接相连的两个城镇的编号。简单起见,城镇从 11 到 �n 编号。

注意:两个城市间可以有多条道路相通。

在输入数据的最后,为一行一个整数 00,代表测试数据的结尾。

输出格式

对于每组数据,对应一行一个整数。表示最少还需要建设的道路数目。

输入输出样例

输入 #1复制

4 2
1 3
4 3
3 3
1 2
1 3
2 3
5 2
1 2
3 5
999 0
0

输出 #1复制

1
0
2
998

说明/提示

数据规模与约定

对于 100%100% 的数据,保证 1≤�<10001≤n<1000 。

思路:并查集,把所有的关系都连接起来,然后遍历所有情况,找到没有连接的,计数器自增

#include 
using namespace std;
#define lowbit(x) (x& - (x))
#define int long long
#define INF 0x3f3f3f3f3f3f3f3f
const int N=2e5+5;

int f[N],n,m,cnt,p;

int find(int x){
	if (f[x]==x) return x;
	else if (f[x]!=x){
		f[x]=find(f[x]);
		return f[x];
	}
}

void merge(int i,int j){
	f[find(i)]=find(j);
}

signed main(){
	while (1){
		cnt=0;
		cin>>n;
		if (n==0) return 0;
		cin>>m;
		for (int i=1;i<=n;++i) f[i]=i;
		for (int i=0;i>a>>b;
			merge(a,b);
		}
		for (int i=1;i<=n;++i){
			if (find(i)!=find(p)){
				cnt++;
				merge(i,p);
			}
		}
		cout<
玉蟾宫https://www.luogu.com.cn/problem/P4147

题目背景

有一天,小猫 rainbow 和 freda 来到了湘西张家界的天门山玉蟾宫,玉蟾宫宫主蓝兔盛情地款待了它们,并赐予它们一片土地。

题目描述

这片土地被分成 �×�N×M 个格子,每个格子里写着 'R' 或者 'F',R 代表这块土地被赐予了 rainbow,F 代表这块土地被赐予了 freda。

现在 freda 要在这里卖萌。。。它要找一块矩形土地,要求这片土地都标着 'F' 并且面积最大。

但是 rainbow 和 freda 的 OI 水平都弱爆了,找不出这块土地,而蓝兔也想看 freda 卖萌(她显然是不会编程的……),所以它们决定,如果你找到的土地面积为 �S,它们每人给你 �S 两银子。

输入格式

第一行两个整数 �N,�M,表示矩形土地有 �N 行 �M 列。

接下来 �N 行,每行 �M 个用空格隔开的字符 'F' 或 'R',描述了矩形土地。

输出格式

输出一个整数,表示你能得到多少银子,即 (3×最大 ’F’ 矩形土地面积3×最大 ’F’ 矩形土地面积) 的值。

输入输出样例

输入 #1复制

5 6
R F F F F F
F F F F F F
R R R F F F
F F F F F F
F F F F F F

输出 #1复制

45

说明/提示

对于 50%50% 的数据,1≤�,�≤2001≤N,M≤200。
对于 100%100% 的数据,1≤�,�≤10001≤N,M≤1000。

思路:用悬线法写,l数组存每个点向左最多可以到哪里,r数组为右,h数组存的是向上最多可以到哪里,但是在找h数组的时候,需要同时判断垂直的点,lr数组的关系

#include 
using namespace std;
#define lowbit(x) (x& - (x))
#define int long long
#define INF 0x3f3f3f3f3f3f3f3f
const int N=1010;

int h[N][N],m,n,l[N][N],r[N][N];
char a[N][N];

signed main(){
	cin>>n>>m;
	for (int i=1;i<=n;++i){
		for (int j=1;j<=m;++j){
			cin>>a[i][j];
			if (a[i][j]=='F')h[i][j]=1;
			l[i][j]=j;
			r[i][j]=j;
		}
	}
	for (int i=1;i<=n;++i){
		for (int j=2;j<=m;++j){
			if (a[i][j]=='F' && a[i][j-1]=='F'){
				l[i][j]=l[i][j-1];
			}
		}
		for (int j=m-1;j>=1;--j){
			if (a[i][j]=='F' && a[i][j+1]=='F'){
				r[i][j]=r[i][j+1];
			}
		}
	}
	int ans=0;
	for (int i=1;i<=n;++i){
		for (int j=1;j<=m;++j){
			if (a[i][j]=='F' && a[i-1][j]=='F'){
				h[i][j]=h[i-1][j]+1;
				l[i][j]=max(l[i][j],l[i-1][j]);
				r[i][j]=min(r[i][j],r[i-1][j]);
			}
			if (a[i][j]=='F')ans=max(ans,(r[i][j]-l[i][j]+1)*h[i][j]);
		}
	}
	cout<
随机排列https://www.acwing.com/problem/content/5469/

给定一个 1∼n1∼� 的排列 a1,a2,…,an�1,�2,…,��。

我们规定,交换操作指从排列中随机选择两个不同元素并交换彼此位置。

给定两种打乱排列的方式:

  1. 对排列连续进行 3n3� 次交换操作。
  2. 对排列连续进行 7n+17�+1 次交换操作。

已知,给定排列 a1∼an�1∼�� 就是由 1,2,…,n1,2,…,� 经过上述两种打乱方式之一得到的。

请你判断,给定排列具体是由哪一种打乱方式得到的。

输入格式

第一行包含整数 n�。

第二行包含 n� 个整数 a1,a2,…,an�1,�2,…,��。

输出格式

如果给定排列是由方式 11 打乱得到的,则输出 11,如果给定排列是由方式 22 打乱得到的,则输出 22。

保证给定排列一定是由两种打乱方式之一得到的。

数据范围

前 33 个测试点满足 2≤n≤102≤�≤10。
所有测试点满足 2≤n≤1062≤�≤106,保证 a1∼an�1∼�� 是一个 1∼n1∼� 的排列。

输入样例:
5
2 4 5 1 3
输出样例:
1

思路:本质上是找逆序数,由于每次改变都会改变逆序数的奇偶性,所以,当前数组的逆序数个数的奇偶性与改变次数的奇偶性相同

#include 
#include 
#include 
using namespace std;

const int N = 1e6+5;
int tr[N],w[N];

int lowbit(int x){
    return x & -x;
}

void update(int x, int k){
    while(x<=N){
        tr[x]+=k;
        x+=lowbit(x);
    }
}

int query(int x){
    int res=0;
    while (x>0){
        res+=tr[x];
        x-=lowbit(x);
    }
    return res;
}

int main()
{
    int n;
    cin>>n;
    int res=0;
    for (int i = 0; i < n; ++ i ){
        int x;
        cin>>x;
        res=(res+query(n)-query(x))%2;
        update(x,1);
    }
    if (res== 3*n%2) cout<<1;
    else cout << 2;
}
增进感情https://www.luogu.com.cn/problem/P2080

题目背景

小明和小红的感情,是慢慢发展起来的。

题目描述

他们对对方分别有一个好感值。定义两人的亲密程度为两人的好感值之和。

如果他们的亲密程度达到 �v,则他们将走到一起。他们以后的生活将取决于两人的好感值之差的绝对值,这个值越小,他们的生活将越幸福。

现在,他们对对方的好感值都为 00,小明有 �n 件事可以干,每件事可以增加他对小红的好感 ��ai​ 点,并且增加小红对他的好感 ��bi​ 点。(可能为负数)

小明可以任选一些事做,请你帮小明求出怎样才能让他们的生活更加幸福(求出两人在一起的前提下,好感值之差的最小绝对值即可)。

输入格式

第一行,两个正整数 �,�n,v。

之后 �n 行,每行两个空格隔开的整数 ��,��ai​,bi​。

输出格式

一行,一个非负整数,表示两人在一起的前提下,好感值之差的最小绝对值。如果无论如何两人也无法在一起,输出 -1

输入输出样例

输入 #1复制

4 15
5 6
-1 8
7 2
1 0

输出 #1复制

3

说明/提示

数据范围与约定

  • 对于 20%20% 数据,保证 �≤10n≤10。

  • 对于 100%100% 数据,保证 1≤�≤301≤n≤30,1≤∣��∣,∣��∣≤1001≤∣ai​∣,∣bi​∣≤100。

#include 
using namespace std;
#define lowbit(x) (x& - (x))
#define int long long
#define INF 0x3f3f3f3f3f3f3f3f

const int N=35;

int n,v,ans=INF,l,r;
int a[N],b[N],vis[N];

void dfs(int idx){
	if (l+r>=v){
		ans=min(ans,abs(l-r));
	}
	if (idx>n || ans==0){
		return ;
	}
	for (int i=idx;i<=n;++i){
		if (!vis[i]){
			vis[i]=1;
			l+=a[i],r+=b[i];
			dfs(i+1);
			l-=a[i],r-=b[i];
			vis[i]=0;
		}
	}
}

signed main(){
	cin>>n>>v; 	//v是最大的好感度
	for (int i=1;i<=n;++i){
		cin>>a[i]>>b[i];
	} 
	dfs(1);
	if (ans==INF) cout<<-1;
	else cout<
医院设置https://www.luogu.com.cn/problem/P1364

题目描述

设有一棵二叉树,如图:

2.15学习总结_第1张图片

其中,圈中的数字表示结点中居民的人口。圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为 11。如上图中,若医院建在 11 处,则距离和 =4+12+2×20+2×40=136=4+12+2×20+2×40=136;若医院建在 33 处,则距离和 =4×2+13+20+40=81=4×2+13+20+40=81。

输入格式

第一行一个整数 �n,表示树的结点数。

接下来的 �n 行每行描述了一个结点的状况,包含三个整数 �,�,�w,u,v,其中 �w 为居民人口数,�u 为左链接(为 00 表示无链接),�v 为右链接(为 00 表示无链接)。

输出格式

一个整数,表示最小距离和。

输入输出样例

输入 #1复制

5     
13 2 3
4 0 0
12 4 5
20 0 0
40 0 0

输出 #1复制

81

说明/提示

数据规模与约定

对于 100%100% 的数据,保证 1≤�≤1001≤n≤100,0≤�,�≤�0≤u,v≤n,1≤�≤1051≤w≤105。

思路:n<=100数据量小,可以用floyd算法,找到最短路径,然后根据权,再遍历所有的点找到最适合的点

#include 
using namespace std;
#define lowbit(x) (x& - (x))
#define int long long
#define INF 0x3f3f3f3f3f3f3f3f

const int N=105;

int n,a[N][N],w[N];

signed main(){
	cin>>n;
	for (int i=1;i<=n;++i){
		for (int j=1;j<=n;++j){
			if (i==j) a[i][j]=0;
			else a[i][j]=INF;
		}
	}
	for (int i=1;i<=n;++i){
		int l,r;
		cin>>w[i]>>l>>r;
		if (l>0) a[i][l]=a[l][i]=1;
		if (r>0) a[i][r]=a[r][i]=1;
	}
	for (int k=1;k<=n;++k){
		for (int i=1;i<=n;++i){
			for (int j=1;j<=n;++j){
				a[i][j]=min(a[i][j],a[i][k]+a[k][j]);
			}
		}
	}
	int sum=INF;
	for (int i=1;i<=n;++i){
		int cnt=0;
		for (int j=1;j<=n;++j){
			cnt+=a[i][j]*w[j];	
		}
		sum=min(sum,cnt);
	}
	cout<

你可能感兴趣的:(学习,深度优先,算法)