作者: 负雪明烛
id: fuxuemingzhu
个人博客: http://fuxuemingzhu.cn/


题目地址:https://leetcode.com/problems/satisfiability-of-equality-equations/

题目描述

Given an array equations of strings that represent relationships between variables, each string equations[i] has length 4 and takes one of two different forms: "a==b" or "a!=b". Here, a and b are lowercase letters (not necessarily different) that represent one-letter variable names.

Return true if and only if it is possible to assign integers to variable names so as to satisfy all the given equations.

Example 1:

Input: ["a==b","b!=a"]
Output: false
Explanation: If we assign say, a = 1 and b = 1, then the first equation is satisfied, but not the second. There is no way to assign the variables to satisfy both equations.

Example 2:

Input: ["b==a","a==b"]
Output: true
Explanation: We could assign a = 1 and b = 1 to satisfy both equations.

Example 3:

Input: ["a==b","b==c","a==c"]
Output: true

Example 4:

Input: ["a==b","b!=c","c==a"]
Output: false

Example 5:

Input: ["c==c","b==d","x!=z"]
Output: true

Note:

  1. 1 <= equations.length <= 500
  2. equations[i].length == 4
  3. equations[i][0] and equations[i][3] are lowercase letters
  4. equationsi is either ‘=’ or ‘!’
  5. equations[i][2] is ‘=’

题目大意

给了一连串的等式和不等式,判断这些等式和不等式能否同时都成立。

解题方法

DFS

这题最难的部分就是数据抽象,其实并不难:把每个字母当做一个节点,把等号代表两个节点之间有连接,把不等号代表两个节点之间没有连接。判断这样的图是否存在。

由此可见,我们首先可以根据等号关系,把这个网的所有节点之间的连接构成一个无向图。然后对于不等号,我们记性dfs搜索,判断等号两边的节点能不能有路。如果有路代表两者通过一定的等号关系能够相互转化,那么就不符合不等号关系,返回false;如果所有不等号两边的节点都没有路,那么返回true.

这个题直接搜索的话会超时,可以把已经访问过的节点保存下来,这样的话,等我们在搜索下一条路的时候不会走已经搜索过的节点。

python代码如下:

class Solution(object):
def equationsPossible(self, equations):
"""
:type equations: List[str]
:rtype: bool
"""
self.m = collections.defaultdict(list)
for eq in equations:
if eq[1] == '=':
self.m[eq[0]].append(eq[3])
self.m[eq[3]].append(eq[0])
for eq in equations:
if eq[1] == '!':
if self.find(set(), eq[0], eq[3]) or self.find(set(), eq[3], eq[0]):
return False
return True def find(self, visited, begin, end):
if begin in visited:
return False
visited.add(begin)
if begin == end:
return True
for n in self.m[begin]:
if self.find(visited, n, end):
return True
return False

并查集

判断两个点直接是否有连接的一个更为简单的方法就是使用并查集。如果两个节点的祖先相同,说明两者之间有路。

下面是一种并查集的写法,这个写法在查找某个节点的祖先的时候做了路径压缩,使得被查找节点直接挂载到了祖先上,这样下次查找的时候就不用搜索那么多了。而合并的这一步骤,是直接查找到两个节点的左右祖先,把其中一个的祖先设置成为了另一个。

不要太纠结,记住,对于a==b,无论是a的祖先设置成b、还是把b的祖先设置成a,都不想影响查找出来的a和b的祖先是同一个元素。

python代码如下:

class Solution(object):
def equationsPossible(self, equations):
"""
:type equations: List[str]
:rtype: bool
"""
dsu = DSU()
for eq in equations:
if eq[1] == '=':
dsu.u(ord(eq[0]) - ord('a'), ord(eq[3]) - ord('a'))
for eq in equations:
if eq[1] == '!':
if dsu.f(ord(eq[0]) - ord('a')) == dsu.f(ord(eq[3]) - ord('a')):
return False
return True class DSU(object):
def __init__(self):
self.m = range(26) def f(self, x):
if self.m[x] != x:
self.m[x] = self.f(self.m[x])
return self.m[x] def u(self, x, y):
px = self.f(x)
py = self.f(y)
self.m[px] = py

C++代码如下:

class Solution {
public:
bool equationsPossible(vector<string>& equations) {
init();
for (string& s : equations) {
if (s[1] == '=') {
u(s[0] - 'a', s[3] - 'a');
}
}
for (string& s : equations) {
if (s[1] == '!') {
if (f(s[0] - 'a') == f(s[3] - 'a'))
return false;
}
}
return true;
}
private:
int _fa[26]; void init() {
for (int i = 0; i < 26; ++i) {
_fa[i] = i;
}
}
int f(int x) {
if (_fa[x] == x) return x;
return _fa[x] = f(_fa[x]);
}
void u(int a, int b) {
int pa = f(a);
int pb = f(b);
_fa[pa] = pb;
}
};

日期

2019 年 2 月 21 日 —— 一放假就再难抓紧了

【LeetCode】990. Satisfiability of Equality Equations 解题报告(C++ & python)的更多相关文章

  1. LeetCode 990. Satisfiability of Equality Equations

    原题链接在这里:https://leetcode.com/problems/satisfiability-of-equality-equations/ 题目: Given an array equat ...

  2. 【LeetCode】299. Bulls and Cows 解题报告(Python)

    [LeetCode]299. Bulls and Cows 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 题 ...

  3. 【LeetCode】743. Network Delay Time 解题报告(Python)

    [LeetCode]743. Network Delay Time 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzhu 个人博客: ht ...

  4. 【LeetCode】518. Coin Change 2 解题报告(Python)

    [LeetCode]518. Coin Change 2 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 题目 ...

  5. 【LeetCode】474. Ones and Zeroes 解题报告(Python)

    [LeetCode]474. Ones and Zeroes 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ ...

  6. 【LeetCode】731. My Calendar II 解题报告(Python)

    [LeetCode]731. My Calendar II 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 题 ...

  7. 【LeetCode】785. Is Graph Bipartite? 解题报告(Python)

    [LeetCode]785. Is Graph Bipartite? 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu. ...

  8. 【LeetCode】895. Maximum Frequency Stack 解题报告(Python)

    [LeetCode]895. Maximum Frequency Stack 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxueming ...

  9. 【LeetCode】764. Largest Plus Sign 解题报告(Python)

    [LeetCode]764. Largest Plus Sign 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn ...

随机推荐

  1. R绘图布局包 customLayout

    今天介绍一个R画图布局的包,地址如下: https://github.com/zzawadz/customLayout https://www.customlayout.zstat.pl/index. ...

  2. 二进制免编译My SQL

    一 下载 MySQL 安装包教程 https://blog.csdn.net/zhan107876/article/details/100701135 ll -h mysql-5.6.47-linux ...

  3. CSS3单行文本两端对齐

    CSS3实现单行文本两端对齐 p { height: 24px; text-align: justify; text-last-align: justify; } p::after { display ...

  4. day06 目录结构

    day06 目录结构 文件目录 /bin # 存放系统常用命令的目录 /boot # 系统引导程序+内核 /dev # 设备.光驱.硬盘 /etc # 存放系统或服务的配置文件 /home # 普通用 ...

  5. E: Unable to fetch some archives, maybe run apt-get update or try with --fix-missing

    解决办法:apt-get update或者apt-get cleanapt-get update 或者 apt-get update --fix-missing问题解析1 source本身的问题 根据 ...

  6. [转] Java中对数据进行加密的几种方法

    加密算法有很多种:这里只大约列举几例: 1:消息摘要:(数字指纹):既对一个任意长度的一个数据块进行计算,产生一个唯一指纹.MD5/SHA1发送给其他人你的信息和摘要,其他人用相同的加密方法得到摘要, ...

  7. Oracle中常用的系统表

    1.dba开头的表 dba_users 数据库用户信息 dba_segments 表段信息 dba_extents 数据区信息 dba_objects 数据库对象信息 dba_tablespaces ...

  8. ORACEL 创建DIRECTORY

    oracle要直接对文件进行读写必须先创建一个DIRECTORY. 语法如下: CREATE DIRECTORY UTL_FILE_DIR AS '/home/oracle/oradir'; 可以通过 ...

  9. Spring Boot对日志的控制

    一.logback日志技术介绍 Spring Boot中使用的日志技术为logback.其与Log4J都出自同一人,性能要优于Log4J,是Log4J的替代者. 在Spring Boot中若要使用lo ...

  10. XML名命空间

    XML的名命空间就类似于java的包,命名空间定义:xmlns:***="URI",默认命名空间定义:xmlns="URI" 引号中的URl内容用来唯一标识命名 ...