相似度

6 篇文章 0 订阅

相似度
【问题描述】
小 G 通过摆放一些城市和道路构成了一个世界地图。趁着小 G 出去玩的时候,大 G 把小 G 的世界地图上的城市全部打乱并放在了原来这些城市所在的位置(并不是一一对应),又修改了一些道路。小 G 玩完回来后发现自己的东西被打乱了,感到非常生气,但是他又被一个更有趣的问题吸引了:被修改之后的世界地图与原来的世界地图的最大相似度是多少?
(ps:相似度的定义为将城市还原后还有多少条道路和之前的道路相同)
【输入格式】
第一行为两个整数 n,m,表示一共有 n 个城市,m 条道路
接下来 m 行,每行两个整数 x,y,表示原来小 G 的世界地图中有一条道路
连接编号为 x 和 y 的两个城市。
紧接着 m 行,每行两个整数 x’,y’,表示被大 G 修改后的世界地图中有一
条道路连接编号为 x’和 y’的两个城市。
【输出格式】
一行一个整数,表示最大相似度。
【样例输入】
4 5
4 3
2 1
3 2
2 4
2 3
1 4
3 2
2 1
1 3
4 4
【样例输出】
4
【样例解释】
原图中的 1,2,3,4 号城市分别对应现在图中的 4,1,2,3
将修改后的图还原
1 4->2 1
3 2->4 3
2 1->3 2
1 3->2 4
4 4->1 1
与原图比较发现有 4 条边是一样的。
【数据规模和约定】
对于 30%的数据,1 ≤ n ≤ 3,1 ≤ m≤ 20。
对于 60%的数据,1 ≤ n ≤ 7,1 ≤ m≤ 70。
对于 100%的数据,1 ≤ n ≤ 9,1 ≤ m≤ 300。

思路:暴搜

先枚举每个点放在不同位置的不同方法,及点集的全排列
带入每张图,求相似度

#include <algorithm>
#include <iostream> 
#include <cstring>
#include <vector>
#include <cstdio>
#include <queue>
#include <cmath>
#define LL long long 
using namespace std;

const int N=1100; 
int n,m,u,v,ans,a[N][N],b[N][N],c[N];
bool f[N];
int xsd ()
{
	int num(0);
	for (int i=1;i<=n;i++)
	  for (int j=1;j<=n;j++)
		if (a[i][j]&&b[c[i]][c[j]])
		  num+=min (a[i][j],b[c[i]][c[j]]);
	return num;         //相似度
}
void dfs (int k)
{
	if (k>n)
	{
		ans=max(ans,xsd ());
		return;
	}
	for (int i=1;i<=n;i++) if (!f[i]) c[k]=i,f[i]=1,dfs(k+1),f[i]=0;      //全排列
}
int main ()
{
	freopen ("similarity.in","r",stdin);
	freopen ("similarity.out","w",stdout);
	scanf ("%d%d",&n,&m);
	for (int i=1;i<=m;i++) scanf ("%d%d",&u,&v),a[u][v]++;
	for (int i=1;i<=m;i++) scanf ("%d%d",&u,&v),b[u][v]++;
	 //注意!!!是“++”,两个点之间不一定只有一条边!!!
	dfs(1);
	printf ("%d",ans);
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值