当前位置: 首页 > article >正文

备战蓝桥杯---搜索(完结篇)

再看一道不完全是搜索的题:

解法1:贪心+并查集:

把冲突事件从大到小排,判断是否两个在同一集合,在的话就返回,不在的话就合并。

下面是AC代码:

#include<bits/stdc++.h>
using namespace std;
int n,m,a,b,c;
struct node{
    int x,y,qi;
}a1[100010];
int fa[50000];
bool cmp(node a,node b){
    return a.qi>b.qi;
}
int find(int x){
    if(fa[x]==x) return x;
    else return fa[x]=find(fa[x]);
}
void merge(int x,int y){
    fa[find(x)]=find(y);
}
int main(){
    cin>>n>>m;
    for(int i=1;i<=m;i++){
        scanf("%d%d%d",&a1[i].x,&a1[i].y,&a1[i].qi);
    }
    for(int i=1;i<=2*n+1;i++){
        fa[i]=i;
    }
    sort(a1+1,a1+1+m,cmp);
    int f=0;
    for(int i=1;i<=m;i++){
        int xx=a1[i].x;
        int yy=a1[i].y;
        if(find(xx)==find(yy)){
            cout<<a1[i].qi;
            f=1;
            break;
        }
        else{
            merge(xx,n+yy);
            merge(xx+n,yy);
        }
    }
    if(f==0) cout<<0;
}

解法2:二分+DFS

显然这是一个0/1单调函数,我们可以进行二分。那我们二分出值如何判断是否可行?

我们可以把有怨气值的连边,对每个联通块种的大于二分值的DFS,先把自己-》1,与他相连的赋为0,以此类推,看是否有两个0/1值相同并相连的节点。

下面是AC代码:

#include<bits/stdc++.h>
using namespace std;
#define int long long
int n,m,a,b,c,qi;
struct node{
    int aa,qi1;
};
vector<node> tu[20005];
int vis[20005];
int heibai[20005];
int dfs(int x,int fa,int mid){
    int f=0;
    vis[x]=1;
    heibai[x]=1-heibai[fa];
    for(int i=0;i<tu[x].size();i++){
    if(tu[x][i].qi1<=mid) continue;
    if(tu[x][i].aa==fa) continue;
    if(vis[tu[x][i].aa]==1&&heibai[tu[x][i].aa]==heibai[x]){
        f=1;
        continue;
    }
    if(vis[tu[x][i].aa]==1) continue;
    if(dfs(tu[x][i].aa,x,mid)==1) f=1;
   }
return f;
}
int check(int mid){
    memset(vis,0,sizeof(vis));
    memset(heibai,0,sizeof(heibai));
    int f=1;
    for(int i=1;i<=n;i++){
        if(vis[i]==1) continue;
        if(dfs(i,0,mid)==1){
            f=0;
            break;
        }
    }
    return f;
}
signed main(){
    cin>>n>>m;
    for(int i=1;i<=m;i++){
       scanf("%d%d%d",&a,&b,&c);
        tu[a].push_back({b,c});
        tu[b].push_back({a,c});
        qi=max(qi,c);
    }
    int i=0,j=qi;
    while(i<j){
        int mid=(i+j)/2;
        if(check(mid)==1) j=mid;
        else i=mid+1;
    }
  
    cout<<i;
}


http://www.kler.cn/news/233065.html

相关文章:

  • 无人机系统组装与调试,多旋翼无人机组装与调试技术详解,无人机飞控系统原理
  • 机器学习11-前馈神经网络识别手写数字1.0
  • 【OpenHarmony硬件操作】WIFI模块的操作(udp+tcp)
  • 比较Kamailio和OpenSIPS的重写contact函数
  • 华为机考入门python3--(10)牛客10-字符个数统计
  • PKI - 借助Nginx 实现Https 服务端单向认证、服务端客户端双向认证
  • 电脑通电自启动设置
  • 使用Python语言生成区块链地址
  • Android矩阵Matrix动画缩放Bitmap移动手指触点到ImageView中心位置,Kotlin
  • 力扣-137. 只出现一次的数字 II
  • 联合体知识点解析
  • 如何用Hexo搭建一个优雅的博客
  • 单片机学习笔记---DS1302时钟
  • django中实现登录
  • 微信小程序的图片色彩分析,窃取网络图片的主色调
  • Python中使用opencv-python库进行颜色检测
  • 【芯片设计- RTL 数字逻辑设计入门 7 -- 同步复位与异步复位详细介绍】
  • 使用Collections.singletonList()遇到的问题
  • Unity学习笔记之【IK反向动力学操作】
  • 20240209-最大可整分子集
  • c#: 表达式树的简化
  • 科研绘图-半小提琴图-
  • List 差集
  • 【开源】JAVA+Vue+SpringBoot实现房屋出售出租系统
  • vue父子组件通讯的几种方式总结学习
  • 基于SpringBoot的记账系统项目
  • Vagrant 虚拟机工具基本操作指南
  • echarts 一条折线图上显示不同颜色
  • 【Android】GridLayout实现等比布局
  • DC-9靶机渗透详细流程