- 浏览: 84342 次
文章分类
- 全部博客 (136)
- 我的技术资料收集 (98)
- 具体技术 (1)
- 的技术资料收集 (4)
- All Articles (1)
- 机器学习 Machine Learning (1)
- 网络编程 (1)
- java (2)
- ava (1)
- 零散技术 (1)
- C# (3)
- 技术资料收集 (1)
- CQRS (1)
- 数据库技术(MS SQL) (1)
- .Net微观世界 (1)
- Oracle SQL学习之路 (1)
- C/C++ (1)
- JS/JQ (1)
- Js封装的插件/实例/方法 (2)
- 敏捷个人 (2)
- Javascript (1)
- 程序设计---设计模式 (1)
- Bug (1)
- 未知分类 (1)
- 程序设计 (1)
- Sharepoint (1)
- Computer Graphic (1)
- IT产品 (1)
- [06]JS/jQuery (1)
- [07]Web开发 (1)
- .NET Solution (1)
- Android (3)
- 机器学习 (1)
- 系统框架设计 (1)
- Others (1)
- 算法 (1)
- 基于Oracle Logminer数据同步 (1)
- 网页设计 (1)
- 原创翻译 (1)
- EXTJS (1)
- Jqgrid (1)
- 云计算 (1)
最新评论
我一直以来对于exception的态度都是很明确的。首先exception是好的,否则就不会有绝大多数的语言都支持他了。其次,error code也没什么问题,只是需要一个前提——你的语言得跟Haskell一样有monad和comonad。你看Haskell就没有exception,大家也写的很开心。为什么呢?因为只要把返回带error code结果的函数给做成一个monad/comonad,那么就可以用CPS变换把它变成exception了。所以说CPS作为跟goto同样基本的控制流语句真是当之无愧呀,只是CPS是type rich的,goto是type poor的。
其实很多人对于exception的恐惧心理在于你不知道一个函数会抛什么exception出来,然后程序一crash你就傻逼了。对于server来讲情况还好,出了问题只要杀掉快速重启就行了,如今没个replication和fault tolerance还有脸说你在写后端(所以不知道那些做web的人究竟在反对什么)?这主要的问题还是在于client。只要client上面的东西还没保存,那你一crash数据就完蛋了是不是——当然这只是你的想象啦,其实根本不是这样子的。
我们的程序抛了一个access violation出来,和抛了其它exception出来,究竟有什么区别呢?access violation是一个很奇妙的东西,一旦抛了出来就告诉你你的程序没救了,继续执行下去说不定还会有破坏作用。特别是对于C/C++/Delphi这类语言来说,你不小心把错误的东西写进了什么乱七八糟的指针里面去,那会儿什么事情都没发生,结果程序跑着跑着就错了。因为你那个算错了得到的野指针,说不定是隔壁的不知道什么object的成员变量,说不定是heap里面的数据结构,或者说别的什么东西,就这么给你写了。如果你写了别的object的成员变量那封装肯定就不管用了,这个类的不变量就给你破坏了。既然你的成员函数都是基于不变量来写的,那这个时候出错时必须的。如果你写到了heap的数据结构那就更加呵呵呵了,说不定下次一new就崩了,而且你还不知道为什么。
出了access violation以外的exception基本是没什么危害的,最严重的大概也就是网线被拔了,另一块不是装OS的硬盘突然坏了什么的这种反正你也没办法但是好歹还可以处理的事情。如果这些exception是你自己抛出来的那就更可靠了——那都是计划内的。只要程序未来不会进入access violation的状态,那证明你现在所能拿到的所有变量,还有指针指向的memory,基本上都还是靠谱的。出了你救不了的错误,至少你还可以吧数据安全的保存下来,然后让自己重启——就跟word一样。但是你有可能会说,拿出了access violation怎么就不能保存数据了呢?因为这个时候内存都毁了,指不定你保存数据的代码new点东西然后挂了,这基本上是没准的。
所以无论你喜欢exception还是喜欢error code,你所希望达到的效果本质上就是避免程序未来会进入access violation的状态。想做到这一点,方法也是很简单粗暴的——只要你在函数里面把运行前该对函数做的检查都查一遍就好了。这个无论你用exception还是用error code,写起来都是一样的。区别在于调用你的函数的那个人会怎么样。那么我来举个例子,譬如说你觉得STL的map实在是太傻比了,于是你自己写了一个,然后有了一个这样子的函数:
// exception版本
Symbol* SymbolMap::Lookup(const wstring& name);
// error code版本
int SymbolMap::Lookup(const wstring& name, Symbol*& result);
// 其实COM就是你们最喜欢的error code风格了,写起来应该很开心才对呀,你们的双重标准真严重
HRESULT ISymbolMap::Lookup(BSTR name, ISymbol** result);
于是拿到了Lookup函数之后,我们就要开始来完成一个任务了,譬如说拿两个key得到两个symbol然后组合出一个新的symbol。函数的错误处理逻辑是这样的,如果key失败了,因为业务的原因,我们要告诉函数外面说key不存在的。调用了一个ComposeSymbol的函数丢出什么IndexOutOfRangeException显然是不合理的。但是合并的那一步,因为业务都在同一个领域内,所以suppose里面的异常外面是可以接受的。如果出现了计划外的异常,那我们是处理不了的,只能丢给上面了,外面的代码对于不认识的异常只需要报告任务失败了就可以了。于是我们的函数就会这么写:
Symbol* ComposeSymbol(const wstring& a, const wstring& b, SymbolMap* map)
{
Symbol* sa=0;
Symbol* sb=0;
try
{
sa=map->Lookup(a);
sa=map->Lookup(b);
}
catch(const IndexOutOfRangeException& ex)
{
throw SymbolKeyException(ex.GetIndex());
}
return CreatePairSymbol(sa, sb);
}
看起来还挺不错。现在我们可以开始考虑error code的版本了。于是我们需要思考几个问题。首先第一个就是Lookup失败的时候要怎么报告?直接报告key的内容是不可能的,因为error code是个int。
题外话,error code当然可以是别的什么东西,如果需要返回丰富内容的错误的话,那怎样都得是一个指针了,这个时候你们就会面临下面的问题——这已经他妈不满足谁构造谁释放的原则了呀,而且我这个指针究竟直接返回出去外面理不理呢,如果只要有一个环节不理了,那内存岂不是泄露了?如果我要求把错误返回在参数里面的话,我每次调用函数都要创建出那么个结构来保存异常,不仅有if的复杂度,还有创建空间的复杂度,整个代码都变成了屎。所以还是老老实实用int吧……
那我们要如何把key的信息给编码在一个int里面呢?因为key要么是来自于a,要么是来自于b,所以其实我们就需要两个code了。那Lookup的其他错误怎么办呢?CreatePairSymbol的错误怎么办呢?万一Lookup除了ERROR_KEY_NOT_FOUND以外,或者是CreatePairSymbol的错误刚好跟a或者b的code重合了怎么办?对于这个问题,我只能说:
要不你们team的人先开会讨论一下最后记录在文档里面备查以免后面的人看了傻眼了……
好了,现在假设说会议取得了圆满成功,会议双方加深了互相的理解,促进了沟通,最后还写了一个白皮书出来,有效的落实了对a和b的code的指导,于是我们终于可以写出下面的代码了:
#define SUCCESS 0 // global error code for success
#define ERROR_COMPOSE_SYMBOL_WRONG_A 1
#define ERROR_COMPOSE_SYMBOL_WRONG_B 2
int ComposeSymbol(const wstring& a, const wstring& b, SymbolMap* map, Symbol*& result)
{
int code=SUCCESS;
Symbol* sa=0;
Symbol* sb=0;
switch(code=map->Lookup(a, sa))
{
case SUCCESS:
break;
case ERROR_SYMBOL_MAP_KEY_NOT_FOUND:
return ERROR_COMPOSE_SYMBOL_WRONG_A;
default:
return code;
}
switch(code=map->Lookup(b, sb))
{
case SUCCESS:
break;
case ERROR_SYMBOL_MAP_KEY_NOT_FOUND:
return ERROR_COMPOSE_SYMBOL_WRONG_B;
default:
return code;
}
return CreatePairSymbol(sa, sb, result);
}
啊,好像太长,干脆我还是不负责任一点吧,反正代码写的好也涨不了工资,干脆不认识的错误都返回ERROR_COMPOSE_SYMBOL_UNKNOWN_ERROR好了,于是就可以把代码变成下面这样……都到这份上了不要叫自己程序员了,叫程序狗吧……
#define SUCCESS 0 // global error code for success
#define ERROR_COMPOSE_SYMBOL_WRONG_A 1
#define ERROR_COMPOSE_SYMBOL_WRONG_B 2
#define ERROR_COMPOSE_SYMBOL_UNKNOWN_ERROR 3
int ComposeSymbol(const wstring& a, const wstring& b, SymbolMap* map, Symbol*& result)
{
Symbol* sa=0;
Symbol* sb=0;
if(map->Lookup(a, sa)!=SUCCESS)
return ERROR_COMPOSE_SYMBOL_UNKNOWN_ERROR;
if(map->Lookup(b, sb)!=SUCCESS)
return ERROR_COMPOSE_SYMBOL_UNKNOWN_ERROR;
if(CreatePairSymbol(sa, sb, result)!=SUCCESS)
return ERROR_COMPOSE_SYMBOL_UNKNOWN_ERROR;
return SUCCESS;
}
当然,如果大家都一样不负责任的话,还是exception完爆error code:
Symbol* ComposeSymbol(const wstring& a, const wstring& b, SymbolMap* map)
{
return CreatePairSymbol(map->Lookup(a), map->Lookup(b));
}
大部分人人只会用在当前条件下最容易写的方法来设计软件,而不是先设计出软件然后再看看怎样写比较容易,这就是为什么我说,只要你一个月给程序员还给不到一狗半,还是老老实实在政策上落实exception吧。至少exception写起来还不会让人那么心烦,可以把程序写得坚固一点。
好了,单线程下面至少你还可以争吵说究竟exception好还是error code好,但是到了异步程序里面就完全不一样了。现在的异步程序都很多,譬如说有良心的手机app啦,譬如说javascript啦,metro程序等等。一个try根本没办法跨线程使用所以一个这样子的函数(下面开始用C#,C++11的future/promise我用的还不熟):
class Normal
{
public string Do(string args);
}
最后就会变成这样:
class Async
{
// before .NET 4.0
IAsyncResult BeginDo(string args, Action<IAsyncResult> continuation);
string EndDo(IAsyncResult ar);
// after .NET 4.0
Task<string> DoAsync(string args);
}
当你使用BeginDo的时候,你可以在continuation里面调用EndDo,然后得到一个string,或者得到一个exception。但是因为EndDo的exception不是在BeginDo里面throw出来的,所以无论你EndDo返回string也好,返回Tuple<string, Exception>也好,对于BeginDo和EndDo的实现来说其实都一样,没有上文所说的exception和error code的区别。
不过.NET从BeginDo/EndDo到DoAsync经历了一个巨大的进步。虽然形式上都一样,但是由于C#并不像Haskell那样可以完美的操作函数,C#还是面向对象做得更好,于是如果我们吧Task<T>看成下面的样子,那其实两种写法是没有区别的:
class Task<T>
{
public IAsyncResult BeginRun(Action<IAsyncResult> continuation);
public T EndRun(IAsyncResult ar);
}
不过如果还是用BeginRun/EndRun这种方法来调用的话,使用起来还是很不方便,而且也很难把更多的Task组合在一起。所以最后.NET给出的Task是下面这个样子的(Comonad!):
class Task<T>
{
public Task<U> ContinueWith<U>(Func<Task<T>, U> continuation);
}
尽管真实的Task<T>要比上面那个复杂得多,但是总的来说其实就是围绕着基本简单的函数建立起来的一大堆helper function。到这里C#终于把CPS变换在异步处理上的应用的这一部分给抽象出来了。在看CPS的效果之前,我们先来看一个同步函数:
void button1_Clicked(object sender, EventArgs e)
{
// 假设我们有string Http.Download(string url);
try
{
string a = Http.Download(url1);
string b = Http.Download(url2);
textBox1.Text=a+b;
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
}
这段代码显然是一个GUI里面的代码。我们如果在一个GUI程序里面这么写,就会把程序写得跟QQ一样卡了。所以实际上这么做是不对的。不过为了表达程序需要做的所有事情,就有了这么一个同步的版本。那么我们尝试吧这个东西修改成异步的把!
void button2_Clicked(object sender, EventArgs e)
{
// 假设我们有Task<string> Http.DownloadAsync(string url);
// 需要MethodInvoker是因为,对textBox1.Text的修改只能在GUI线程里面做
Http.DownloadAsync(url1).ContinueWith(ta=>new MethodInvoker(()=>
{
try
{
// 这个时候ta已经运行完了,所以对ta.Result的取值不会造成GUI线程等待IO。
// 而且如果DownloadAsync内部出了错,异常会在这里抛出来。
string a=ta.Result;
Http.DownloadAsync(url2).ContinueWith(tb=>new MethodInvoker(()=>
{
try
{
string b=tb.Result;
textBox1.Text=a+b;
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
})));
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
})));
}
我们发现,异步操作发生的异常,把优越的exception拉低到了丑陋的error code的同一个情况上面——我们需要不断地对每一个操作重复同样的错误处理过程!而且在这种地方我们连“不负责任”的选项都没有了,如果你不try-catch(或者不检查error code),那到时候程序就会发生一些莫名其妙的问题,在GUI那一层你什么事情都不知道,整个程序就变成了傻逼。
现在可以开始解释一下什么是CPS变换了。CPS变换就是把所有g(f(x))都给改写成f(x, r=>g(r))的过程。通俗一点讲,CPS变换就是帮你把那个同步的button1_Click给改写成异步的button2_Click的这个过程。尽管这么说可能不太严谨,因为button1_Click跟button2_Click所做的事情是不一样的,一个会让GUI卡成qq,另一个不会。但是我们讨论CPS变换的时候,我们讨论的是对代码结构的变换,而不是别的什么东西。
现在就是激动人心的一步了。既然CPS可以把返回值变换成lambda表达式,那反过来我们也可以把所有的以这种形式存在的lambda表达式都改写成返回值嘛。现在我们滚回去看一看button2_Click,会发现这个程序其实充满了下面的pattern:
// lambda的参数名字故意起了跟前面的变量一样的名字(previousTask)因为其实他们就是同一个东西
previousTask.ContinueWith(previousTask=>new MethodInvoker(()=>
{
try
{
continuation(previousTask.Result);
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
})));
我们可以“发明”一个语法来代表这个过程。C#用的是await关键字,那我们也来用await关键字。假设说上面的代码永远等价于下面的这个代码:
try
{
var result=await previousTask;
continuation(result);
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
两段代码的关系就跟i++;和i=i+1;一样是可以互相替换的,只是不同的写法而已。那我们就可以用相同的方法来把button2_Click给替换成下面的button3_Click了:
void button3_Click(object sender, EventArgs e)
{
try
{
var a=await Http.DownloadAsync(url1);
try
{
var b=await Http.DownloadAsync(url2);
textBox1.Text=a+b;
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
}
聪明的读者立刻就想到了,两个try其实是重复的,那为什么不把他们合并成一个呢!当然我想告诉大家的是,异常是在不同的线程里面抛出来的,只是我们用CPS变换把代码“改写”成这种形式而已。理论上两个try是不能合并的。但是!我们的C#编译器君是很聪明的。正所谓语言的抽象高级了一点,那么编译器对你的代码也就理解得更多了一点。如果编译器发现你在try里面写了两个await,马上就明白了过来他需要帮你复制catch的部分——或者说他可以帮你自动的复制catch的部分,那情况就完全不同了,最后就可以写成:
// C#要求函数前面要加一个async来允许你在函数内使用await
// 当然同时你的函数也就返回Task而不是void了
// 不过没关系,C#的event也可以接受一个标记了async的函数,尽管返回值不一样
// 设计语言这种事情就是牵一发而动全身呀,加个await连event都要改
async void button4_Click(object sender, EventArgs e)
{
try
{
string a=await Http.DownloadAsync(url1);
string b=await Http.DownloadAsync(url2);
textBox1.Text=a+b;
}
catch(Exception ex)
{
textBox1.Text=ex.Message;
}
}
把两个await换成回调已经让我们写的够辛苦了,那么如果我们把await写在了循环里面,事情就不那么简单了。CPS需要把循环翻译成递归,那你就得把lambda表达时拿出来写成一个普通的函数——这样他就可以有名字了——然后才能递归(写出一个用于CPS的Y-combinator是一件很困难的事情,尽管并没有比Y-combinator本身困难多少)。这个例子就复杂到爆炸了,我在这里就不演示了。
总而言之,C#因为有了CPS变换(await),就可以把button4_Click帮你写成button3_Click然后再帮你写成button2_Click,最后把整个函数变成异步和回调的形式(真正的做法要更聪明一点,大家可以反编译去看)在异步回调的写法里面,exception和error code其实是一样的。但是CPS+exception和CPS+error code就跟单线程下面的exception和error code一样,有着重大的区别。这就是为什么文章一开始会说,我只会在带CPS变换的语言(Haskell/F#/etc)里面使用error code。
在这类语言里面利用相同的技巧,就可以不是异步的东西也用CPS包装起来,譬如说monadic parser combinator。至于你要选择monad还是comonad,基本上就是取决于你要自动提供错误处理还是要手动提供错误处理。像上面的Task.ContinueWith,是要求你手动提供错误处理的(因为你catch了之后可以干别的事情,Task无法自动替你选择最好的措施),所以他就把Task.ContinueWith写成了comonad的那个样子。
写到这里,不禁要同情写前端的那帮javascript和自以为可以写后端的node.js爱好者们,你们因为小小的eval的问题,不用老赵的windjs(windjs给javascript加上了await但是它不是一个altjs所以得显式调用eval),是一个多大的损失……
发表评论
-
C#WebBrowser控件使用教程与技巧收集--苏飞收集 - sufeinet
2013-06-28 12:07 1073原帖地址:http://www.cnblogs.com/suf ... -
我要喷一个自认为很垃圾的网站架构 - 老赵【苏州】
2013-06-28 12:01 1134原帖地址:http://www.cnblogs.com/lao ... -
[翻译] Oracle Database 12c 新特性Multitenant - Cheney Shue
2013-06-28 11:43 624原帖地址:http://www.cnblogs.com/ese ... -
memcahd 命令操作详解 - 阿正-WEB
2013-06-28 11:37 475原帖地址:http://www.cnblogs.com/azh ... -
面向过程的代码符合大众的思维方式吗? - 史蒂芬.王
2013-06-27 10:28 597原帖地址:http://www.cnblogs.com/ste ... -
面向过程的代码符合大众的思维方式吗? - 史蒂芬.王
2013-06-27 10:28 560原帖地址:http://www.cnblogs.com/ste ... -
RPG游戏之组队测试 - zthua
2013-06-27 10:22 560原帖地址:http://www.cnblogs.com/zth ... -
IT人们给个建议 - SOUTHER
2013-06-26 14:06 526原帖地址:http://www.cnblogs.com/sou ... -
Java向前引用容易出错的地方 - 银河使者
2013-06-26 14:00 497原帖地址:http://www.cnblogs.com/nok ... -
使用Func<T1, T2, TResult> 委托返回匿名对象 - 灰身
2013-06-26 13:54 801原帖地址:http://www.cnblo ... -
【web前端面试题整理03】来看一点CSS相关的吧 - 叶小钗
2013-06-25 10:45 788原帖地址:http://www.cnblogs.com/yex ... -
Windows 8 动手实验系列教程 实验6:设置和首选项 - zigzagPath
2013-06-25 10:27 624原帖地址:http://www.cnblogs.com/zig ... -
闲聊可穿戴设备 - shawn.xie
2013-06-25 10:21 568原帖地址:http://www.cnblo ... -
CentOS下Mysql安装教程 - 小学徒V
2013-06-23 15:24 612原帖地址:http://www.cnblogs.com/xia ... -
vmware安装ubuntu12.04嵌套安装xen server(实现嵌套虚拟化) - skyme
2013-06-23 15:18 840原帖地址:http://www.cnblogs.com/sky ... -
之前专门为IE6、7开发的网站如何迁移到IE10及可能遇到的问题和相应解决方案汇总 - 海之澜
2013-06-23 15:12 956原帖地址:http://www.cnblogs.com/wuz ... -
Android学习笔记--解析XML之SAX - 承香墨影
2013-06-23 15:01 413原帖地址:http://www.cnblo ... -
SQL Server 性能优化之——T-SQL TVF和标量函数
2013-06-19 09:32 676原帖地址:http://www.cnblogs.com/Boy ... -
Nginx学习笔记(二) Nginx--connection&request
2013-06-19 09:26 671原帖地址:http://www.cnblogs.com/cod ... -
从郭美美霸气侧漏看项目管理之项目经理防身术
2013-06-19 09:20 504原帖地址:http://www.cnblogs.com/had ...
相关推荐
《C++语言描述——数据结构算法与应用》是一本深入探讨C++编程语言在数据结构和算法应用方面的专业书籍。本书旨在帮助读者理解和掌握如何利用C++高效地实现各种数据结构和算法,从而提升编程技能和解决问题的能力。...
计算机程序设计语言的本质在于它是一种描述和实现计算机过程的规则化人工语言符号系统。从认识论的角度来看,高级语言不仅仅是一套指令集,更重要的是它们提供了一种思考问题和解决问题的方法论。算法设计则是这一...
"R——不仅仅是一门语言.pdf" 从文件的标题和描述中,我们可以看到R语言不仅仅是一门语言,它还具有强大的统计计算和机器学习能力。作者阿稳,一个来自学术界的程序员,擅长使用R语言进行数据挖掘和高性能计算应用...
《汇编语言程序设计——从DOS到Windows》是一门深入探讨计算机底层运作原理和编程技术的课程。汇编语言,作为与机器指令最接近的编程语言,是理解计算机硬件和软件交互的基础。本课程旨在带领学习者从DOS(Disk ...
模拟CMOS集成电路设计是电子工程中的一门重要课程,旨在介绍模拟集成电路的设计和分析原理。本文将对模拟CMOS集成电路设计的相关知识点进行总结和概述。 一、模拟集成电路设计的特点 模拟集成电路设计是一门复杂的...
面向对象的程序设计语言C++是一种强大的、灵活的编程工具,它在软件开发领域占据了重要的地位。C++是由Bjarne Stroustrup于1983年基于C语言发展而来的,旨在提供一种支持面向对象编程(OOP)概念的语言环境。 **...
《面向对象程序设计——Java语言》是一门深入探讨如何使用Java进行面向对象编程的课程,由知名讲师程细柱制作的PPT电子课件。面向对象编程(Object-Oriented Programming,简称OOP)是现代软件开发中的核心编程范式...
Java语言程序设计是一门核心的计算机科学课程,它教授学生如何使用Java编程语言来解决问题和构建软件系统。基于课程设计的考核方式,是教学方法的一种创新,旨在通过实践项目来评估学生的理解和技能,而非仅仅依赖...
《大班语言——蚕和蝉》就是这样一份旨在通过生动故事与互动式教学,使幼儿在语言学习中获得丰富体验的专业课件。 课件的一开始,首先介绍了“蚕”和“蝉”这两个汉字。汉字教学不仅仅是让孩子们认识两个字的笔画,...
简化代码、想法、概念和设计有助于避免复杂性,从而使代码更加简洁易维护。 2. **模块化与灵活性:** 保持代码模块化的同时增加灵活性,以适应不断变化的需求。 3. **一致性:** 保持语言规则的一致性,帮助学习者更...
在这个课程设计——“汽车转弯灯”项目中,学生将理论知识应用于实际问题解决,通过设计实现一个模拟汽车转弯灯控制系统,提升对微处理器、接口电路和编程的实践理解。 首先,设计的核心是微处理器,它是整个系统的...
对于有志于深入学习C++和数据结构算法的读者而言,《C++语言描述——数据结构算法与应用》无疑是一本不可多得的学习资料。无论你是计算机科学的初学者,还是有一定经验的软件开发人员,本书都能提供宝贵的知识和实践...
快速掌握一门语言 快速掌握一门语言 快速掌握一门语言
《单片机C语言程序设计实训100例——基于8051+Proteus仿真》是一本针对初学者和进阶者深入学习单片机编程的实用教材。本书的核心在于通过100个实际的C语言编程实例,帮助读者掌握8051系列单片机的使用技巧,同时结合...
数据库课程设计是计算机专业的一门重要课程,旨在培养学生的数据库设计和开发能力。本文主要介绍了图书管理系统的设计和实现,涵盖了需求分析、概念设计、逻辑设计、物理设计、数据库实施等方面。 一、 可行性分析 ...
面向对象程序设计是一门关键的计算机科学课程,它强调抽象思维和实践操作,对于培养学生的编程能力和解决问题能力至关重要。以河南工业大学为例,该校在改革面向对象程序设计课程时,遵循了以学生为中心的教学理念,...
总的来说,"数据结构与程序设计——C++语言描述"是一门深入探讨数据结构和算法的课程,结合源代码实践,能帮助你提升编程能力和问题解决能力,为后续的软件开发打下坚实基础。在学习过程中,务必动手实践,不断思考...
在汇编语言编程中,设计图形显示和动画是极具挑战性的任务,因为它涉及到与硬件的直接交互,尤其是视频显示系统。本复习资料主要探讨了如何使用汇编语言进行图形显示和控制物体运动方向的设计。 首先,汇编语言在...
而“机械毕业设计——法兰盘设计连续模设计.zip”这一文件,不仅是一份毕业设计作品,更是工程实践和技术创新的体现。 法兰盘连续模设计是实现高效、高精度法兰盘制造的关键技术。连续模,或称为级进模,是一种能够...
总的来说,城市规划与设计中的场地设计是一门综合性的艺术,需要兼顾功能、美学、生态和经济等多个维度。只有深入理解和重视这些容易被忽略的细节,才能创造出既实用又美观,且与环境和谐共生的城市空间。在实际操作...