随笔 - 2, 文章 - 0, 评论 - 0, 引用 - 0
数据加载中……

2006年4月14日

borland传奇故事 -- 英雄[Anders]落难

英雄落难

Philippe Kahn和Anders共同创造了传奇的Borland,两人之间有着浓厚的感情。在
Borland工作时,对于Anders任何的想法和计划,Philippe Kahn都是不遗余力地支持。
也正是这个重要的支持力量,才有随后极为成功的Borland Pascal以及Delphi的问世。

但是在Philippe Kahn离开Borland之后,Anders再也没有了这股来自最亲密战友的强
力支援。1997年,Borland新的CEO Delbert Yocam在掌握了大权之后,Borland整个
公司开始走向第二个重要的转变,Delbert对于Borland产品的开发和趋势也有了不同
于Philippe Kahn的看法。当Java在1996年逐渐快速发展之后,睿智的Anders也看到
了Java成功的未来。因此在Anders不再积极参与Delphi 2/3的开发工作之后,他非常
希望能够主导Borland Java开发工具的开发,期望能够像当初的Delphi 1.0一样,为
Borland再次开发出全世界一级的Java开发工具。

不过,由于当时Delphi是Borland最重要的收入来源,高层仍然希望Anders继续在
Delphi产品线上投入全力,因此当时的Borland CEO Delbert Yocam并没有批准Anders
的请求。Borland的下一个重要的开发工具JBuilder,当时的产品开发名称为Latte,
仍然交由其他小组负责。依据我的推想,由于当时Anders对于Java已经有许多的想法,
因此才会有后来的VJ++以及C#,这些产品和程序语言的许多特性想必已经在Anders的脑
中存在了一段时间了。

Delbert没有允许Anders带领Latte开发小组,但Anders仍然没有放弃他的新计划。也
许是Anders注定和Borland的缘分已经到了尽头,这个时候正好Microsoft展开了有史
以来对Borland最大的挖角行动。在Anders无法在Borland取得满意的支持之后,
Microsoft提供的优厚条件顿时对Anders产生了致命的吸引力,从而造成了Borland无
法挽回的遗憾。

虽然Anders没有显赫的学历,无法获得Turning Awards(即图灵奖,信息科学界最高
荣誉的奖项,等同于诺贝尔奖)。但是我认为Anders的实力和贡献绝不输于任何一位
Turning Awards的得奖人。Anders是最好的信息实践型人物,在2001年,他终于获得
了信息界最具权威的信息刊物Dr. Dobbs' Journal颁发的Excellent Programming
Awards,以表彰Anders为信息界做出的卓越贡献。我想Anders应该是许多本身没有高
学历或不是计算机信息科系出身的优秀程序员最好的效仿对象。

Anders Hejlsberg这位不世出的软件天才,是目前全世界最顶尖的软件技术人员之一。
论实现技术,Anders可能是目前的第一高手,因为他精通程序语言、编译器技术、开发
工具、Framework以及系统架构。我虽然知道许多软件界重要的人物和好手,但是尚
不知有任何人能像Anders一样在这么多领域都能成为大家。下面是笔者整理出
Anders Hejlsberg到目前为止重要的功绩、贡献以及获颁的重要大奖:

" 和Philippe Kahn共同创办Borland

" 开发出Turbo Pascal,当时首创的In-Memory Compiler震惊了全世界

" 开发出全世界最畅销的Pascal产品,Turbo Pascal(这是许多信息人员学习Pascal
和Data Structure使用的经典产品)以及Borland Pascal。Turbo/Borland Pascal合
计销售超过了数百万套。Dr. N. Wirth(Pascal语言的创始人员)也应该向Anders致敬,
表达Anders对于Pascal语言的贡献

" Anders使用汇编语言撰写编译器,其功力无人能出其右。创造出了全世界速度最快、
品质也是一流的Pascal编译器。在Anders离开了Borland之后,几乎没有人能够修改
Anders的编译器

" 开发出影响深远的Delphi这个伟大的RAD工具

" 开发出VJ++语言

" Microsoft .NET的Architect

" Microsoft颁授Microsoft Distinguish Engineer大奖

" 发明C#这个又将造成重大影响的语言

" 获颁2001年Dr. Dobbs' Journal的Excellence In Programming大奖

一个人一生能够做出几件让全世界都津津乐道的事业呢?Anders却成就了无数PC界伟
大的功绩,并且在程序语言、编译器、开发工具以及Framework方面都有重要的贡献。
PC软件界因为有了Anders而精彩、丰富了许多,也创造了许多令人惊叹的故事。更
棒的是Anders现在仍然在继续贡献他惊人的天分,就让我们拭目以待,看看Anders还
能创造什么功迹吧。不过,不管以后如何,相信Anders应该是大部分软件人员希望学
习的目标。Anders的功力也是大部分软件人员一生企望能够达到的境界。

在2002年Borland Developers' Conference中,Anders Hejlsberg是排名第一的
Keynote Speaker,尚在Java的创始人James Gosling之前。根据现场同时聆听这两场
Keynote Speech的听众报道,Anders的Keynote Speech是非常精彩的,而James的
Keynote Speech则相对的枯燥,许多人因此而提前离席。而且Anders在开始进行
Keynote Speech之时,便获得了现场所有听众起立鼓掌致敬,看来,在大多数Borland
开发工具使用者的心中,Anders Hejlsberg是永远的巨星。

Microsoft的挖角和Anders的离开

Anders在不介入Delphi的开发、并且无法主导Borland Java开发工具开发的情况下充
满了挫折感。没有了Philippe Kahn的强力支援,Anders虽然是Borland最顶尖的技术
人才,却也无法对抗Borland管理阶层的力量。当然这也是从Philippe Kahn离开了
Borland之后、Bodand开始转型有关,这在稍后Borland的转变一文中,我会作详细的
说明。

虽然Anders在Borland遇到了挫折,但是对于Microsoft来说这却是千载难逢的好机会,
在此时Microsoft展开了大规模的挖角行动,而且是明目张胆地进行,正是由于
Microsoft如此大胆的行动,因此也造成了不久之后Borland对于Microsoft的法律控诉。

这次的挖角行动中,Microsoft同时锁定了数个Borland最杰出或是重要的人物,当然
Anders是名列第一的挖角对象。时值1996年,Microsoft终于展开了行动,使用的方
式是最直接的攻击。Microsoft直接派遣加长型的大轿车到Borland大门口找Anders吃
饭,第一次Microsoft开出了年薪百万美元以上的条件。不过在Borland知道了这件事
情之后,也很快进行了加码的动作,因此Anders并没有对Microsoft进行响应。
Microsoft在苦等无应、按捺不住之下,很快就再次用大轿车找Anders。这次Microsoft
提出了两百万美元以上的条件,希望Anders能够首肯。对于这次的喊价,Borland可有
点为难了,因为两百万美元不是笔小数目,这已经比当时Borland许多副总裁的年薪还
高。此外,如果Borland答应也加到两百万以上,那么是不是Chuck也要如此加码?其
他的Delphi R&D小组要如何调整?这些都是非常棘手的问题。

不过Borland很快找到了解决的方案,那就是允许Anders从每一套卖出的Delphi版本
中抽取一定数量的版权费。如此一来Delphi卖得愈好,Anders便能取得愈多的回馈。
不过就我的了解,Anders注重的并不是金钱上的问题,因为在Borland创立的初期,
由于Turbo Pascal的编译器都是Anders撰写的,因此当时Anders也是卖一套Turbo
Pascal就可以抽取版税的。依照Turbo/Borland Pascal全世界销售数百万套来算,
Anders早就是富翁了。薪水多一点,少一点并无所谓,Anders心中想的是自由发展的
空间。在Borland提出了Delphi的随版抽税,再加上Microsoft并不知道Anders真正想
要的东西,因此Anders仍然没有响应Microsoft提出的优厚条件。

不过,Anders实在是太重要的人物,而且Microsoft在面对Java与日俱增的威胁下,
非常渴望能够有像Anders这样的人才带领Microsoft开发下一代的开发工具,这当然
也是由于Microsoft以前向Borland挖来的人都做出了不小的贡献所致。Microsoft食
髓知味,当然希望能够得到Borland的镇山之宝。在Anders两次不为所动之后,
Microsoft决定祭出最后的王牌,由Bill Gates亲自找Anders吃饭,进行最终的挖角
行动。

不管读者喜不喜欢Bill Gates,不可否认的是Bill也是一个天才。自古英雄惜英雄,
在Anders和Bill相谈甚欢的情形下,Microsoft开出了年薪三百万以上、数万股的
Microsoft股票这个超高的条件,再以当时Microsoft高贵的股价来计算,真是一笔庞
大的数字,也许对于搞软件技术的人来说,这已经是不可能的天文数字了。不过这些
条件并不是打动Anders的主要原因,Bill最后提出的条件是"答应给Anders一个小组
和极多的资源,让Anders尽情地发挥"。这个条件可说是打到了Anders的心底,因为
Anders正渴望有人能够支持他完成新的计划和想法。我想,在软件产业中大概也只有
Microsoft能够拥有这种雄厚的资源可以挖角任何人吧。

在Bill Gates提了这样的条件之后,Borland再也没有本钱能够和Microsoft进行比价
了,只好眼睁睁地看着Anders离开Borland前往Microsoft再开创下一个人生的高峰。
在Anders到了Microsoft之后,Bill Gates果然重用Anders,也立刻让Anders负责激
活Microsoft的下一拨开发工具计划,当然这个计划也是Microsoft对抗SUN/Java的整
体计划之一。Anders到了Microsoft之后立刻展现了实力,让Microsoft的编译器技术
又精进不少,最明显的例子就是Microsoft后期的Java Virtual Machine是PC上执行
效率最好的,而且在2、3年后,VJ++编译出来的虚拟机械码的执行效率不但比任何的
Java开发工具还快,在某些方面甚至比原生的Windows开发工具,例如Delphi、VB、
甚至是VC++还有效率。这真是令人震撼,当然Anders为VJ++打下的基础现在也展现在
NET的C#编译器以及.NET的JIT(Just In Time)编译器之上,.NET的JIT在许多程序代
码最佳化方面比Delphi还先进。因此在2、3年前当VJ++即将推出之际,在Borland内
部也引起了非常大的骚动,并且严阵以待,当然这又是另外一个故事了。

对于Anders来说,到了Microsoft之后不久又再次登上了生涯的另一个巅峰。因为当
初Anders在Borland之时,就有如孙子兵法中叙述的"藏于九地之下",虽然是不世出
的天才,但是仅为少数的人所知,即使是使用Borland产品的人在当时可能也不知道
Anders这号人物。因为Anders和Borland的作风很像,都是行事低调,不到最后绝不
随意出手。但是Anders被挖角到Microsoft之后,由于Microsoft的企业文化向来是前
进、积极的侵略性方式,因此Anders也就转变为"动于九天之上",负责Microsoft开
发工具大军的核心大将,不但广为人知,成为许多软件人员效法的对象,而且屡获大
奖。他不但获得了信息软件业界的推崇,最后也终于获得了信息学术界的认可,可说
是实至名归。

Anders的离开对于Borland来说是一个很大的损失,不过对于Delphi R&D小组来说却
是有好有坏,因为Delphi开发小组虽然失去了最重要的支柱,但是也给Danny Thorpe
一个快速崛起的机会,在1年后,Danny果然立刻成为了Delphi/C++Builder/Kylix中
最杰出的人物之一(另外一个当然就是Anders的老搭挡Chuck Jazdzewski了),Danny
也是我认为目前在Borland RAD(注)部门中功力最厉害的人物。在稍后的内文中我会
对Danny进行比较详细的说明。

注:Borland RAD部门是指Borland的Rapid Application Development部门,RAD负责
的产品包含了C++Builder、Delphi、Kylix以及未来的.NET以及Mobile的新产品。

转至 <http://www.gameres.com >

posted @ 2006-04-14 16:00 风中的雷鸟 阅读(246) | 评论 (0)编辑 收藏

用C/C++库实现的一个简单的词法分析器

/* gg.h */
/* another file to define macros */
#ifndef GG_H
#define GG_H
#include <string>
#include <vector>
#include <fstream>
#include <iostream>
using namespace std;

/* function declaration */
string readFile(char* filename);       // deal with files in C++ methds
void startprint();
void endprint();
vector<string> findwords(const char* str); // calculate words number
void statistics(vector<string> strstr);
void numscore();
// void lastmodified();            // complete the last invoked for main function

#define LENGTH(x) (sizeof(x) / sizeof(*(x)))    // the length of array...

typedef struct code{
    char num[20];
    int nnm;
}NUMCODE;

#endif

#include "gg.h"
#include <cstdio>
#include <fstream>
#include <string>
#include <vector>
#include <iostream>
using namespace std;

/* define globa variable */
NUMCODE array[]={
    { "sy_if",     0 },
    { "sy_then",     1},
    { "sy_else",     2},
    { "sy_while",     3},
    { "sy_begin",     4},
    { "sy_do",      5},
    { "sy_end",    6},
    { "a",        7},
    { "semicolon",     8},
    { "e",        9},
    { "jinghao",    10},
    { "S",        11},
    { "L",        12},
    { "tempsy",    13},
    { "EA",        18},
    { "EO",        19},
    { "#",    30},
    { "/",  31},
    { "*",  32},
    { "-",  33},
    { "+",    34},
    { "times",     36},
    { "becomes",     38},
    { "Op_and",    39},
    { "Op_or",    40},
    { "Op_not",    41},
    { "rop",    42},
    { "(",    48},
    { ")",    49},
    { "ident",    56},
    { "intconst",    57}
};
int linecount =0;
int element = 0;

void startprint(){
    puts("--------------START---------------");
}

void endprint(){
    puts("--------------END--------------");
}

/* a function read a file and puts its words to a container
 * using C++ standard library */
string readFile(char *filename){
    vector<string> words;
    string word;
    string strtext;

    ifstream in(filename);
   
    while(getline(in, word)){
        strtext += word + '\n';
        ++linecount;
    }
       
    cout << strtext << endl;
    return strtext;
}
 
/* searching for words and other print them */
vector<string> findwords(const char *str){
    char *pf = const_cast<char*>(str);    // type cast
    char *ptemp;
    int i = 0;
    int j = 0;        // calculate
    bool isword = false;
    bool redundance = false;
    vector<string> strs;
    vector<char> ch;

    if( !isspace(*pf) )
        isword = true;
   
    while(*pf){
        if(isword){
            ptemp = pf;
            ++i;        // the number of words
        }
       
        if( isalpha(*pf) || isdigit(*pf) || *pf == '_' \
        && *pf != '(' && *pf != ')' && *pf != ';' &&\
        *pf != '+' && *pf != '-' && *pf != '*' && *pf != '/'){
            isword = false;
            j++;
        }

        if( isspace(*pf) || *pf == '(' || *pf ==')' || \
            *pf == '[' || *pf == ']' || *pf == '+' || *pf == '-'\
            || *pf == '*' || *pf == '/' || *pf == '#'){    // || ispunct(*pf)){
           
            // adds special characters
            if(*pf == '(' || *pf == ')' || *pf == '#'\
            || *pf == '+' || *pf == '-' || *pf == '*'\
            || *pf == '/' || *pf == ':'){
                string tempstr;
                tempstr = *pf;
                strs.push_back(tempstr);
                redundance = true;
            }
            else
                redundance = false;
            if(redundance)
                goto xxx;

            ptemp[j] = '\0';
            strs.push_back(ptemp);

            isword = true;
            j = 0;
            ptemp = pf;
        }
        xxx:    pf++;
    }
    return strs;
}

/* statistics the words which is in words table */
void statistics(vector<string> strstr){
    int i;
    int j;
    bool isin = false;
    cout << "\n  the words can be recognized as follows" << endl;
    cout << "==========================================\n";
   
    for(i = 0; i < strstr.size(); i++){   
        for(j = 0; j < LENGTH(array); j++){
            if( 0 == strcmp(strstr[i].data(), array[j].num)){
                cout << "(\""<<strstr[i]<<"\"," <<array[j].nnm << " )" << endl;
                ++element;
                continue;
            }
            else if (j == LENGTH(array)-1 && 0 != strcmp(strstr[i].data(), "\0") &&\
                0 != strcmp(strstr[i].data(), "+") && 0 != strcmp(strstr[i].data(), "-")&&\
                0 != strcmp(strstr[i].data(), "*") && 0 != strcmp(strstr[i].data(), "/")&&\
                0 != strcmp(strstr[i].data(), "(") && 0 != strcmp(strstr[i].data(), ")")){
                cout << "(\"" << strstr[i] << "\", " << 56 << ")"<< endl;
                ++element;
            }
        }
    }
    return ;
}

void numscore(){
    cout << "----------------- numbers of elements ------------- "<< endl;
    cout << "\tlines\t\telements" <<endl;
    cout << "\t" << linecount << "\t\t" << element << endl;
}

//=====================================================
/* gg.cpp
 * compiler: g++, c++
 * abet standard C/C++ library
 */
#include "gg.h"

int main(char argc, char **args){
    const char *filearray;
    vector<string> dealstr;
    string strname;

    startprint();
    if(argc == 1){
        printf("error: please input file name.\n");
        printf("dirpath: %s\n", args[argc-1]);
       
        while(true){
            cout << "Please input file name: ";
            cin >> strname;
            filearray = readFile((char*)strname.data()).data();
            dealstr = findwords(filearray);
            statistics(dealstr);            numscore();
            break;
        }
       
        endprint();
        system("PAUSE");
        exit(0);
    }else if(argc == 2){
        printf("filename: %s\n", args[argc-1]);
        puts("---text content----");
        filearray = readFile(args[argc-1]).data();
       
        endprint();
        dealstr = findwords(filearray);
        statistics(dealstr);            numscore();
    }
    system("PAUSE");
}
// 里面的算法感觉都非常的笨,可是又想不出来好的,好的东西从哪里来...

posted @ 2006-04-14 15:52 风中的雷鸟 阅读(1105) | 评论 (0)编辑 收藏