问题:课程表

现在你总共有 n 门课需要选,记为 0 到 n-1

在选修某些课程之前需要一些先修课程。 例如,想要学习课程 0 ,你需要先完成课程 1 ,我们用一个匹配来表示他们: [0,1]

给定课程总量以及它们的先决条件,判断是否可能完成所有课程的学习?

示例 1:

输入: 2, [[1,0]]
输出: true
解释: 总共有 2 门课程。学习课程 1 之前,你需要完成课程 0。所以这是可能的。

示例 2:

输入: 2, [[1,0],[0,1]]
输出: false
解释: 总共有 2 门课程。学习课程 1 之前,你需要先完成​课程 0;并且学习课程 0 之前,你还应先完成课程 1。这是不可能的。

说明:

  1. 输入的先决条件是由边缘列表表示的图形,而不是邻接矩阵。详情请参见图的表示法
  2. 你可以假定输入的先决条件中没有重复的边。

提示:

  1. 这个问题相当于查找一个循环是否存在于有向图中。如果存在循环,则不存在拓扑排序,因此不可能选取所有课程进行学习。
  2. 通过 DFS 进行拓扑排序 - 一个关于Coursera的精彩视频教程(21分钟),介绍拓扑排序的基本概念。
  3. 拓扑排序也可以通过 BFS 完成。

链接:https://leetcode-cn.com/problems/course-schedule/description/

分析:

1,一门课的先修课可能有多个,只要有一个没学就没法学

2,如果进行一轮后,没有新增学到的课,那么后续也不会再增加了

3,如果学到的课就是课程数量,则学了所有的,否则学不完。

所以可以设定如下数据结构:

class Class
{
public:
int Num=INT_MAX; //为止
int Status=; // 未学习
vector<int> dps;
};

Num是课程编号,Status是课程状态,0表示没学,1表示学了,dps表示以来课程。

可以假设所有的课程都不依赖任何课,初始化完成后,根据依赖关系更新,然后找到不依赖其他课的课程,

以没有先修课的课程为基础,持续更新, 直到不再新增学到的课程位置。

更新方式为:如果一个课程的先修课都已经学过了,这门课也可以学习。

AC Code:

class Class
{
public:
int Num=INT_MAX; //为止
int Status=; // 未学习
vector<int> dps;
};
class Solution {
public: bool canFinish(int numCourses, vector<pair<int, int>>& prerequisites) {
bool ret = false;
//如果能找到一个起始课程,即不依赖任何课程的,从这个开始学习所有
//如果找不到,则不能进行
vector<Class> classes;
for (int i = ; i < numCourses; i++)
{
Class tmp;
tmp.Num = i;
tmp.Status = ;
tmp.dps = vector<int>();
classes.emplace_back(tmp);
}
int n = prerequisites.size();
sort(prerequisites.begin(), prerequisites.end());
for (int i = ; i <n; i++)
{
int tmpnum = prerequisites[i].first;
int depnum = prerequisites[i].second;
for (int j = ; j < classes.size(); j++)
{
if (classes[j].Num == tmpnum)
{
classes[j].dps.emplace_back(depnum);
break;
}
}
} //找到所有的不依赖其他的课程,并且从哪些出发得打所有的不依赖的课程
int learned = ;
vector<int> learnedclass;
for (int i = ; i < classes.size(); i++)
{
if (classes[i].dps.size() == )
{
classes[i].Status = ;
learned++;
learnedclass.emplace_back(classes[i].Num);
}
}
while (true)
{
int currentlearned = learned;
for (int i = ; i < classes.size(); i++)
{
if (classes[i].Status == )
{
continue;
}
else
{
//这门课没学,看先修课有没有学
vector<int> tmpre = classes[i].dps;
int learnflag = ;
for (int j = ; j < tmpre.size(); j++)
{
if (find(learnedclass.begin(), learnedclass.end(), tmpre[j]) == learnedclass.end())
{
learnflag = ; //有先修课没学,学不了了
break;
}
}
if (learnflag == )
{
//这门课可以学
classes[i].Status = ;
learned++;
learnedclass.emplace_back(classes[i].Num);
} }
}
if (currentlearned == learned)
{
//没有增加学习的课程,可以结束了
//if(learned == )
return learned == numCourses;
}
} return ret;
}
};

其他:

1.QQ群里有人提到这个题,周赛第四题没搞懂dp状态转移方程,解决这个问题消磨时间吧。

提示中有提到图论等知识,不过也就学数据结构的时候学过这个,平时不怎么用,或者说不知道主动去用,没用图论的知识,虽然过了,效率比较低,以后有机会好好补补相关知识。

2.太冷动手,没太大兴趣看了,不过可以优化的地方有,一门课是否能学,看其先修课是否已经学过了,那么下一次更新的时候,再次查看就浪费时间,可以看缺了哪些课,这样待处理的数据会逐渐减少避免重复运算。

3.用时最短code:

 static const auto _ = []()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
return nullptr;
}(); class Solution {
vector<int> *vex; //vex[a]={b,c}:要学习a,须先学b,c
char *visit; //0:从未访问,1:正在访问,2:曾经访问
//若成环,返回false
bool DFS(int vID){
if(visit[vID]==)
return false; //成环了
if(visit[vID]==){ //0:从未访问
visit[vID]=; //1:正在访问
for(int i=;i<vex[vID].size();++i){
if(!DFS(vex[vID][i]))return false; //成环了
}
visit[vID]=; //2:曾经访问
}
return true; //访问过了
}
public:
bool canFinish(int numCourses, vector<pair<int, int>>& prerequisites) {
vex= new vector<int>[numCourses];
visit= new char[numCourses];
fill(visit,visit+numCourses,); //0:从未访问
for(int i=;i<prerequisites.size();++i){
auto &edge=prerequisites[i];
vex[edge.first].push_back(edge.second); //[1,0],学习课程 1 之前,你需要完成课程 0。
}
//逐个访问
for(int i=;i<numCourses;++i){
if(!DFS(i))return false;
}
return true; //没有成环即可
}
};

LeetCode207 课程表的更多相关文章

  1. [Swift]LeetCode207. 课程表 | Course Schedule

    There are a total of n courses you have to take, labeled from 0 to n-1. Some courses may have prereq ...

  2. Leetcode207. Course Schedule课程表

    现在你总共有 n 门课需要选,记为 0 到 n-1. 在选修某些课程之前需要一些先修课程. 例如,想要学习课程 0 ,你需要先完成课程 1 ,我们用一个匹配来表示他们: [0,1] 给定课程总量以及它 ...

  3. sql面试题(学生表_课程表_成绩表_教师表)

    原帖链接:http://bbs.csdn.net/topics/280002741 表架构 Student(S#,Sname,Sage,Ssex) 学生表 Course(C#,Cname,T#) 课程 ...

  4. 设有一数据库,包括四个表:学生表(Student)、课程表(Course)、成绩表(Score)以及教师信息表(Teacher)。

    一.            设有一数据库,包括四个表:学生表(Student).课程表(Course).成绩表(Score)以及教师信息表(Teacher).四个表的结构分别如表1-1的表(一)~表( ...

  5. [deviceone开发]-课程表的例子

    一.简介 这个例子是根据一个真实app的一个页面的需求来实现的demo,通过动态add ui的方式,动态bind数据构建一个完整的课程表示例.示例并不完善,但是可以给大家一个启发. 二.效果图 三.相 ...

  6. android课程表的实现

    //图片下方的码段主要实现了课程表所要显示的基本布局,采用ondraw的方法. //别的内容可以根据自己兴趣添加,下面是本人做的,仅供参考.                  package com. ...

  7. 爬虫再探实战(五)———爬取APP数据——超级课程表【四】——情感分析

    仔细看的话,会发现之前的词频分析并没有什么卵用...文本分析真正的大哥是NLP,不过,这个坑太大,小白不大敢跳...不过还是忍不住在坑边上往下瞅瞅2333. 言归正传,今天刚了解到boson公司有py ...

  8. SQL 存储过程(学生,课程表,选修表)

    SQL 存储过程(学生,课程表,选修表) 一.存储过程的分类 在SQL Server中存储过程分过两类: 1)系统存储过程("sp_"作为前缀) 2)用户自定义存储过程 二.创建和 ...

  9. SQL 触发器(学生,课程表,选修表)

    SQL 触发器(学生,课程表,选修表) 触发器是一种特殊类型的存储过程,它不由用户通过命令来执行,而是在用户对表执行了插入,删除或修改表中数据等操作时激活执行.可以这样形容:存储过程像一个遥控炸弹,我 ...

随机推荐

  1. 60、Docker 学习笔记(CentOS 7.1)

    #基本概念 -x86_64-minimal.tar.gz | docker import - centos:v7.mini``` 然后查看导入的镜像: ##上传镜像 >用户可以通过 docker ...

  2. POJ 2594 —— Treasure Exploration——————【最小路径覆盖、可重点、floyd传递闭包】

    Treasure Exploration Time Limit:6000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64 ...

  3. DevExtreme 搭建Node.js开发环境

    简介 DevExtreme is a component suite for creating highly responsive web applications for touch devices ...

  4. Hibernate多对多删除问题的解决

    原出处:http://superleo.iteye.com/blog/154587 Hibernate多对多的例子不少,但仔细一看,大多数都是保存的,删除谈的少,但问题还不少,因此有必须简单测试一下, ...

  5. 在mac上使用github for mac 创建并上传项目

    1.下载github for mac https://mac.github.com/ 2.登陆 偏好设置 3.用Xcode 创建一个项目,勾上“create local git respository ...

  6. #CSS的盒子模型、元素类型

    CSS的盒子模型.元素类型   本文首先介绍了CSS元素的统一内部结构模型,盒子模型:然后介绍了CSS元素基于不同分类标准定义的元素类型,包括基于不同内容设置方式定义的replaced元素和non-r ...

  7. jQueryMobile(二)

    三].按钮 <!-- 一个jQueryMobile页面 --> <div data-role='page'> <div data-role='header'>< ...

  8. Laravel 5.5 官方推荐 Nginx 配置学习

    Laravel 5.5 版本官方放出了 Nginx 服务器的配置,中文文档:服务器配置 Nginx server { listen 80; server_name example.com; root ...

  9. 新人学习微信小程序开发之框架篇

    大家好我是智哥,一名专注于前端领域的一名码农. 咱们今天主要来说说微信小程序, 最近一段时间微信群里的小程序,小游戏各种分享是突然一下子就爆发了,现在来看小程序作为微信的重磅功能无疑又是下一个风口.咱 ...

  10. 10大炫酷的HTML5文字动画特效欣赏

    文字是网页中最基本的元素,在CSS2.0时代,我们只能在网页上展示静态的文字,只能改变他的大小和颜色,显得枯燥无味.随着HTML5的发展,现在网页中的文字样式变得越来越丰富了,甚至出现了文字动画,HT ...