31 | 深度和广度优先搜索:如何找出社交网络中的三度好友关系?

问题导入

给你一个用户,如何找出这个用户的所有三度(其中包含一度、二度和三度)好友关系?

搜索算法

算法是作用于具体数据结构之上的,深度优先搜索算法和广度优先搜索算法都是基于“图”这种数据结构的。这是因为,图这种数据结构的表达能力很强,大部分涉及搜索的场景都可以抽象成“图”。

无向图的实现代码:每个顶点都是一条链表,类似hash表


public class Graph { // 无向图private int v; // 顶点的个数private LinkedList<Integer> adj[]; // 邻接表public Graph(int v) {this.v = v;adj = new LinkedList[v];for (int i=0; i<v; ++i) {adj[i] = new LinkedList<>();}}public void addEdge(int s, int t) { // 无向图一条边存两次adj[s].add(t);adj[t].add(s);}
}

BFS

广度优先搜索(Breadth-First-Search),我们平常都简称 BFS。直观地讲,它其实就是一种“地毯式”层层推进的搜索策略,即先查找离起始顶点最近的,然后是次近的,依次往外搜索。

代码实现:


public void bfs(int s, int t) {if (s == t) return;boolean[] visited = new boolean[v];visited[s]=true;Queue<Integer> queue = new LinkedList<>();queue.add(s);int[] prev = new int[v];for (int i = 0; i < v; ++i) {prev[i] = -1;}while (queue.size() != 0) {int w = queue.poll();for (int i = 0; i < adj[w].size(); ++i) {int q = adj[w].get(i);if (!visited[q]) {prev[q] = w;if (q == t) {print(prev, s, t);return;}visited[q] = true;queue.add(q);}}}
}private void print(int[] prev, int s, int t) { // 递归打印s->t的路径if (prev[t] != -1 && t != s) {print(prev, s, prev[t]);}System.out.print(t + " ");
}

三个重要的辅助变量 visited、queue、prev:

  • visited 是用来记录已经被访问的顶点,用来避免顶点被重复访问。如果顶点 q 被访问,那相应的 visited[q]会被设置为 true。
  • queue 是一个队列,用来存储已经被访问、但相连的顶点还没有被访问的顶点。因为广度优先搜索是逐层访问的,也就是说,我们只有把第 k 层的顶点都访问完成之后,才能访问第 k+1 层的顶点。当我们访问到第 k 层的顶点的时候,我们需要把第 k 层的顶点记录下来,稍后才能通过第 k 层的顶点来找第 k+1 层的顶点。所以,我们用这个队列来实现记录的功能。
  • prev 用来记录搜索路径。当我们从顶点 s 开始,广度优先搜索到顶点 t 后,prev 数组中存储的就是搜索的路径。不过,这个路径是反向存储的。prev[w]存储的是,顶点 w 是从哪个前驱顶点遍历过来的。比如,我们通过顶点 2 的邻接表访问到顶点 3,那 prev[3]就等于 2。为了正向打印出路径,递归地来打印。

分解图:

时间复杂度:O(E)。

空间复杂度:主要在几个辅助变量 visited 数组、queue 队列、prev 数组上。这三个存储空间的大小都不会超过顶点的个数,所以空间复杂度是 O(V)

DFS

深度优先搜索(Depth-First-Search),简称 DFS。最直观的例子就是“走迷宫”。

1、深度优先搜索用的是一种比较著名的算法思想,回溯思想。

2、深度优先搜索找出来的路径,并不是顶点 s 到顶点 t 的最短路径。

代码实现


boolean found = false; // 全局变量或者类成员变量public void dfs(int s, int t) {found = false;boolean[] visited = new boolean[v];int[] prev = new int[v];for (int i = 0; i < v; ++i) {prev[i] = -1;}recurDfs(s, t, visited, prev);print(prev, s, t);
}private void recurDfs(int w, int t, boolean[] visited, int[] prev) {if (found == true) return;visited[w] = true;if (w == t) {found = true;return;}for (int i = 0; i < adj[w].size(); ++i) {int q = adj[w].get(i);if (!visited[q]) {prev[q] = w;recurDfs(q, t, visited, prev);}}
}

空间复杂度:O(V)

时间复杂度:从面画的看出,每条边最多会被访问两次,一次是遍历,一次是回退。所以,图上的深度优先搜索算法时间复杂度是 O(E),E 表示边的个数

总结

社交网络可以用图来表示。这个问题就非常适合用图的广度优先搜索算法来解决,因为广度优先搜索是层层往外推进的。第一层是1度好友,第二层是2度好友,第3层是3度好友。改造一下广度优先搜索代码,用一个数组来记录每个顶点与起始顶点的距离,非常容易就可以找出三度好友关系。

1、广度优先搜索,通俗的理解就是,地毯式层层推进,从起始顶点开始,依次往外遍历。广度优先搜索需要借助队列来实现,遍历得到的路径就是,起始顶点到终止顶点的最短路径。

2、深度优先搜索用的是回溯思想,非常适合用递归实现。换种说法,深度优先搜索是借助栈来实现的。在执行效率方面,深度优先和广度优先搜索的时间复杂度都是 O(E),空间复杂度是 O(V)。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

可能是最好理解的二叉树的层序遍历

题目描述&#xff1a;二叉树的层序遍历&#xff0c;按层数输出每一层的结果数组 代码实现 class Solution {public List<List<Integer>> levelOrder(TreeNode root) {List<List<Integer>> res new ArrayList<List<Integer>>();Queue<…

Spring学习9-MyEclipse中Spring工程使用@Resource注释的问题

在MyEclipse 的Spring工程中&#xff0c;有时候要使用Resource注释来驱动Spring配置。但是在MyEclipse添加Spring开发能力的操作中&#xff0c;并没有 把相关的库添加到工程的classpath中&#xff0c;所以使用该注解时会产生找不到类的错误&#xff0c;这是由于MyEclipse带的Sp…

java 8的一些新用法

http://www.manongjc.com/article/44995.html 1.排序 按升序排 &#xff1a;Collections.sort(shopInfoVO1List, Comparator.comparing(ShopInfoVO1::getDistance)); 按降序排: Collections.sort(shopInfoVO1List, Comparator.comparing(ShopInfoVO1::getDistance).reversed()…

jdk1.8对synchronized锁的优化

synchronized 锁的优化&#xff1a;锁的四种状态-无锁&#xff0c;偏向锁、轻量级锁&#xff0c;重量级锁 1、偏向锁&#xff1a;原因是大多数时候是不存在锁竞争的&#xff0c;常常是一个线程多次获得同一个锁&#xff0c;因此如果每次都要竞争锁会增大很多没有必要付出的代价…

03.校准时间

复制代码保存为vbs,js等文件的时候,报莫名其妙的错误,把文件的编码格式保存为ANSI , utf8不行;VBS校准系统时间在有线网络行&#xff0c;无线就不行&#xff0c;请修改&#xff01;&#xff01;已解决&#xff01; - VBS求助&讨论 - 批处理之家 批处理_BAT_CMD_DOS_VBS_Per…

淘宝开发平台 java 调用实例

Java调用示例代码 更新日期&#xff1a;2016-02-06访问次数&#xff1a;53432 主要步骤 填充公共参数 填充业务参数 计算请求签名 发起API调用 获取API结果 示例代码 import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.…

LeetCode 293. Flip Game

原题链接在这里&#xff1a;https://leetcode.com/problems/flip-game/description/ 题目&#xff1a; You are playing the following Flip Game with your friend: Given a string that contains only these two characters: and -, you and your friend take turns to flip…

Redis缓存,你真的懂了吗

为什么要用缓存&#xff08;缓存的优点、场景&#xff09; &#xff08;1&#xff09;在项目中缓存是如何使用的&#xff1f; 结合你自己项目的业务来&#xff0c;你如果用了那恭喜你&#xff0c;你如果没用那不好意思&#xff0c;你硬加也得加一个场景吧&#xff01; &…

Java sdk 调用淘宝开发平台

public static void main(String[] args) throws Exception { // TOP服务地址&#xff0c;正式环境需要设置为http://gw.api.taobao.com/router/rest String serverUrl “http://gw.api.tbsandbox.com/router/rest”; String appKey “test”; // 可替换为您的沙箱环境应用的…

编写一个函数func(),将此函数的输入参数(int型)逆序输出显示,如54321 – 12345,要求使用递归,并且函数体代码不超过8行...

public class Test{  //中间变量private String res "0";  //方法public int func(int i){if(i>0){int temp i%10;res resString.valueOf(temp);func(i/10);}return Integer.valueOf(res);}public static void main(String[] args){Test tnew Test();int a…

你会用Java实现两个大数相加吗

两个大数相加(Java)* 1、是整数&#xff1b;* 2、两个数无限大&#xff0c;long都装不下&#xff1b;* 3、不能用BigInteger&#xff1b;* 4、不能用任何包装类提供的运算方法&#xff1b;* 5、两个数都是以字符串的方式提供。 * 思路&#xff1a;* 字符串逐位相加&#xff0c;…

获取淘宝开发平台的sessionKey

淘宝API调用 申请 获取session key 在调用淘宝的API时&#xff0c;我们都会用到appkey,appsecret,appsession。 1、我们申请应用就会有appkey和appsecret了 2、正式环境下获取SessionKey 注意&#xff1a;web插件平台应用和web其它应用在正式环境下是同样的获取方法 1&…

PERL 实现微信登录

get 请求: https://login.weixin.qq.com/jslogin? appidwx782c26e4c19acffb &redirect_urihttps%3A%2F%2Fwx.qq.com%2Fcgi-bin%2Fmmwebwx-bin%2Fwebwxnewloginpage &funnew &langzh_CN &_1455501911998参数: _ 1455501911998 appid wx782…

安装git客户端

https://blog.csdn.net/sinat_16998945/article/details/81062278 https://blog.csdn.net/qq_38225558/article/details/86220668