谈lisp
The Lisp Curse /Lisp魔咒
http://winestockwebdesign.com/Essays/Lisp_Curse.html 英文出处
http://www.soimort.org/posts/124/ 中文翻译的出处 谷幽
The power of Lisp is its own worst enemy.
Here's a thought experiment to prove it: Take two programming languages, neither of which are object-oriented. Your mission, if you choose to accept it, is to make them object-oriented, keeping them backward-compatible with the original languages, modulo some edge cases. Inserting any pair of programming languages into this thought experiment will show that this is easier with some languages than with others. That's the point of the thought experiment. Here's a trivial example: Intercal and Pascal.
Now make this thought experiment interesting: Imagine adding object orientation to the C and Scheme programming languages. Making Scheme object-oriented is a sophomore homework assignment. On the other hand, adding object orientation to C requires the programming chops of Bjarne Stroustrup.
The consequences of this divergence in needed talent and effort cause The Lisp Curse:
Lisp is so powerful that problems which are technical issues in other programming languages are social issues in Lisp.
Consider the case of Scheme, again. Since making Scheme object-oriented is so easy, many Scheme hackers have done so. More to the point, many individual Scheme hackers have done so. In the 1990s, this led to a veritable warehouse inventory list of object-oriented packages for the language. The Paradox of Choice, alone, guaranteed that none of them would become standard. Now that some Scheme implementations have their own object orientation facilities, it's not so bad. Nevertheless, the fact that many of these packages were the work of lone individuals led to problems which Olin Shivers wrote about in documenting the Scheme Shell, scsh.
Programs written by individual hackers tend to follow the scratch-an-itch model. These programs will solve the problem that the hacker, himself, is having without necessarily handling related parts of the problem which would make the program more useful to others. Furthermore, the program is sure to work on that lone hacker's own setup, but may not be portable to other Scheme implementations or to the same Scheme implementation on other platforms. Documentation may be lacking. Being essentially a project done in the hacker's copious free time, the program is liable to suffer should real-life responsibilities intrude on the hacker. As Olin Shivers noted, this means that these one-man-band projects tend to solve eighty-percent of the problem.
Dr. Mark Tarver's essay, The Bipolar Lisp Programmer, has an apt description of this phenomenon. He writes of these lone-wolf Lisp hackers and their
...inability to finish things off properly. The phrase 'throw-away design' is absolutely made for the BBM and it comes from the Lisp community. Lisp allows you to just chuck things off so easily, and it is easy to take this for granted. I saw this 10 years ago when looking for a GUI to my Lisp. No problem, there were 9 different offerings. The trouble was that none of the 9 were properly documented and none were bug free. Basically each person had implemented his own solution and it worked for him so that was fine. This is a BBM attitude; it works for me and I understand it. It is also the product of not needing or wanting anybody else's help to do something.
Once again, consider the C programming language in that thought experiment. Due to the difficulty of making C object oriented, only two serious attempts at the problem have made any traction: C++ and Objective-C. Objective-C is most popular on the Macintosh, while C++ rules everywhere else. That means that, for a given platform, the question of which object-oriented extension of C to use has already been answered definitively. That means that the object-orientated facilities for those languages have been documented, that integrated development environments are aware of them, that code libraries are compatible with them, and so forth.
Dr. Mark Tarver's essay on bipolar Lispers makes the point:
Now in contrast, the C/C++ approach is quite different. It's so damn hard to do anything with tweezers and glue that anything significant you do will be a real achievement. You want to document it. Also you're liable to need help in any C project of significant size; so you're liable to be social and work with others. You need to, just to get somewhere.
And all that, from the point of view of an employer, is attractive. Ten people who communicate, document things properly and work together are preferable to one BBM hacking Lisp who can only be replaced by another BBM (if you can find one) in the not unlikely event that he will, at some time, go down without being rebootable.
Therefore, those who already know C don't ask "What object system should I learn?" Instead, they use C++ or Objective-C depending on what their colleagues are using, then move on to "How do I use object-oriented feature X?" Answer: "Goog it and ye shall find."
Real Hackers, of course, have long known that object-oriented programming is not the panacea that its partisans have claimed. Real Hackers have moved on to more advanced concepts such as immutable data structures, type inferencing, lazy evaluation, monads, arrows, pattern matching, constraint-based programming, and so forth. Real Hackers have also known, for a while, that C and C++ are not appropriate for most programs that don't need to do arbitrary bit-fiddling. Nevertheless, the Lisp Curse still holds.
Some smug Lisp-lovers have surveyed the current crop of academic languages (Haskell, Ocaml, et cetera) and found them wanting, saying that any feature of theirs is either already present in Lisp or can be easily implemented — and improved upon — with Lisp macros. They're probably right.
Pity the Lisp hackers.
Dr. Mark Tarver — twice-quoted, above — wrote a dialect of Lisp called Qi. It is less than ten thousand lines of macros running atop Clisp. It implements most of the unique features of Haskell and OCaml. In some respects, Qi surpasses them. For instance, Qi's type inferencing engine is Turing complete. In a world where teams of talented academics were needed to write Haskell, one man, Dr. Tarver wrote Qi all by his lonesome.
Read that paragraph, again, and extrapolate.
Exercise for the reader: Imagine that a strong rivalry develops between Haskell and Common Lisp. What happens next?
Answer: The Lisp Curse kicks in. Every second or third serious Lisp hacker will roll his own implementation of lazy evaluation, functional purity, arrows, pattern matching, type inferencing, and the rest. Most of these projects will be lone-wolf operations. Thus, they will have eighty percent of the features that most people need (a different eighty percent in each case). They will be poorly documented. They will not be portable across Lisp systems. Some will show great promise before being abandoned while the project maintainer goes off to pay his bills. Several will beat Haskell along this or that dimension (again, a different one in each case), but their acceptance will be hampered by flame wars on the comp.lang.lisp Usenet group.
Endgame: A random old-time Lisp hacker's collection of macros will add up to an undocumented, unportable, bug-ridden implementation of 80% of Haskell because Lisp is more powerful than Haskell.
The moral of this story is that secondary and tertiary effects matter. Technology not only affects what we can do with respect to technological issues, it also affects our social behavior. This social behavior can loop back and affect the original technological issues under consideration.
Lisp is a painfully eloquent exemplar of this lesson. Lisp is so powerful, that it encourages individual independence to the point of bloody-mindedness. This independence has produced stunningly good innovation as in the Lisp Machine days. This same independence also hampers efforts to revive the "Lisp all the way down" systems of old; no "Lisp OS" project has gathered critical mass since the demise of Symbolics and LMI.
One result of these secondary and tertiary effects is that, even if Lisp is the most expressive language ever, such that it is theoretically impossible to make a more expressive language, Lispers will still have things to learn from other programming languages. The Smalltalk guys taught everyone — including Lisp hackers — a thing or two about object oriented programming. The Clean programming language and the Mozart/Oz combo may have a few surprises of their own.
The Lisp Curse does not contradict the maxim of Stanislav Datskovskiy: Employers much prefer that workers be fungible, rather than maximally productive. Too true. With great difficulty does anyone plumb the venality of the managerial class. However, the last lines of his essay are problematic. To wit:
As for the “free software” world, it eagerly opposes industrial dogmas in rhetoric but not at all in practice. No concept shunned by cube farm hells has ever gained real traction among the amateur masses.
In a footnote, he offers Linux as an example of this unwillingness to pursue different ideas. To be sure, he has a point when it comes to operating systems (the topmost comment, in particular, is infuriatingly obtuse). He does not have a point when it comes to programming languages. Python and Ruby were influenced by Lisp. Many of their fans express respect for Lisp and some of their interest has augmented the Lisp renaissance. With some justice, JavaScript has been described as "Scheme in C's clothing" despite originating in those cube farm hells.
Nevertheless, in spite of this influence, in both the corporate and open source worlds, Lisp still has only a fraction of the developer mind share which the current crop of advanced scripting languages have attracted. The closed-mindedness of MBA's cannot be the only explanation for this. The Lisp Curse has more explanatory power.
The free development environments available for Lisp further exemplify the Lisp Curse.
It's embarrassing to point this out, but it must be done. Forget about the Lisp Machine; we don't even have development systems that match what the average Smalltalk hacker takes for granted ("I've always felt Lisp is the superior language and Smalltalk is the superior environment." - Ramon Leon). Unless they pay thousands of dollars, Lisp hackers are still stuck with Emacs.
James Gosling, the author of the first Emacs that ran on Unix, has correctly pointed out that Emacs has not fundamentally changed in more than twenty years. This is because the Emacs maintainers are still layering code atop a design which was settled back when Emacs was a grad-student project at the MIT AI Lab, i.e., when Emacs development was still being indirectly financed by the national debt. A Slashdotter may object that Emacs is already quite capable and can do anything that any other development environment can do, only better. Those who have used Lisp Machines say otherwise.
So why don't the Lisp hackers put the Smalltalk guys in their proper place? Why don't they make a free development system that calls to mind some of the lost glories of the LispM, even if they can't reproduce another LispM?
The reason why this doesn't happen is because of the Lisp Curse. Large numbers of Lisp hackers would have to cooperate with each other. Look more closely: Large numbers of the kind of people who become Lisp hackers would have to cooperate with each other. And they would have to cooperate with each other on a design which was not already a given from the beginning. And there wouldn't be any external discipline, such as a venture capitalist or other corporate master, to keep them on track.
Every project has friction between members, disagreements, conflicts over style and philosophy. These social problems are counter-acted by the fact that no large project can be accomplished otherwise. "We must all hang together, or we will all hang separately." But the expressiveness of Lisp makes this countervailing force much weaker; one can always start one's own project. Thus, individual hackers decide that the trouble isn't worth it. So they either quit the project, or don't join the project to begin with. This is the Lisp Curse.
One could even hack Emacs to get something that's good enough. Thus, the Lisp Curse is the ally of Worse is Better.
The expressive power of Lisp has drawbacks. There is no such thing as a free lunch.
这篇文章的标题叫“The Lisp Curse”。
我给它加了一个副标题,叫“对Lisp的非技术性吐槽”。
毫无疑问,这是一篇非技术性质的文章,但是它也许比很多技术文章能更好地解释一些疑问。列举几个无聊的的命题:“为什么世界上最好的编程语言没有得到它应有的地位”、“为什么自底向上支撑着我们个人计算机乃至整个网络的Unix / BSD / GNU / GTK+ / Qt / Linux / Apache / MySQL...不是用Lisp/Scheme写的”、“为什么王垠批完Google批学术界却没做多少牛逼哄哄的项目”或者“为什么说‘孤狼黑客’对开源软件的生态环境是有害的”……诸如此类。
顺便说一句,我最初是在@yukihiro_matz的推上看到这篇文章的分享的。所谓的“孤狼黑客”,并不是指Matz和Van Rossum这些人——至少从Python和Ruby发布到社区的一刻起就不再是了。
再补充一句,我相信不管是这篇文章的原作者还是翻译君本人,都没有任何对Lisp社区不敬的意思——只是就一项事实陈述和复述某种观点而已。
好了,废话少说,以下是正文内容。
这篇文章是另一次尝试,旨在解释Lisp语言在具备强大力量的同时、为何Lisp社区却无法重现它们在“AI之冬”之前的辉煌。毫无疑问,即便在式微之后,Lisp语言仍然是各种奇思妙想的重要来源。这一事实,加之各种Lisp实现的优异架构,还有如今在长达十年之后Lisp的复兴、显示着那些Lisp拥护者们是多么需要为自己的得意之作找到一点优越感;尽管有这一切,他们却没能成功地把Lisp的力量转换成一场压倒性的技术革新。
在本文中,我将阐述这样一个论点:Lisp那极其强大的表达能力,实际上是如何成为它缺乏前进动力的致命诱因的。
Lisp的力量也是它自身最危险的天敌。
要证明这件事情,试想一个简单的思维实验:选择两种非面向对象的程序语言。你的任务,如果你愿意接受的话,就是为它们提供面向对象编程范式的支持,并且保持它们与原语言向后兼容——排除一些边界情况以外。把任意两种语言放到这个思维实验的设定当中,你会很容易发现一些语言较另一些语言更容易完成任务。这正是这个思维实验的要点。随手举个简单的例子:INTERCAL和Pascal。
现在让我们把这个思维实验变得更有说服力些。想象一下给C和Scheme添加面向对象的支持。让Scheme支持面向对象不过是个稍微费点脑筋的家庭作业。另一方面,让C支持面向对象,你恐怕得有Bjarne Stroustrup的本事才能办到。
这种对于解决问题所需才能和努力程度上的分歧,造成了Lisp的魔咒:
Lisp是如此强大,以至于在其他编程语言中的技术问题,到了Lisp中变成了社会问题。
想一想Scheme的情形吧。因为让Scheme支持面向对象是如此轻而易举,许多Scheme黑客都完成过这件事情,更准确地说,是许多独立的Scheme黑客都完成过。这就导致了在20世纪90年代,这个语言的面向对象支持包像工厂量产出来的库存清单一样数不胜数。想想选择谬论就知道,没有哪一个包能够成为正式的标准。如今某些Scheme实现已经有了它们自己的面向对象功能,这还不算太坏。尽管如此,那些五花八门的由不同独立开发者开发出来的包所造成的问题,正如Olin Shivers在给Scheme Shell(scsh)写文档的时候所提到的一样。
独立黑客们写出来的程序基本上遵循“抓痒模型”。这些程序解决了写程序的黑客们自己关心的问题,但是却未必能很好地处理对于其他人来说有用的部分功能。况且,虽说这些程序无疑可以在这个黑客自己的环境配置上运行得很好,但却不一定能移植到其他的Scheme实现、甚至不同平台上的同一Scheme实现上。文档可能会极度匮乏。从现实的角度讲,一个黑客用自己挤出来的空闲时间做出来的项目,当然也可能会因为黑客自己的现实生活问题而烂尾。正如Olin Shivers指出的那样,这些个人性质的项目更倾向于解决整个问题的百分之八十。
Mark Tarver博士的文章,The Bipolar Lisp Programmer(双面Lisp程序员),对这种现象有一个确切的表述。他写道,这些“孤狼”式的Lisp黑客以及他们:
……不能把事情恰当地做完收尾。所谓的“用过就扔设计”绝对和拉屎这件事儿没什么两样(注:原文如此),而它来源于Lisp社区。Lisp允许你如此虎头蛇尾地了结一件事,而且还让你心安理得地这么去做。十年前,我有一次需要给我的Lisp程序写一个GUI。没问题,我找到了9个GUI库。麻烦在于这9个库没有一个拥有足够完整的文档,而且它们全部是bug充斥的。基本上,每个人都实现了他们自己的一套解决方案,对于他们来说能用就行。这是一种拉屎式的态度(注:原文如此);反正我做到了,我消化它了。这同样也是无须在他人帮助下即可得到的产物。
那么再想一想C语言在上述思维实验中的情形吧。由于在C上面实现面向对象支持的困难程度,只有两个严肃的解决方案摆上了台面:C++,以及Objective-C。Objective-C在Mac上最为流行,而C++几乎统治了其他一切平台。这意味着,给定一个平台,如何让C支持面向对象的扩展几乎已经被唯一确定了。这意味着这些语言的面向对象支持将拥有完善的文档,高度集成的IDE,和兼容性良好的库,等等。
Mark Tarver博士的文章说到这一点:
现在与之相反,C/C++的做事方式完全不同。用镊子和胶水一步步搭建成一个东西实在太他妈困难了,所以你所做的一切都是实实在在的成就。你想要为它好好地写些文档。你会在做一个规模可观的C语言项目时候需要他人的帮助;因此你也更加容易变得社会性、学会去与他人合作。你需要做到这些,因为你需要完成某件事情。
而全部的这些,从一个雇主的角度来讲,是非常吸引人的。十个能够相互交流、写文档与团队协作的人显然会比一个像拉翔一样自己去hack些Lisp代码的人更有用,而这种翔的替代品只能是另一坨翔,这些翔们随时都可能因为某些个人的问题、自己退出项目而丢下一个不可收拾的烂摊子。
所以说,那些懂C的人不会去纠结“我应该用哪种面向对象系统?”相反,他们会去选择C++或者Objective-C,就像他们的同事所选择的一样,然后他们会提问“我该怎样使用面向对象的功能X?”答案很简单:“咕狗一下,你就知道。”
真正的黑客,当然早就知道面向对象并非如它的拥趸们所宣称的那样是解决一切问题的灵丹妙药。真正的黑客已经在探寻更加高阶的概念,诸如不可变数据结构、类型推断、惰性求值、monad、arrow、模式匹配、约束编程,等等。真正的黑客也都知道,C/C++对于写大部分不需要进行任意位操作的程序来说并不合适。尽管如此,Lisp的魔咒仍然存在。
一些沾沾自喜的Lisp发烧友,调研了当前学术界编程语言的硕果(Haskell、OCaml等等)后,发现它们所缺失的一些特性,要么就是已经在Lisp中存在、要么就是可以用Lisp很轻易地实现——并且可以改进——基于Lisp宏。他们也许是对的。
但是太可惜了,Lisp黑客们。
Mark Tarver博士——在上面已经两次引用过——曾经设计过一个Lisp的方言,叫做Qi。它仅仅由少于一万行的宏组成,基于Clisp运行。它实现了绝大部分Haskell和OCaml所独有的特性。从某个方面来说,Qi甚至超越了它们。举个例子说吧,Qi的类型推断引擎本身是图灵完全的。在这样的一个由天才科学家组成的杰出团队才能开发出Haskell语言的世界中,Tarver博士,他完全是一个人做出来了Qi。
再看一眼上面这段话。仔细想想,是不是可怕极了。
给读者的习题:假想Haskell与Common Lisp之间发生了激烈的对抗,下一步将会发生什么?
答案:Lisp魔咒应验了。每两个或者三个Lisp黑客就会开发出一套属于自己的惰性求值、纯函数式、arrow、模式匹配、类型推断等等的实现。大部分这种项目都是孤狼式开发。因而,它们具备大部人所需要的80%功能(虽然这80%的定义会随情况不同而各异)。它们的文档通常会很差。它们无法在不同的Lisp系统上移植。有些项目可能起初信誓旦旦地会维护下去,直到开发者决定自己跑到别处赚钱去了,结果丢下一个无法收拾的烂摊子。有一些可能会在某种程度上多多少少地打败Haskell,但是它的认可度会被comp.lang.lisp新闻组里面的口水战淹没。
故事的结局:任意一个传统的Lisp黑客的宏能够拼凑成一个文档匮乏的、不可移植的、bug充斥的80%的Haskell实现,仅仅因为Lisp是一种比Haskell表达力更加强大的语言。
这个故事的教育意义在于,次级效应和三级效应至关重要。技术不只是影响我们解决技术问题的方式,它也影响着我们的社会行为。而社会行为会反馈并施加影响于我们最初试图解决的技术问题。
Lisp是一个活生生的事例。Lisp是如此强大有力,它鼓励个人的、狂热的特立独行。在Lisp机器曾经盛极一时的年代,这种特立独行产生了举世瞩目的成果。但也正是同样的特立独行,阻碍了所谓“自底而上纯Lisp实现”的计算机系统的复苏;自Symbolics和LMI夭折之后,再也没有一个“Lisp操作系统”项目达到过值得令人关注的高度。
这些次级和三级效应的一个后果是,即使Lisp是有史以来最富于表达力的编程语言,以至于理论上几乎不可能创造出比它更具表达力的语言,Lisper们将仍然有许多从其他编程语言那里学习的东西。Smalltalk程序员们教会了每个人——包括那些Lisp黑客们——多多少少一点关于面向对象的概念。Clean语言和Mozart/Oz也有着一些自己的奇特之处。
Lisp魔咒并不违背Stanislav Datskovskiy的至理名言:雇主们更喜欢可以被取代的雇员,而不是生产率最高的雇员。说得太实在了。你早该醒悟到那些管理学课程只是骗钱的把戏。然而,他这篇文章的最后几行似乎存在问题。请看:
在“自由软件”的世界里,工业界教条仅仅是在口头上被激烈地批判,但却从未在实践中被反对过。那些“办公隔间地狱”里被唯恐避之不及的概念,同样也未曾在业余爱好者中间得到过青睐。
在脚注中,他将Linux作为一个拒绝追求新奇想法的实例。为了例证,他将操作系统作为自己的一个论点(下面评论的1L是SB)。他并没有提到编程语言。Python和Ruby都受到了Lisp的影响。很多它们的饭表示了对Lisp的尊重,而他们的一些兴趣则促进了Lisp的复兴。公正地讲,JavaScript也曾被描述为“披着C外衣的Scheme”,尽管它诞生在那些办公隔间地狱中间。
即便有如此大的影响力,在工业和开源界里,Lisp也仅仅只吸引了一部分程序员的一部分注意力,而这也是拜最近脚本语言的兴起所赐。那些拿着MBA学位的高富帅码农们的思维封闭并不是唯一的原因。“Lisp魔咒”本身能解释更多的事情。
提供给Lisp的、可用的自由开发环境可以作为“Lisp魔咒”的一个例证。
尽管说出来让人难堪,但必须得有人去做这件事情。忘掉Lisp机器吧;我们甚至还没有一个能达到算得上Smalltalk黑客小康标准的开发环境(“我总赶脚到Lisp是一个牛逼的语言,而Smalltalk是一个牛逼的环境。”——Ramon Leon如是说)。除非他们愿意付上千刀美元,否则Lisp黑客们仍然会受制于Emacs。(翻译君:比肾还贵的LispWorks,包邮哦亲,欲购从速哦)
James Gosling,第一个在Unix上运行的Emacs的作者,恰当地指出了Emacs已经长达二十年没有任何基础上的变动。这是因为,Emacs维护者们只是不断地在这个当年由一个MIT的AI实验室的研究生做的设计上垒代码,那时Emacs项目仍然间接地从国债那里获得资助。也许Slashdot喷子会反驳说Emacs已经无所不能,它可以完成其它任何开发环境所能做的事情,而且只会完成得更好。不然那些当年曾经用过Lisp机器的也要这么说。
那么,为什么Lisp黑客们没有把那些Smalltalk家伙们给彻底打败呢?为什么他们没有做出一个自由的开发环境,可以从某种程度上唤回Lisp机器曾经的辉煌,即使他们不能够重现出另一个Lisp机器?
这件事没有发生的原因来自于Lisp魔咒。大量的Lisp黑客应该去协作。说得更详细些:大量成为Lisp黑客的人们应该去协作。而且他们应该学会合作去做一个新的设计、而非遵从一个从一开始就写死了的现有设计。这过程中不应该有任何来自外界的压力,例如风险资本家或者企业雇主,来干涉他们做事的方式。
每个项目都会存在参与者的分歧,诸如意见不合、风格或哲学上的冲突。如果这些社会性的问题持续下去,任何大的项目都无法完成,这就产生了一个让它们倾向于解决的反作用力。“要么我们团结一心,要么我们都会被吊死在同一棵树上”。而Lisp的强大表达能力削弱了这个反作用力;一个人总可以着手去自搞一套。这样,刚愎自用的黑客们认为不值得去应付观点分歧带来的麻烦;因此他们要么退出了合作项目,要么就干脆不参加已有的项目、而选择自力更生从头开始。这就是Lisp魔咒。
我们甚至可以自己去hack Emacs,为了追求个人理念中所谓的“足够好”。于是乎,Lisp诅咒差不多就变成了“坏即是好(Worse is Better)”的同义词。
谈lisp的更多相关文章
- C,C++,Lisp,Java,Perl,Python
(译注:圣经记载:在远古的时候,人类都使用一种语言,全世界的人决定一起造一座通天的塔,就是巴别塔,后来被上帝知道了,上帝就让人们使用不同的语言,这个塔就没能造起来. 巴别塔不建自毁,与其说上帝的分化将 ...
- 奇怪的Lisp和难懂的计算机程序的构造和解释
最近用新买的 Kindle 看<黑客与画家>的Lisp部分,发现作者 Paul Graham 很推崇 Lisp 语言,并且认为其它语言都没有Lisp简洁“成熟”,并且举例证明其它语言都在往 ...
- 从并发处理谈PHP进程间通信(二)System V IPC
.container { margin-right: auto; margin-left: auto; padding-left: 15px; padding-right: 15px } .conta ...
- 【转】Lisp 已死,Lisp 万岁!
Lisp 已死,Lisp 万岁! 有一句古话,叫做“国王已死,国王万岁!”它的意思是,老国王已经死去,国王的儿子现在继位.这句话的幽默,就在于这两个“国王”其实指的不是同一个人,而你咋一看还以为它自相 ...
- STL之父Stepanov谈泛型编程的发展史
这是一篇Dr. Dobb's Journal对STL之父stepanov的采访.文中数次提到STL的基本思想.语言的特性.编程的一些根本问题等,非常精彩.这篇文章让我想去拜读下stepanov的大作& ...
- 【转】.NET(C#):浅谈程序集清单资源和RESX资源 关于单元测试的思考--Asp.Net Core单元测试最佳实践 封装自己的dapper lambda扩展-设计篇 编写自己的dapper lambda扩展-使用篇 正确理解CAP定理 Quartz.NET的使用(附源码) 整理自己的.net工具库 GC的前世与今生 Visual Studio Package 插件开发之自动生
[转].NET(C#):浅谈程序集清单资源和RESX资源 目录 程序集清单资源 RESX资源文件 使用ResourceReader和ResourceSet解析二进制资源文件 使用ResourceM ...
- Lisp的本质(The Nature of Lisp)
Lisp的本质(The Nature of Lisp) 作者 Slava Akhmechet 译者 Alec Jang 出处: http://w ...
- python进阶_浅谈面向对象进阶
python进阶_浅谈面向对象进阶 学了面向对象三大特性继承,多态,封装.今天我们看看面向对象的一些进阶内容,反射和一些类的内置函数. 一.isinstance和issubclass class F ...
- 异数OS谈发展国产操作系统的问题
异数OS谈发展国产操作系统的问题 为什么写本文 最近中兴被美制裁的问题以及红芯使用开源技术宣称国产自主技术引发了舆论不少对国产CPU以及国产操作系统自主技术的讨论,为什么我们国家有BAT,有原子弹,能 ...
随机推荐
- nopcommerce 4.1 core 学习 增加商城配置属性
需求: 原本是想用nop 来做国际版的商城,可以像亚马逊那样 国内外通用, 专门增加一个跨进元素属性. 学习里面的一些架构思想. 国内的行情还是 像himall 会比较实用. 这是在商城的综合 ...
- spring 定时任务执行2次
eclipse 上定时任务执行没有问题,生产环境可以看到定时任务同时执行了2次,排除代码原因,网上找了些资料,最后发现是tomcat的原因, Host 节点中有一个appBase 属性指向了webap ...
- 怎么单独为ionic2应用的某一组件设置两个平台一致的样式
今天在继续项目的过程中,发现ionic2在显示样式上是根据不同的平台采用不同的样式,使在不同平台上的应用保持相应的风格,于是问题来了. ios的风格比较好看,android的风格略微不如ios的,所以 ...
- 纯css实现评分
用到的知识点: E:checked:单选或复选框被选中 E ~ F: 选择后面的兄弟节点们:选择后面的兄弟节点 E::after,E::before: 伪元素选择器 在匹配E的元素后面(前面)插入内容 ...
- 给大家介绍一个实用的RN神器DeviceEventEmitter
再不出来更新一下自己都感觉不到自己还存在了,这个监听最常用的地方莫过于单选和全选了,,当然远不止这个了,大家可以自己去多尝试几波,举个栗子 A组件全选所在 //全选 choose(bool){ //选 ...
- 基于Openstack环境下开启SRIOV
主题思想: 先在系统层面修改配置文件,再去openstack里面修改配置文件 compute node系统层面: 先确认下是否含有ixgbe moudle lsmod |grep ixgbe 修改/e ...
- TP5框架 nginx服务器 配置域名 隐藏index.php
server { listen ; #server_name localhost; server_name hhy.com;/**这里写自己的域名*/ #charset koi8-r; #access ...
- 【5】用vector进行直接插入排序
百分百自己编的程序,越来越觉得编程很好玩了. 但这算是第一次自己用vector这种不是那么无脑的方法编程,只能最多对3个数进行排序wwwww 今天我要回去搬宿舍了,等明天有时间,我一定要把bug找到! ...
- Request对象和Response对象 JsonResponse对象 和 Django shortcut functions 和QueryDict对象
request request属性 属性: django将请求报文中的请求行.头部信息.内容主体封装成 HttpRequest 类中的属性. 除了特殊说明的之外,其他均为只读的. ''' 0.Http ...
- 以太坊上发行ERC20代币
ERC20 代币生成 环境 虚拟主机: ubuntu 18虚拟机 宿主主机: win10; ip:192.168.0.160 1.部署以太坊 1.1 安装GO 安装go,并编译geth 将下载好的go ...