海王算法(看完不会变成海王)

在这里插入图片描述

                                                                  💧学了海王算法会变成海王吗,它又能解决什么样的问题呢?💧          


🌷 仰望天空,妳我亦是行人.✨
🦄 个人主页——微风撞见云的博客🎐
🐳 数据结构与算法专栏的文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺
🪁 希望本文能够给读者带来一定的帮助🌸文章粗浅,敬请批评指正!🐥


文章目录

  • 🌊海王算法的概念
    • 前景提要
    • 具体做法
      • 💧find()
      • 💧在主函数这样做
  • 🌊暧昧情侣代码如下:
  • 🌊巩固加深
      • 💧邻接矩阵解法
      • 💧链式前向星解法
  • 🐳结语


🌊海王算法的概念

💧海王算法又叫 匈牙利算法 \color{#00BFFF}{匈牙利算法} 匈牙利算法是由匈牙利数学家Edmonds于1965年提出,因而得名。匈牙利算法是基于Hall定理中充分性证明的思想,它是 部图匹配 \color{#00BFFF}{部图匹配} 部图匹配最常见的算法,该算法的核心就是 寻找增广路径 \color{#00BFFF}{寻找增广路径} 寻找增广路径,它是一种用增广路径 求二分图最大匹配 \color{#00BFFF}{求二分图最大匹配} 求二分图最大匹配的算法。

前景提要

    🍊我个人很喜欢 章若楠 \color{#FF1493}{章若楠} 章若楠 许光汉 \color{#FF1493}{许光汉} 许光汉两位明星,那么,今天我就要来当个媒婆,干啥事儿呢?我要撮合他们,尽量让他们在一起。
    🍊由于我只提到了上面两位明星,那么我就用他们不同的照片来分别作为男一号、男二号、男三号、男四号女一号、女二号、女三号、女四号,可能有点不太好,但是,我目前觉得还挺好,嘿嘿,就这样吧!

    🍊今天,我是一名伟大的媒婆!我要把男主们和女主们尽最大努力撮合在一起!既然是两个人在一起,那还是得先遵循他们自己的意愿吧。下图中的连线就表示两人相互有好感,也就是说可以尝试在一起一下,我们用个小本本记录一下,看看有哪些连线。(海王算法嘛,当然有海王咯,海王们一般都不专一,一次性喜欢多个人虽然很下头,但还是比较合理的。狗头保命
在这里插入图片描述

具体做法

    🐲 这个时候,天色骤变,海面上出现了一个 巨大的怪兽 \color{#00BFFF}{巨大的怪兽} 巨大的怪兽🦕🌊🌊🌊,它咆哮着说:“我就是万恶的 海王 \color{#00BFFF}{海王} 海王(呜~~~),今天这桩事必须听我的,让我来给他们分配对象,否则我就…”。

    🏄‍♂️好吧那我们就听它的,看看它能玩出什么花样。

    🐲海王说:“第一步:我先试着给男一号找对象。”

    🏄‍♂️我给海王说:“我这里有个小本子,上面有他们彼此之间的暧昧关系💗。”

    🐲海王一看:“好家伙,到底谁是海王? 这样吧,男一号和女一号都还没确定关系,那我们就先给他们俩搓一对儿吧”。就这样,男一号和女一号暂时牵手成功了💘。(这里用粉紫色的双向箭头替换原来的连线,表示他们双向奔赴,在一起了。)
在这里插入图片描述

    🏄‍♂️紧接着,海王发现男二号和女二号也都没有对象,于是给他们俩也安排上了。男二女二暂时牵手成功💘。
在这里插入图片描述
    🏄‍♂️接下来是男三号,海王发现与男二号有好感的女一号和女二号都已经名花有主了。这可咋办呢?

    🐲海王发话了:“男三号喜欢女一号是吧,那就先把女一号和男一号拆开,重新分配一下。”(用灰色的先表示分手。)
在这里插入图片描述

    🏄‍♂️绝了,什么操作啊?人家手还没牵热乎呢!真不愧是海王,什么事都做得出来!”

    🐲海王:“你少废话,信不信我把你…”

    🏄‍♂️于是我眼睁睁看着男一号和女一号分开了。男三号和女一号在一起了。那男一号咋办呢?
在这里插入图片描述

    🐲海王:“男一号不是还喜欢女二号吗,让他们俩试试。”

    🏄‍♂️于是男二号和女二号分开了,男一号和女二号在一起了。那男二号咋办呢?
在这里插入图片描述

    🐲海王:“男二号不是还喜欢女三号吗,让他们俩试试。”

在这里插入图片描述

    🏄‍♂️于是男二号和女三号在一起了。诶嘿,你别说,你还真别说,虽然海王这样的做法很下头,但是确实多撮合了一对儿出来,还算有点东西。那我们的男四号呢?

    🐲海王看了看小本子说:“这最后一个男生,我按照刚才的转移大法,也不能给他个女生出来,毕竟如果我满足了他,那前面的男生总会有一个分手,既然不能多撮合一对儿,那我就省省功力了,放弃这个男生吧。年轻人,我就一个要求,不要做海王,别来和我抢饭碗。后会有期!🦕🌊🌊🌊”

    🏄‍♂️随着海王的离开,天空也明亮了,太阳格外耀眼,因为我在海王的帮助下,成功撮合了三对暧昧情侣!

    💧怎么样?看完海王这一顿操作,你悟出了什么道理?没错,有机会要上,别要怂,没机会也要创造机会,尽量撮合更多的暧昧情侣

    💧回顾海王的操作,我发现转移大法的精髓就是分配递归寻找

    💧我们需要用以下变量来维护这些操作,它们分别是:

boolean[][] line = new boolean[N][N];//邻接矩阵建立连线
int[] girl = new int[N];//存储每个女生被分配给了哪个男生  girl[1] = 2 --> 1号女生分配给了2号男生
boolean[] used;//这里标记过的意思是这次查找曾试图改变过该妹子的归属问题,但是没有成功,所以就不用瞎费工夫了

💧find()

private static boolean find(int x) {
        for (int j = 1; j <= m; j++) {//枚举每个女主
            if (line[x][j] && !used[j]) {//如果有暧昧并且还没有标记过
                used[j] = true;
                if (girl[j] == 0 || find(girl[j])) {//名花无主 或者 能腾出个位置来【递归实现】
                    girl[j] = x;
                    return true;
                }
            }
        }
        return false;
    }

💧在主函数这样做

 //看看能否给男生i分配一个女生
        for (int i = 1; i <= n; i++) {
            used = new boolean[N];//每次都给他new个新的,目的就是重置状态
            if (find(i)) ans++;
        }

🌊暧昧情侣代码如下:

public class 暧昧情侣 {
    static int N = (int) 1e4 + 10;
    static boolean[][] line = new boolean[N][N];//邻接矩阵建立连线
    static int[] girl = new int[N];//存储每个女生被分配给了哪个男生  girl[1] = 2 --> 1号女生分配给了2号男生
    static boolean[] used;//这里标记过的意思是这次查找曾试图改变过该妹子的归属问题,但是没有成功,所以就不用瞎费工夫了
    static int n, m, ans;
    static StreamTokenizer in = new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));

    public static void main(String[] args) throws IOException {
        in.nextToken();
        n = (int) in.nval;//男
        in.nextToken();
        m = (int) in.nval;//女
        in.nextToken();
        int k = (int) in.nval;//k条记录,表示男女之间彼此之间有好感
        int temp;
        while (k-- > 0) {
            in.nextToken();
            temp = (int) in.nval;
            in.nextToken();
            line[temp][(int) in.nval] = true;
        }
        //看看能否给男生i分配一个女生
        for (int i = 1; i <= n; i++) {
            used = new boolean[N];
            if (find(i)) ans++;
        }
        System.out.println(ans);
    }

    private static boolean find(int x) {
        for (int j = 1; j <= m; j++) {//枚举每个妹子
            if (line[x][j] && !used[j]) {//如果有暧昧并且还没有标记过
                used[j] = true;
                if (girl[j] == 0 || find(girl[j])) {//名花无主 或者 能腾出个位置来【递归实现】
                    girl[j] = x;
                    return true;
                }
            }
        }
        return false;
    }
}

相关解释:

/**
*find(3), x=3, j=1, girl[j]=1(女士1已暂时被许配给了男士1)->find(girl[j])(进入递归,看能不能给男士1>再找个对象,把女士1让出来给男士3),
*find(1), used[1]=1,所以跳过j=1,进入j=2(男士1也很欣赏女士2),girl[j]=2(但女士2已暂时被许配给了男>士2) ->
*-> find(girl[j])(进入递归,看能不能给男士2再找个对象,把女士2让出来给男士1),
*find(2), used[2]=true,跳过j=1,j=2,进入j=3(男士2也很欣赏女士3),girl[j]=0(女士3单身), 正好,>girl[j]=2(女士3许配给男士2),return true,
*此时回溯到上一步find(1)这里面,既然find(girl[j])=True(给男士2重新找了个对象女士3)了,>girl[j]=1(女士2恢复单身了,和男士1在一起了),return true
*此时再回溯到上上一步find(3)这里面,既然find(girl[j])=True(给男士1找到了新对象女士2),girl[j]=3>(女士1恢复单身了,和男士3在一起了),return true
*
*至此由男士3单身引起的一系列递归和回溯结束了,并凑成了三对情侣。
*再回到大循环下,继续向下执行就好了
*/

🌊巩固加深

题目:完美牛棚
在这里插入图片描述
在这里插入图片描述

💧邻接矩阵解法

import java.io.*;
import java.util.Arrays;

public class Main {
    static int N = 210;
    static boolean[][] line = new boolean[N][N];
    static int[] match = new int[N];
    static boolean[] vis = new boolean[N];
    static int n, m;
    static StreamTokenizer in = new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));

    public static void main(String[] args) throws IOException {
        in.nextToken();
        n = (int) in.nval;//牛
        in.nextToken();
        m = (int) in.nval;//棚
        for (int i = 1; i <= n; i++) {
            int count;
            in.nextToken();
            count = (int) in.nval;
            while (count-- > 0) {
                in.nextToken();
                line[i][(int) in.nval] = true;
            }
        }
        int res = 0;
        for (int i = 1; i <= n; i++) {
            Arrays.fill(vis, false);
            if (find(i)) res++;
        }
        System.out.println(res);
    }

    private static boolean find(int x) {
        for (int i = 1; i <= m; i++) {
            if (!vis[i] && line[x][i]) {
                vis[i] = true;
                if (match[i] == 0 || find(match[i])) {
                    match[i] = x;
                    return true;
                }
            }
        }
        return false;
    }
}

💧链式前向星解法

import java.io.*;
import java.util.Arrays;

public class Main {
    static int N = 210, M = N * N;//注意:最多有N * N条边
    static int[] match = new int[N];
    static boolean[] vis = new boolean[N];
    static int n, m;
    static int total;//第n条边,从1开始
    static int[] head = new int[M], next = new int[M], ends = new int[M];//这里不需要考虑边权
    static StreamTokenizer in = new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));

    static void add(int start, int end) {
        ends[++total] = end;
        next[total] = head[start];
        head[start] = total;
    }

    public static void main(String[] args) throws IOException {
        in.nextToken();
        n = (int) in.nval;//牛
        in.nextToken();
        m = (int) in.nval;//棚
        Arrays.fill(head, -1);
        for (int i = 1; i <= n; i++) {
            in.nextToken();
            int count = (int) in.nval;
            while (count-- > 0) {
                in.nextToken();
                add(i, (int) in.nval);
            }
        }
        int res = 0;
        for (int i = 1; i <= n; i++) {
            Arrays.fill(vis, false);
            if (find(i)) res++;
        }
        System.out.println(res);
    }

    private static boolean find(int x) {
        for (int i = head[x]; i != -1; i = next[i]) {
            int j = ends[i];
            if (!vis[j]) {
                vis[j] = true;
                if (match[j] == 0 || find(match[j])) {
                    match[j] = x;
                    return true;
                }
            }
        }
        return false;
    }
}

在这里插入图片描述


🐳结语

🐬初学一门技术时,总有些许的疑惑,别怕,它们是我们学习路上的点点繁星,帮助我们不断成长。

🐟文章粗浅,希望对大家有帮助!

🦄参考文章:趣写算法系列之–匈牙利算法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/4468.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

内存池解释及线程池(Linux)实现

1.内存池1.什么是内存池内存池是一种内存分配方式。在真正使用内存之前&#xff0c;先申请分配一定数量的、大小相等的内存块留作备用。当有新的内存需求时&#xff0c;就从内存池中分出一部分内存块&#xff0c;若内存块不够再继续申请新的内存。使用内存池的优点有&#xff1…

Pyspark_SQL3

Pyspark 注&#xff1a;大家觉得博客好的话&#xff0c;别忘了点赞收藏呀&#xff0c;本人每周都会更新关于人工智能和大数据相关的内容&#xff0c;内容多为原创&#xff0c;Python Java Scala SQL 代码&#xff0c;CV NLP 推荐系统等&#xff0c;Spark Flink Kafka Hbase Hi…

会声会影2023新版本功能详情讲解

会声会影2023Corel VideoStudio一款功能丰富的视频编辑软件。会声会影2023简单易用&#xff0c;具有史无前例的强大功能&#xff0c;拖放式标题、转场、覆叠和滤镜&#xff0c;色彩分级、动态分屏视频和新增强的遮罩创建器&#xff0c;超越基本编辑&#xff0c;实现影院级效果。…

【Django 网页Web开发】12. 实战项目:分页组件的封装 面向接口编程(05)(保姆级图文)

目录1. 对象的方式使用分页组件2. 项目结构3. 编写pagination.py3.1 pagination.py3.2 view.py4. bug修改之&#xff1a;url中搜索关键词q和page4.1 构造url的一个雏形4.2 修改我们的分页组件4.3 搜索小bug5. 应用分页组件&#xff0c;几行代码实现用户管理分页5.1 批量创建用户…

『 MySQL篇 』:MySQL 索引相关问题

目录 一 . 认识索引 二. 索引的数据结构 1 . B Tree vs Hash 2 . B Tree vs 二叉树/红黑树 3 . B 树 vs B树 三. 索引的使用 1. 索引分类 2. 索引用法 一 . 认识索引 当我们在查询一本书中的内容时 , 你会选择翻页每一页去查询呢 ? 还是说按照书的目录去找 ? 答案是…

springmvc(一)

SpringMVC是隶属于Spring框架的一部分&#xff0c;主要是用来进行Web开发&#xff0c;是对Servlet进行了封装。 对于SpringMVC我们主要学习如下内容: SpringMVC简介 请求与响应 REST风格 SSM整合(注解版) 拦截器 SpringMVC是处于Web层的框架&#xff0c;所以其主要的作用就是用…

微信小程序开发:微信小程序生命周期总结

前言 在微信小程序开发中&#xff0c;关于微信小程序API的使用是必备技能&#xff0c;但是关于微信小程序的生命周期也是首先要了解和掌握的知识点。尤其是现在的前端开发领域&#xff0c;关于前端的各种框架和技术都要会&#xff0c;而且微信小程序的语法就是JS的翻版&#xf…

Java 线程安全

一、什么是线程安全 当多个线程访问共享资源时&#xff0c;每个线程都会各自对共享资源进程操作&#xff0c;导致数据不一致&#xff0c;造成程序不能正确的得到结果&#xff0c;此时需要让多个线程排队访问共享资源&#xff0c;让线程安全&#xff0c;才能保证数据安全的被访问…

Jdk动态代理和Cglib动态代理的区别

一&#xff1a; 前言&#xff1a; 代理模式分为 静态代理 和 动态代理&#xff0c;今天我要讲的是动态代理的两种常见、也是被广泛使用的实现方式-------jdk动态代理 和 Cglib动态代理 二&#xff1a;Jdk动态代理实现分析&#xff1a; 结构示意图如下&#xff0c;我定义了一…

FrIf-FrIf_Transmit发送流程【配置参数FrIfImmediate:立即传输还是解耦传输】和代码分析

总目录链接==>> AutoSAR入门和实战系列总目录 文章目录 1 FrIf_Transmit中的 PDU 的配置的传输模式2 代码分析1 FrIf_Transmit中的 PDU 的配置的传输模式 每当FrIf的上层模块想要请求特定 PDU 的传输时,它都会调用 FrIf_Transmit 函数。调用 FrIf_Transmit的时候传递…

C语言--文件操作

目录前言什么是文件程序文件数据文件文件指针FILE结构的维护文件的打开和关闭文件的打开方式文件的顺序读写fputcfgetcfputsfgetsfprintffscanf文件流 标准输入/输出流sscanf和sprintf前言 在讲文件操作之前&#xff0c;我们先来思考这个问题&#xff1a; 我们为什么要使用文件…

大数据技术之Spark(一)——Spark概述

大数据技术之Spark&#xff08;一&#xff09;——Spark概述 文章目录前言一、Spark基础1.1 Spark是什么1.2 Spark VS Hadoop1.3 Spark优势及特点1.3.1 优秀的数据模型和丰富计算抽象1.3.3 spark的特点1.4 Spark 运行环境1.5 Spark运行架构1.5.1 Driver1.5.2 Executor1.5.3 Mas…

Java设计模式-4、适配器模式

适配器模式 在我们的应⽤程序中我们可能需要将两个不同接⼝的类来进⾏通信&#xff0c;在不 修改这两个的前提下我们可能会需要某个中间件来完成这个衔接的过程。 这个中间件就是适配器。所谓适配器模式就是将⼀个类的接⼝&#xff0c;转换成客 户期望的另⼀个接⼝。它可以让原…

【协议】03、深度解剖之HTTP协议

协议是指计算机通信网络中两台计算机之间进行通信所必须共同遵守的标准、约定或者规则的集合&#xff0c;超文本传输协议(HTTP)是一种通信协议&#xff0c;它允许将超文本标记语言(HTML)文档从Web服务器传送到客户端的浏览器。使用的默认端口为80端口。浏览器的默认端口也是80端…

【操作系统】第二章:进程管理

第二章&#xff1a;进程管理 OVERVIEW第二章&#xff1a;进程管理一、进程与线程1.进程概述&#xff08;1&#xff09;进程PCB&#xff1a;&#xff08;2&#xff09;进程的组成&#xff1a;&#xff08;3&#xff09;进程的特征&#xff1a;2.进程的状态与转换&#xff08;1&a…

基于储能进行调峰和频率调节研究【超线性增益的联合优化】(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

算法练习随记(三)

1.全排列 给定一个不含重复数字的数组 nums &#xff0c;返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3] 输出&#xff1a;[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]]示例 2&#xff1a; 输入&#x…

如何用AI玩转IG广告,打造高互动的引流营销?

如何用AI玩转IG广告&#xff0c;打造高互动的引流营销&#xff1f;相信做引流的卖家&#xff0c;都有接触过IG广告&#xff0c;然而流量引过来了&#xff0c;怎么处理客户的私信&#xff1f; 私信对话是你与粉丝培养深度关系的管道&#xff0c;好的互动不仅能养成高黏着度的铁粉…

5 个Python高级特性让你在不知不觉中成为Python高手

你已经使用 Python 编程了一段时间&#xff0c;编写脚本并解决各种问题。是你的水平出色吗&#xff1f;你可能只是在不知不觉中利用了Python的高级特性。 从闭包&#xff08;closure&#xff09;到上下文管理器&#xff08;context managers&#xff09;&#xff0c;本文给出一…

Linked List

链表在力扣上的介绍&#xff1a;链表&#xff08;Linked List&#xff09;是最简单的线性的、动态数据结构。理解它是理解树结构、图结构的基础。区别于数组&#xff0c;链表中的元素不是存储在内存中连续的一片区域&#xff0c;链表中的数据存储在每一个称之为「结点」复合区域…