Processing math: 100%

2011年12月28日星期三

最大的子序列和问题


最大的子序列和问题描述:
给定整数A1,A2,…,AN,求SUM(Ak),k从i到j,的最大值。为了方便起见,如果所有整数均为
负数,则最大子序列和为0。

例:
输入-2,11,-4,13,-5,-2时,答案为20(从A2到A4)

最大子序列和问题的解。
算法一:
穷举,复杂度为O(N^3)
int maxsubsequencesum(const int A[],int N){
    int thissum,maxsum,i,j,k;
   
    maxsum=0;
    for(i=0;i<N;i++)
        for(j=i;j<N;j++){
                thissum=0;
                for(k=i;k<=j;k++)
                        thissum += A[k];
               
                if(thissum > maxsum)
                        maxsum = thissum;
        }
    return maxsum;
}
算法二:
对算法一进行优化,减少内循环,复杂度为O(N^2)
int maxsubsequencesum_2(const int A[],int N){
    int thissum,maxsum,i,j;
   
    maxsum=0;
    for(i=0;i<N;i++){
        thissum=0;
        for(j=i;j<N;j++){
             thissum += A[j];
                    
             if(thissum > maxsum)
                   maxsum = thissum;
        }
    }
    return maxsum;
}
算法三:
分治:最大子序列可能在三处出现,或者整个出现在数据的左半部,或者整个出现在右半部,
或者跨越输入数据的中部而占据左右两半部分。前两种情况可以递归求解,第三种情况的最
大和可以通过求出前半部分的最大和(包含前半部分的最后一个元素)以及后半部分的最大
和(包含后半部分的第一个元素),将这两个和相加而得到。
复杂度为O(NlgN)
int maxsubsum(const int A[], int left,int right){
    int maxleftsum,maxrightsum;
    int maxleftbordersum,maxrightbordersum;/*包含边界的最大值*/
    int leftbordersum,rightbordersum;
    int center,i;

    if(left == right){
         if(A[left] > 0)
             return A[left];
         else
             return 0; /*由题给条件,负数的最大子序列长度为0*/
    }

    /* divide */
    center = (left + right)/2;
    maxleftsum = maxsubsum(A,left,center);
    maxrightsum = maxsubsum(A,center+1,right);

   /*conquer*/
   maxleftbordersum=0;
   leftbordersum=0;
   for(i=center; i>=left;i--){
       leftbordersum += A[i];
       if(leftbordersum > maxleftbordersum)
          maxleftbordersum = leftbordersum;
   }

  maxrightbordersum=0;
  rightbordersum=0;
  for(i=center+1; i <= right; i++){
      rightbordersum += A[i];
      if(rightbordersum > maxrightbordersum)
          maxrightbordersum = rightbordersum;
  }

  /* max3 取三者最大值 */
  return max3(maxleftsum, maxrightsum, maxleftbordersum + maxrightbordersum);
}

int maxsubsequencesum_3(const int A[], int N){
    return maxsubsum(A,0,N-1);
}
算法四:
在线算法,复杂度O(N),
int maxsubsequencesum(const int A[],int N){
    int thissum,maxsum,j;
    thissum = maxsum = 0;

    for(j=0; j < N; j++){
        thissum += A[j];
       
        if(thissum > maxsum)
            maxsum = thissum;
        else if(thissum < 0)
            thissum = 0;
    }
    return maxsum;
}

2011年12月14日星期三

Blogger使用SyntaxHighlighter插件及测试

Blogger中,也就是这个博客,使用SyntaxHighlighter来实现语法高亮,需要在模板文件中加入以下语句:

<head>
     <b:include data='blog' name='all-head-content'/>
     <!-- adel zhang 20111214 -->
     <!-- syntaxhighlighter -->
     <script src="http://sites.google.com/site/triadelphous/filestorage/shCore.txt" type="text/javascript">
     <script src='http://sites.google.com/site/triadelphous/filestorage/shBrushCpp.txt' type='text/javascript'/>
     <script src='http://sites.google.com/site/triadelphous/filestorage/shBrushJScript.txt' type='text/javascript'/>
     <script src='http://sites.google.com/site/triadelphous/filestorage/shBrushXml.txt' type='text/javascript'/>
     <script src='http://sites.google.com/site/triadelphous/filestorage/shBrushJava.txt' type='text/javascript'/>
     <script src='http://sites.google.com/site/triadelphous/filestorage/shBrushBash.txt' type='text/javascript'/>
     <script src='http://sites.google.com/site/triadelphous/filestorage/shBrushCss.txt' type='text/javascript'/>
     <link href='http://sites.google.com/site/triadelphous/filestorage/shThemeEmacs.css' rel='stylesheet' type='text/css'/> 
     <script type='text/javascript'>
     SyntaxHighlighter.defaults["quick-code"]=false;
     SyntaxHighlighter.config.bloggerMode = true;
     SyntaxHighlighter.defaults["toolbar"] = false;
     SyntaxHighlighter.all();
     </script>

因为Blogger无法存储外部文件,所以将需要的js文件托管在Google Sites上。至于文件名为什么要改成txt,这是由于有帖子
But Google Sites, like most free host doesn’t accept javascripts. Luckily there is a workaround -change the javascript file into a text file, by changing the file extension from .js to .txt.


Update:20111217
本来将下载的SyntaxHighlighter托管在Google Sites上,但不知为何链接不到了(上回测试时是可以的)。所以重新修改了模板,使用的是SytaxHighlighter作者提供的pub服务。
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shCore.js' type='text/javascript'/>
    <script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushCpp.js' type='text/javascript'/>
    <script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushJScript.js' type='text/javascript'/>
    <script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushXml.js' type='text/javascript'/>
    <script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushJava.js' type='text/javascript'/>
    <script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushBash.js' type='text/javascript'/>
    <script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushCss.js' type='text/javascript'/>
    <link href='http://alexgorbatchev.com/pub/sh/current/styles/shThemeEmacs.css' rel='stylesheet' type='text/css'/>

Update: 20120107
遇到的问题是如果代码行太长,则显示时会换行,这样就造成syntaxhighlighter的行号显示出现混乱,极为丑陋。后来发现添加shCore.css这个css后,它会出现滚动条,缓解了行号混乱的问题。但不爽的地方在于会出现竖直滚动条,原因在于shCore.css中对overflow属性使用了visible。所以我们在模板文件中加入下面这些css语句(这些类名是shCore.css里面定义的),以消除竖直方向上的滚动条。特别注意 !important 的用法,它使得css不是按照通常的定义顺序来使用,而是使用有 !important 的css语句。

/* adel: 20120107
------------- for syntaxhighlighter -----------
------------- get rid of y-sroll --------------*/
.syntaxhighlighter a,
.syntaxhighlighter div,
.syntaxhighlighter code,
.syntaxhighlighter table,
.syntaxhighlighter table td,
.syntaxhighlighter table tr,
.syntaxhighlighter table tbody,
.syntaxhighlighter table thead,
.syntaxhighlighter table caption,
.syntaxhighlighter textarea {
   overflow-y: hidden !important;
}

之前的设置去掉了 toolbar,也就是提示syntaxhighlighter信息的小问号,想想还是加上为好。所以最终的代码为:
<!-- adel zhang 20120107 -->
<!-- syntaxhighlighter -->
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shCore.js' type='text/javascript'/>
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushCpp.js' type='text/javascript'/>
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushJScript.js' type='text/javascript'/>
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushXml.js' type='text/javascript'/>
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushJava.js' type='text/javascript'/>
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushBash.js' type='text/javascript'/>
<script src='http://alexgorbatchev.com/pub/sh/current/scripts/shBrushCss.js' type='text/javascript'/>
<link href='http://alexgorbatchev.com/pub/sh/current/styles/shCore.css' rel='stylesheet' type='text/css'/>
<link href='http://alexgorbatchev.com/pub/sh/current/styles/shThemeEmacs.css' rel='stylesheet' type='text/css'/>
<script type='text/javascript'>SyntaxHighlighter.config.bloggerMode = true;SyntaxHighlighter.all();</script>

2011年12月10日星期六

IPV6下GoAgent的使用

先说FAIL的部分。
今天启动系统的时候,在正常的登陆界面下输入用户名和密码后,就出现花屏,进不去系统。当时第一反应是:完蛋鸟,又得折腾一番,而且很大可能是搞不定然后重装系统,毕竟这不是X就是驱动出问题了,我完全没概念的。
更要命的是我刚把网络通给关了(见下文),也就是说不能求助GOOGLE大神。还好在Grub启动项里有个叫Resume的选项,或许能启动系统。在输入用户名和密码后,果不其然能启动到文字界面,还出现了可疑的出错信息sogouproxy.py。我才想起昨天修改过~/.profile文件,在其中添加执行sogouproxy.py,用于启动搜狗代理服务器。罪魁祸首就是系统启动时当执行用户配置文件时,sogouproxy,py就卡住了,造成X不能正常启动。只能说,学术不精,FAIL!!!

再说说IPV6下GoAgent的使用。
以前介绍过GoAgent,主要是备不时之需(你懂的)。最近校园网络改造,原本寝室可以上教育网,现在只能上校内网络(鸡肋),开通10元的网络通才只能有个教育网出口的国内权限(鸡肋),非得20元的网络通才能有国际权限(抢钱阿)。但神奇的是,即使只有校内权限,但IPV6是不受限制的,所以修改过IPV6 host的google系网站还是可以蹭的。
GoAgent是部署在GAE上的,所以很自然地想是否能通过IPV6连接GoAgent,以此作为IPV4的代理。庆幸的是,GoAgent默认支持IPV6的,只要简单地修改配置文件即可:
在proxy.ini 的 [appspot] 节中有
hosts = cn
修改为:
hosts = ipv6
在 [google]节中,默认使用cn的hosts,注释掉了hk和ipv6的hosts。为了使用ipv6,将cn的hosts注释掉,并去掉ipv6的注释。
重启Goagent,就可以通过ipv6免费上网拉,速度很OK。

2011年12月7日星期三

POJ 1009

复制至此有两个原因:
1. 这题我看完毫无头绪,佩服会解该题的人。
2. 下面的代码写的很干净

原文地址:这里


只需考虑原图中数值变化的点,其他点的编码与其左侧的点相同。
最开始用brute force,优化了输出编码中有连续0的情况,sample input可以较快通过,但提交结果超时。又换成上面的方法,可以AC,但代码改得很乱,待重写。


/*
 *  只需考虑原图中数值发生变化的点
 */
#include <stdio.h>
#include <math.h>

typedef struct _Ref {
    int index;
    int code;
} Ref;

int IN[1000][2];
Ref OUT[8000];

int encode(int n, int width, int total);
int getcode(int n);

void sort(Ref *list, int n);

int
main()
{
    int width,i,j,m,n,t,k,cnt,total,current,row,col;
    Ref cur;
   
    while(scanf("%d", &width) != EOF) {
        if(width == 0)
            break;
       
        // input  
        i = total = 0;
        scanf("%d%d", &IN[i][0], &IN[i][1]);
        while(IN[i][0] != 0 || IN[i][1] != 0) {
            total += IN[i][1];
            i++;
            scanf("%d%d", &IN[i][0], &IN[i][1]);  
        }
       
        printf("%d\n", width);
       
        cnt = i;
        n = 1;
        k = 0;
        for(m=0; m<=cnt; m++) {
            row = (n - 1) / width;
            col = (n - 1) % width;
           
            for(i=row-1; i<=row+1; i++) {
                for(j=col-1; j<=col+1; j++) {
                    t = i * width + j;
                    if(i<0 || j<0 || j>width-1 || t>total-1)
                        continue;
                   
                    OUT[k].index = t + 1;
                    OUT[k++].code = encode(t+1, width, total);                    
                }
            }
            n += IN[m][1];
        }
       
        sort(OUT, k);
       
        cur = OUT[0];
        for(i=0; i<k; i++) {
            if(OUT[i].code == cur.code)
                continue;
               
            printf("%d %d\n", cur.code, OUT[i].index - cur.index);
            cur = OUT[i];
        }
       
        printf("%d %d\n", cur.code, total - cur.index + 1);
        printf("0 0\n");
    }
    printf("0\n");
   
    return 0;
}

// 返回第n个元素的编码,n从1开始
int
getcode(int n)
{
    int t,i;
   
    i = t = 0;
    while(t < n) {
        t += IN[i++][1];
    }
   
    return IN[i-1][0];
}

// 对第n个元素进行编码,n从1开始
int
encode(int n, int width, int total)
{
    int i,j,t,code,result,row,col;
   
    code = getcode(n);
   
    row = (n - 1) / width;
    col = (n - 1) % width;
   
    result = 0;
    for(i=row-1; i<=row+1; i++) {
        for(j=col-1; j<=col+1; j++) {
            t = i * width + j;
            if(i<0 || j<0 || j>width-1 || t == n-1 || t>total-1)
                continue;
               
            if(abs(getcode(t+1) - code) > result)
                result = abs(getcode(t+1) - code);
        }
    }
   
    return result;
}

void sort(Ref *list, int n)
{
    Ref tmp;
    int i,j;
   
    for(i=0; i<n-1; i++) {
        for(j=i+1; j<n; j++) {
            if(list[i].index > list[j].index) {
                tmp = list[j];
                list[j] = list[i];
                list[i] = tmp;
            }
        }
    }
}

2011年11月21日星期一

技术人创业建站简略指南


作者: Fenng | 可以转载, 但必须以超链接形式标明文章原始出处和作者信息及版权声明
网址: 

你是一个技术人员,你想创建一个站点,或许是一时心血来潮,或许是为了自己的兴趣爱好,或许是...「创业」前的热身准备?那么,如何少走弯路的构建起来你的 Web 站点呢?别笑,不是每个做技术的都捣鼓过个人站点,不是每个人都独立搭建过网站(我不是说个人Blog这样简单的东西),下面的简略指南或许能帮到你。毫无疑问,以下都是广告。
0. 信用卡
这个真要有!
确保有一张具备境外支付功能信用卡。这一点我想不是难事,就算是在校的学生,申请信用卡的门槛也小了很多。现在各个银行发信用卡都是求着用户的,申请的时候问清是否具备外币支付能力就好了。有了信用卡之后,你可以较为方便的申请 Paypal 、App Store 帐户、Google Checkout...
有了信用卡之后,你可以大大方方的收美元了 -- 如果你是面向非中国用户的话。
1. 购买域名
不要在国内的域名提供商那里购买域名。为了一时方便以后你会付出更多的代价,你可以看一下这里的血泪史。购买域名,我建议在 Godaddy 上购买就可以,顺便说一下,GoDaddy 也支持支付宝。如何买到好的域名? 我没办法告诉你(如果你要从别人手里购买域名的话,可以看一下 4.cn).
域名备案怎么办?如果你的内容不是特别敏感的话,不备案可能问题也不大,低调一点,别自己往墙上撞。
备选: Domain
2. 主机服务
有了域名之后,那么购买主机就要提上日程。建议用 Linode 的 VPS 服务,价格不算贵,关键是容易上手,相对比较稳定,Linode 在东京的 IDC 访问速度居然...和国内某些 IDC 差不多。有些做技术的朋友,可能自己手头有个主机什么的,尽量不要托管在 IDC 了,机器硬件坏了或是被拔了网线,会让你很闹心。
如果你的Web应用已经写的差不多了,购买主机之后不妨进行部署,在线测试。如果应用正式上线,那么不妨买一个备份服务,每个月5美元而已。
有了境外的 VPS 的一个好处是,你可以通过 VPS 「翻墙」,锻炼一下腿脚。怎么做,搜索一下就知道了。如果要简单的优化一下 VPS ,参考这篇
备选: Slicehost
3. 域名解析
为什么要单独提 DNS 解析?GoDaddy 和 Linode 都提供 DNS 解析能力,不过,域名在哪里注册的和域名在哪里解析是两回事。重要的是,DNS 修改之后的有效验证是个不小的问题,还有一个是影响因素 DNS 解析速度,所以,有必要启用智能 DNS 解析服务,DNSPod 做的相当不错。用了之后你就知道,而且,没有副作用 :)
4. 静态文件
服务器在境外,经常遇到的一个性能瓶颈静态文件(尤其是图片)的访问速度上不来,而恰好你的应用要存储较多的静态文件的话,不妨研究一下 UpYun 的服务。如果你是个开发者,你会体会到一定的妙处,去看看又拍云的 API,你会喜欢的。重要的是,价格也可控制。
阅读: 又拍云实战
备选: CloudFlare(如果你的服务是面向国外用户的话)
5. 运维监控
即使是最简单的站点也有必要关注可访问性,监控机器运行状态。推荐监控宝的免费服务,足以满足小型个人站点对于监控的要求。Google Analytics 和 Google Webmasters 有必要启用。百度的统计服务最近一段时间也越做越好。
6. 邮件方案
如果是做邮件托管的话,也就是你的站点本身的邮件帐户解决方案,Google Apps 是不二之选。如果需要发邮件给你站点的注册用户,或者做小规模的 DM , 在 Linode 上启用 EXIM 就差不多了。

N. 接下来呢?
下一步该做什么?或许有必要成为 Github 的付费用户,开发、部署、上线、推广...等你到了一定规模,咱们再来第二季。
恭喜你走上不归路,也祝愿你得到一些因为折腾而带来的乐趣.
--EOF--
(发现还是太简略了,欢迎大家留言补充)

2011年11月8日星期二

如何心算是星期几


这个心算算法的技巧在于利用了这么一个结论:对于任意给定的一年,某些特定的日期总是属于相同的星期几。我们称这些日期为“锚点“。方便记忆的锚点有:5月9号,9月5号,7月11号,11月7号,4月4号,6月6号,8月8号,10月10号,12月12号,以及2月的最后一天(平年时为28号,闰年时为29号)。前面四个日期可以通过"在7-11朝9晚5地工作"这句顺口溜来记忆。

锚点在给定的年份属于星期几是不固定的,但变化是有规律可循的。比如今年2011年,锚点都是星期一,而在2010年,锚点都是星期日。每过一年,锚点星期就往前移一天。这很好理解,每年有365天,365 % 7 = 1,所以锚点星期会移一天。因为闰年有366天,所以当年是闰年的话,跟闰年的上一年比较,锚点是向前移了两天的。比如2012年是闰年,所以锚点都是星期三

这样只要记住一些锚点的星期,就可以很快心算出任意日期的星期了。


参考资料

2011年11月6日星期日

正则表达式中的字符组[ ]


正则表达式中的字符组(Character Classes)用"[…]"表示,它容许使用者列出在某处期望匹配的字符。比如我们需要搜索单词"grey",同时又不确定它是否写成了"gray",就可以使用"gr[ea]y"进行匹配。在字符组内部,字符组元字符"-"(连字符)表示一个范围,"[0-9]"和"[a-z]"是常用的匹配数字和小写字母的简便方式。连字符"-"在字符组内部才是元字符,否则它就只能匹配普通的连字符号;即使在字符组内部,它也不一定是元字符,如果连字符出现在字符组的开头,它表示的就只是一个普通字符,而不是一个范围。问号"?"和点号"."在字符组中也是普通字符。

以上在大多数正则表达式的书中都会提到的,但这里忽略了如何在字符组内部使用"方括号"本身。这就是我遇到的问题,被困扰了好几个小时。问题来自一条sed语句:

sed -ne '/^ID_.*=/ {s/[]()|&;<>`'"'"'\\!$" []/\\&/g;p}'

当在字符组中期望匹配方括号时,"[&[]"会匹配"["和"&","[]&]"会匹配"]"和"&",而同时匹配"&","["和"]"必须写成"[]&[]",这就是上面那个例子的情况,注意最外层的方括号才是代表字符组。与之相对应,让人很困惑的是"[[]]"这样的写法,它匹配的是"[]",即左方括号后紧跟一个右方括号。

总结

所以如果要在字符组中包含"["或者"]",必须分别写在字符组的两端,即中间不该包含其他字符,以免被当作是字符组标记。

2011年11月5日星期六

Bash Shell中命令行选项/参数处理


原文地址:这里


0.引言


   写程序的时候经常要处理命令行参数,本文描述在Bash下的命令行处理方式。

   选项与参数:

   如下一个命令行:
./test.sh -f config.conf ---prefix=/home

   我们称-f为选项,它需要一个参数,即config.conf, -v 也是一个选项,但它不需要参数。

   --prefix我们称之为一个长选项,即选项本身多于一个字符,它也需要一个参数,用等号连接,当然等号不是必须的,/home可以直接写在--prefix后面,即--prefix/home,更多的限制后面具体会讲到。
   在bash中,可以用以下三种方式来处理命令行参数,每种方式都有自己的应用场景。

    * 手工处理方式
    * getopts
    * getopt

   下面我们依次讨论这三种处理方式。

1. 手工处理方式


   在手工处理方式中,首先要知道几个变量,还是以上面的命令行为例:

    *    $0 : ./test.sh,即命令本身,相当于C/C++中的argv[0]
    *    $1 : -f,第一个参数.
    *    $2 : config.conf
    *    $3, $4 ... :类推。
    *    $#  参数的个数,不包括命令本身,上例中$#为4.
    *    $@ :参数本身的列表,也不包括命令本身,如上例为 -f config.conf -v --prefix=/home
    *    $* :和$@相同,但"$*" 和 "$@"(加引号)并不同,"$*"将所有的参数解释成一个字符串,而"$@"是一个参数数组。如下例所示:


 1 #!/bin/bash 2 
 3 for arg in "$*"
 4 do
 5     echo $arg 6 done 7 
 8 for arg in "$@"
 9 do
10     echo $arg11 done12 


执行./test.sh -f config.conf -n 10 会打印:

-f config.conf -n 10    #这是"$*"的输出

-f   #以下为$@的输出

config.conf

-n

10



   所以,手工处理的方式即对这些变量的处理。因为手工处理高度依赖于你在命令行上所传参数的位置,所以一般都只用来处理较简单的参数。如

   ./test.sh 10

   而很少使用./test -n 10这种带选项的方式。 典型用法为:
#!/bin/bash
if [ x$1 != x ]
then
    #...有参数
else
then
    #...没有参数
fi


为什么要使用 x$1 != x 这种方式来比较呢?想像一下这种方式比较:


if [ -n $1 ]  #$1不为空

但如果用户不传参数的时候,$1为空,这时 就会变成 [ -n ] ,所以需要加一个辅助字符串来进行比较。

手工处理方式能满足大多数的简单需求,配合shift使用也能构造出强大的功能,但在要处理复杂选项的时候建议用下面的两种方法。

2. getopts/getopt


处理命令行参数是一个相似而又复杂的事情,为此,C提供了getopt/getopt_long等函数,
C++的boost提供了Options库,在shell中,处理此事的是getopts和getopt.

getopts和getopt功能相似但又不完全相同,其中getopt是独立的可执行文件,而getopts是由Bash内置的。

先来看看参数传递的典型用法:

    * ./test.sh -a -b -c  : 短选项,各选项不需参数
    * ./test.sh -abc   : 短选项,和上一种方法的效果一样,只是将所有的选项写在一起。
    * ./test.sh -a args -b -c :短选项,其中-a需要参数,而-b -c不需参数。
    * ./test.sh --a-long=args --b-long :长选项

我们先来看getopts,它不支持长选项。

使用getopts非常简单:
代码

#test.sh

#
!/bin/bash
while getopts "a:bc" arg #选项后面的冒号表示该选项需要参数do
        
case $arg in
             a)
                echo 
"a's arg:$OPTARG" #参数存在$OPTARG中
                ;;
             b)
                echo 
"b"
                ;;
             c)
                echo 
"c"
                ;;
             
?)  #当有不认识的选项的时候arg为?
            echo 
"unkonw argument"
        exit 
1
        ;;
        esac
done


现在就可以使用:
./test.sh -a arg -b -c

./test.sh -a arg -bc
来加载了。
应该说绝大多数脚本使用该函数就可以了,如果需要支持长选项以及可选参数,那么就需要使用getopt.
下面是getopt自带的一个例子:

#
!/bin/bash

# A small example program 
for using the new getopt(1) program.
# This program will only work with bash(
1)
# An similar program 
using the tcsh(1) script language can be found
as parse.tcsh

# Example input and output (from the bash prompt):
# .
/parse.bash -a par1 'another arg' --c-long 'wow!*\?' -cmore -" very long "
# Option a
# Option c, no argument
# Option c, argument `more
'# Option b, argument ` very long '# Remaining arguments:
--> `par1'--> `another arg'--> `wow!*\?'
# Note that we use `
"$@"' to let each command-line parameter expand to a
# separate word. The quotes around `$@' are essential!
# We need TEMP as the `eval set --' would nuke the return value of getopt.

#
-o表示短选项,两个冒号表示该选项有一个可选参数,可选参数必须紧贴选项
#如
-carg 而不能是-c arg
#
--long表示长选项
#
"$@"在上面解释过
-n:出错时的信息
-- :举一个例子比较好理解:
#我们要创建一个名字为 
"-f"的目录你会怎么办?
# mkdir 
-f #不成功,因为-f会被mkdir当作选项来解析,这时就可以使用
# mkdir 
-- -f 这样-f就不会被作为选项。

TEMP
=`getopt -o ab:c:: --long a-long,b-long:,c-long:: \
     
-'example.bash' -- "$@"`
if [ $? != 0 ] ; then echo "Terminating..." >&2 ; exit 1 ; fi

# Note the quotes around `$TEMP
': they are essential!
#set 会重新排列参数的顺序,也就是改变$1,$2...$n的值,这些值在getopt中重新排列过了
eval 
set -- "$TEMP"

#经过getopt的处理,下面处理具体选项。
while true ; do
        
case "$1" in
                
-a|--a-long) echo "Option a" ; shift ;;
                
-b|--b-long) echo "Option b, argument \`$2'" ; shift 2 ;;
                
-c|--c-long)
                        # c has an optional argument. As we are 
in quoted mode,
                        # an empty parameter will be generated 
if its optional
                        # argument 
is not found.
                        
case "$2" in
                                
"") echo "Option c, no argument"; shift 2 ;;
                                
*)  echo "Option c, argument \`$2'" ; shift 2 ;;
                        esac ;;
                
--) shift ; break ;;
                
*) echo "Internal error!" ; exit 1 ;;
        esac
done
echo 
"Remaining arguments:"
for arg do
   echo 
'--> '"\`$arg'" ;
done


比如我们使用
./test -a  -b arg arg1 -c
你可以看到,命令行中多了个arg1参数,在经过getopt和set之后,命令行会变为:
-a -b arg -c -- arg1
$1指向-a,$2指向-b,$3指向arg,$4指向-c,$5指向--,而多出的arg1则被放到了最后。

3.总结

一般小脚本手工处理也许就够了,getopts能处理绝大多数的情况,getopt较复杂,功能也更强大。

2011年10月29日星期六

Keyboard Not Found

一台机子没有插键盘(PS/2),启动的时候会停在“Keyboard not found”的错误。如果要把这台机子当作服务器,显得没键盘装得才像。

可以通过BIOS设置来去除这个错误。不知道不同版本的BIOS是否设置一致,在我这里是这样的:
在BIOS的某个选项里,有 Halt on All errors,修改为 Halt on no error,然后会出现新的Keyboard选项,将其设置为disable,意思是忽略Keyboard的错误,重启即可。

2011年10月27日星期四

Bash 命令提示符

Bash 命令提示符是由环境变量PS1控制的,PS2控制命令多行输入时的提示符。Bash提供了
一些转义字符,供定制PS1(当然如果要定制PS2也可以使用,但一般很少定制PS2吧)时使用。
以下列举一些常用的:
  1. \h 主机名字
  2. \u 当前用户名字
  3. \w 当前工作目录
  4. \$ UID为0时(也就是root)效果为#,否则为$
  5. \t 当前二十四小时制时间
  6. \T 当前十二小时制时间
  7. \[ 开始一串非打印字符
  8. \] 非打印字符串结束

提示符颜色

还可以给提示符添加颜色,也是通过在PS1中使用特定的序列来实现的。"\e["表示颜色序列
的开始。后面紧跟表示颜色的数字,不同的数字代表不同的颜色,有些表示前景色,有些表
示背景色,有些表示光标闪烁颜色,数字间用“;"分隔,且没有顺序关系,最后"m"表
示颜色序列的结束。后续的字符将按照颜色序列的定义显示,直到"\e[0m"重置为默认值。

比如一个可能的PS1为:“export PS1="\e[32;40m\w> \e[0m”

编码 颜色动作
0 重新设置属性到缺省设置
1 设置粗体
2 设置一半亮度
4 设置下划线
5 设置闪烁
7 设置反向图象
22 设置一般密度
24 关闭下划线
25 关闭闪烁
27 关闭反向图象
30 设置黑色前景
31 设置红色前景
32 设置绿色前景
33 设置棕色前景
34 设置蓝色前景
35 设置紫色前景
36 设置青色前景
37 设置白色前景
38 在缺省的前景颜色上设置下划线
39 在缺省的前景颜色上关闭下划线
40 设置黑色背景
41 设置红色背景
42 设置绿色背景
43 设置棕色背景
44 设置蓝色背景
45 设置紫色背景
46 设置青色背景
47 设置白色背景
49 设置缺省黑色背景


的作用


还要特别提一下
的作用,它通知Bash两者间的字符为非打印字符,
当Bash计算提示符长度时,会忽略这一串字符。这是很有必要的,比如对于定义颜色的特殊
序列,就应该包含在其中,因为颜色序列并非打印字符。如果没有将非打印字符(比如颜色
序列)包含在
之间,当输入较短命令时不会有问题,但是当输入长命令,超过使用的
终端窗口的宽度时,因为Bash将非打印字符都算入提示符长度,造成命令换行的错误,终端
窗口的显示将会错乱。


titlebar


默认情况下,
PS1=\[\e]0;\u@\h: \w\a\]${debian_chroot:+($debian_chroot)}\u@\h:\w\$

分析这一段代码,发现
的“\e]0;\u@\h: \w\a”还是解释不了。其实"\e]0;" 与
"\a"之间的序列用于控制终端窗口titlebar的显示。还可以把0换成1或者2,分别控制任务
栏终端窗口titlebar显示和主窗口的titlebar显示,即:

\e]1;icon-title\a\e]2;main-title\a

参考资料


[1]Bash Shell PS1: 10 Examples to Make Your Linux Prompt like
Angelina Jolie

[2]Tip: Prompt magic
[3]http://tldp.org/HOWTO/Bash-Prompt-HOWTO/xterm-title-bar-manipulations.html
[4]tput 入门

2011年10月22日星期六

Ubuntu登录后死机

给一旧电脑安装Ubuntu ( 9.10 ),安装好后进入桌面没几秒就死机,屡试不爽。TTY登录则一切正常。猜测是机器过于陈旧,所以无法运行桌面。难道就因此放弃桌面么?但是由于以前有一台大抵相同配置的机器是可以运行GNOME,所以对上述轻率的结论还是有疑惑的。

良思无策,放狗一搜,发现这篇文章描述的状况与我遇到的并无二致,原因他也说了,是硬件不支持compiz造成的,但换个窗口管理器还是可以使用桌面环境的。按图索骥,登陆到tty,修改~/.gconf/desktop/gnome/applications/window_manager/%gconf.xml文件,把两个/usr/bin/compiz换成/usr/bin/metacity。重启系统,总算可以正常运行了。不得不感慨下这位同学发现问题和解决问题的能力还是极强的,而我连窗口管理器,X Windows,窗口环境等概念都分不清,差距阿。

compiz 和 metacity 都是窗口管理器,而且metacity是gnome默认的窗口管理器(吐槽:看到的资料是这么说,可是为什么安装好启动就是compiz,那算什么默认阿)。ubuntu wiki这里有提到一点,“系统--首选项--外观--视觉效果”如果不是选“无”,则compiz会作为窗口管理器(吐槽:metacity,你到底算个啥默认窗口管理器哦)。

2011年10月19日星期三

FAIL:20111019

DropBox的主页默认上不去(吐槽:难道我会以为是DropBox挂掉了么?!),但是可以通过某种途径登上,下载了dropbox的deb安装包。

dpkg安装好dropbox deb包(备注:安装的只是一个python程序,在/usr/bin/dropbox),在应用程序--互联网--DropBox 启动 dropbox,提示连接错误,并建议配置http_proxy环境变量。正常反应,我想,毕竟是有认证过的。那怎么配置http_proxy环境变量?写入/etc/profile?不妥,不知道会不会影响其他程序。放狗一搜,利用这篇文章中提到的方法:

http_proxy=http://127.0.0.1:8580 /usr/bin/dropbox start -i

开始下载安装dropbox主程序(第一次运行才会安装,安装在主目录下.dropbox-dist中)。接着会自动运行dropbox,但仍然连接不上服务器,不过有了新的提示:不能建立安全的连接,原因是你的时间和日期设置得不正确。当然系统的时间和日期设置是正确的,放狗搜索,看了N多贴,挣扎了几个小时,毫无头绪,只能用一个字来形容当时的心情:FAIL!

最后让我发现原因是网络没有国际出口权限!我猜dropbox会连接某国际时间服务器来验证本地系统时间是否正确,但不幸的是因为网络没有国际出口权限,它居然提示系统时间和日期设置不正确。还有一个小疑问,虽然DropBox的主页依然“正常”登陆不上去,但是客户端在没有用任何其他途径的情况下,能很好地工作,而且速度还不错(200KB/s),着实让我有点意外。

2011年10月16日星期日

ubuntu下goagent开机自启动

GoAgent的安装和部署可以参考一篇旧文。下面讨论的是如何开机运行GoAgent。

Windows下是没有这种烦恼的(吐槽:linux下其实也没烦恼,“庸人”自扰而已),只需将proxy.exe加入开机启动项即可。在Ubuntu上,每次需要执行python proxy.py才可以运行goagent,而且需要一直开着终端,着实有点不太方便。要实现开机启动GoAgent,可以在 ~/.profile (如果没有该文件,自己创建)中添加以下语句:

# goagent
cd /usr/local/phus-goagent-b1f5004/local
python proxy.py > /dev/null 2>&1 &
cd ~

/usr/local/phus-goagent-b1f5004是我解压goagent的目录。

有几个问题需要交代一下:
  1. nohup

    当我刚开始尝试开机启动GoAgent时,使用的方法是在~/.bash_login(Ubuntu默认没有该文件,自己创建的)中添加 python /usr/local/phus-goagent-b1f5004/local/proxy.py, 显然失败了。这其实有两个问题,一个问题下面会介绍,另一个问题就是没有使用nohup,使得当终端关闭时,程序也就结束了。

    nohup的作用就在于忽略所有挂断(SIGHUP)信号,在shell注销后保持程序继续运行。它会把输出重定向到nohup.out的一个文件中,保存在当前目录下,如果无法在当前目录下创建文件,则会保存在HOME/nohup.out。但是`nohup' does not automatically put the command it runs in the background; you must do that explicitly, by ending the command line with an `&'.",所以通常都需要在nohup后使用&。

    在上面的例子中,我把输出和err输出都重定向到/dev/null,这样就免得生成nohup.out文件。

    Update:
    很不幸地发现,不使用nohup也是完全可以正常使用的(吐槽:之前多么无知无畏阿),但上面关于nohup的介绍是没有任何问题的。提醒一点,即便没有使用nohup, 命令后添加&使得程序运行在后台,还是不能省略的。

  2. /etc/profile .bash_profile .bash_login .profile
    我开始尝试失败的另一个原因就是把命令写在了一个错误的地方~/.bash_login。其实当以图形界面登陆时,并没有读取.bash_login文件,使得命令根本没有运行。shell的初始化文件不同情景下读取顺序情况如下:

    1. 图形模式登录时,顺序读取:/etc/profile和~/.profile
    2. 图形模式登录后,打开终端时,顺序读取:/etc/bash.bashrc和~/.bashrc
    3. 文本模式登录时,顺序读取:/etc/bash.bashrc,/etc/profile和~/.bash_profile
    4. 从其它用户su到该用户,则分两种情况:
    (1)如果带-l参数(或-参数,--login参数),如:su -l username,则bash是lonin的,它将顺序读取以下配置文件:/etc/bash.bashrc,/etc/profile和~ /.bash_profile。
    (2)如果没有带-l参数,则bash是non-login的,它将顺序读取:/etc/bash.bashrc和~/.bashrc
    5. 注销时,或退出su登录的用户,如果是longin方式,那么bash会读取:~/.bash_logout
    6. 执行自定义的shell文件时,若使用“bash -l a.sh”的方式,则bash会读取行:/etc/profile和~/.bash_profile,若使用其它方式,如:bash a.sh, ./a.sh,sh a.sh(这个不属于bash shell),则不会读取上面的任何文件。
    7. 上述原则凡是读取到~/.bash_profile的,若该文件不存在,则读取~/.bash_login,若前两者不存在,读取~ /.profile。

    更详细的讨论可以前往这里参考一下。

参考资料

[1]nohup MAN Page
[2]shell的初始化文件
[2]Ubuntu开机自动启动GAppProxy

2011年10月13日星期四

shell之引用

在shell中有两类字符,一类是普通字符,另一类是所谓的"元字符",在shell中有特殊的含
义或用法。

当我们需要去掉元字符的特殊含义而恢复其字面意义时就必须使用"引用".通常有三种引用
方式,分别是转义(Escape,使用反斜杠字符\),强引用(使用单引号)和弱引用(使用双引号):
  1. 转义是用反斜杠放在需要转义的字符前,使得"元字符"变成普通字符。
  2. 强引用是用单引号把字符串包含起来,其中任何字符都看作普通字符(包括双引号),除
    了单引号自身。所以无法在两个单引号之间包含一个单引号,用\转义也不行(\已被当作普
    通字符,失去了转义的作用)。
  3. 弱引用是用双引号把字符串包含起来,除了“(双引号,double quote),\(反斜杠,
    backslash),$(美元符号),`(注意不是单引号,而是反引号,ESC下的那个键,作用是命令替
    换)之外,其他的大部分字符被当作普通字符。

好像上面已经把规则说得够清楚了,那么试着解读一下下面这个语句:
awk 'BEGIN {print "Here is a single quote <'"'"'>"}'
输出:
Here is a single quote <'>
乍一看,对于这个结果还是蛮惊讶而且比较迷糊。其实这里利用了shell的引用的一个原则:
引用与后面接着的非引用字符,或者其他引用,会整合一个命令参数
所以对于上面那个例子的解释是,其实包括了三个引用:
  1. 'BEGIN { print "Here is a single quote<'
  2. "'"
  3. '>" }'
然后三个引用连接成为一个参数。下面的两个语句会有同样的效果:
awk ’BEGIN { print "Here is a single quote <’\’’>" }’
awk "BEGIN { print \"Here is a single quote <’>\" }"


参考资料:这里

2011年10月11日星期二

FAIL:20111011

这两天遇到四个个问题:
  • 双系统 windows 和 ubuntu 时间差8小时.这是由于Linux 操作系统是以 CMOS 时间做为格林威治标准时间,再根据系统设置的时区来确定目前系统时间。但是Windows 会将CMOS 时间看作本地时间。解决方法之一是让Ubuntu 也使用本地时间,修改/etc/default/rcS,把里面的 UTC=yes 改为 UTC=no。
  • smplayer(源安装)播放 rmvb 时有噪音,略微话音不同步,而用mplayer播放时没有这问题。不得其解。忍,我用mplayer。
  • mplayer不能正常播放mms,话说之前有一阵子是可以播放的,windows下用mediaplayer 播放毫无压力。不得其解。忍?这怎么忍?!尝试用其他播放器(vlc),同样不能播放。
  • virtualbox 下播放视频,或者用mediaplayer播放mms,花屏。去除virtualbox 2D加速,3D加速后,可以正常播放。这是不是应验前几天看到的“VirtualBox 烂死了”的论调?不得其解。可能与我显卡有关(GT540M,nVidia驱动,非源安装)。
所以很自然地,我想重新编译mplayer,或许能解决不能播放mms的问题。这样就开始了一整天炼狱般的体验(吐槽:或许这才是常态,我太naive): 各种库的依赖(吐槽: 忘记了 apt-get build-dep),编译时的漫长等待,编译参数的深邃莫测,以及最重要的...仍然不能播放mms!

一个字:FAIL!

2011年10月9日星期日

Emacs 中的简单排版

Tabs的使用

emacs中的Tab默认长度是8个字符,而且Tabs的行为有点智能,它并不是简单地插入一个制
表符,而是会依照上行的非空白字符的位置,来选择光标的移动位置。比如:
输入:TAB TAB Hello World,得到:“\t \t Hello World”,接着按下ENTER另
起一行,再按下Tab,此时光标会直接跳到与Hello中‘H’对齐的那一列,而不是简
单地插入一个制表符。
通常情况下,这正是我们所需要的行为,但当我们正需要一个Tab时,这种“智能“的判
断会给我们造成困扰。这时,可以使用C-q TAB。

我们还可以对TAB的一些特性进行定制。比如把tab的长度设置为常用的4字符:

(setq-default tab-width 4)

因为TAB会根据非空白字符的位置来移动光标,当需要移动的距离不是TAB字符长度的倍数,
会插入制表符和空格符来满足要求。应用这样规则的文本在用别的编辑器打开时,因为
TAB长度的差异,可能会使排版变得难看,所以更好的方法是当使用TAB键时不要插入制表符
\t,而是插入同等长度的空格符。可以用以下语句实现要求:

(setq-default indent-tabs-mode nil)


Autofill mode与M-q


Autofill 和 M-q 都与一个变量有关系,那就是default-fill-column,它的作用是设定排版
时每行的最大长度,这样排版出来的文章才会显得整理。设置default-fill-column为80,因
为它刚好是我屏幕的一半,这样分屏时也挺不会折行 :)

(setq default-fill-column 80)

Autofill 是一个mini mode,它会使输入超过default-fill-column时自动换行,免去用M-q
命令排版的麻烦。可以在.emacs文件中加入以下语句:

;;所有模式都默认开启auto-fill
(setq-default auto-fill-function 'do-auto-fill)

M-q 的函数是 fill-paragraph,会对所在段落进行排版,主要操作是去除多余的空白字符,
并在超过default-fill-column处加入换行。需要注意的一点是,默认情况下,段落间的标志
是相隔一空行,所以如果只是用ENTER另起一行,在用 M-q 排版时会被当作是同一段落。

fill前缀


Autofill 在换行时,可以添加一些前缀(prefix)。前缀的自动识别由变量
adaptive-fill-regexp 决定,emacs在断行时自动识别出符合的前缀,新起的一行会添加该
前缀。默认情况下空格与制表符都包括在能被识别的前缀里,所以如果新起的一行会保持与
上一行相同的缩进,甚是方便,充分体现了emacs 的强大。

也可以在你需要使用的前缀后敲入"C-x .",则光标之前的字符串会被当作前缀,换行时就会
添加该前缀。利用这个功能,我们可以方便地写出类似下面这样的排版:
EMACS太强大了:我同意
EMACS太强大了:我举双手赞成
EMACS太强大了:我坚决支持
EMACS太强大了:呵呵
另起一行,再输入一次"C-x .",就会取消这个手动设置的前缀。

更详细的介绍可以参考这里

居中


居中是在排版中常见的任务,Emacs提供了函数来实现这个任务:
center-line 绑定键 M-o M-s 使当前行居中
center-paragraph 绑定键 M-o M-S 使当前段居中
center-region 使选中区域居中

缩进

按照上述的设置和使用,在编写时应该能写出符合要求的排版。如果在输入完后,才决
定需要对某段文字进行缩进,这时可以使用C-M-\ (indent-region),它对选中的区域缩进一
层。注意,如果需要缩进多层,不该使用该命令,不然会使排版变得很糟糕。有更适合的命
令来实现多层的缩进,那就是C-x TAB (indent-rigidly),默认情况下,该命令会缩进一个
字符,可以配合C-u 使用。

2011年10月8日星期六

ubuntu 下配置Java环境

下载,安装

在Ubuntu源中可以直接安装jdk,但是只能安装Openjdk,而且版本不一定是最新的。还是选择使用官方版,在Oracle Java SE Downloads(吐槽: SUN 阿 SUN)下载Linux平台的自解压包,后缀为bin,说明是可执行文件。把自解压包放在准备安装的目录(比如/usr/local),需要通过chmod添加可执行属性,然后执行该bin文件解压即可。

配置环境

要使用Java,还需要配置环境参数JAVA_HOME,CLASSPATH,为了在终端中使用JAVA,还需把$JAVA_HOME/bin加入到PATH中。

可以在$HOME/.bashrc 中使用(假设java安装在/usr/local/jdk1.6.0_27):

# JAVA Environment
JAVA_HOME=/usr/local/jdk1.6.0_27
CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar
PATH=$PATH:$JAVA_HOME/bin
export JAVA_HOME CLASSPATH PATH


但这样有一个问题,环境变量并不是全局的(只在所谓的nologin shell中有效),也就是说安装eclipse后,如果在桌面中启动eclipse会提示找不到JRE(eclipse需要JRE环境)。

所以更合适的方法是在/etc/profile中添加上述语句,因为login shell 会读取该文件,使得环境变量是全局的。

2011年10月7日星期五

latex 生成带QR码的名片

抄袭地址:这里


生成的效果如下:


要开发QR CODE译码应用程序除了使用现成的译码库(比如zxing)外,其实还要知道各服务的标准协议才能开发出真正的应用程序。比方说要如何判别什么时候是联系人、什么时候是打电话、什么时候是发短信、什么时候是开启网页、什么时候是纯文本备注。
这些都有标准协议,以下就简单说明各QR CODE应用协议:
发送短信
SMSTO:13800138000:BABY I LOVE YOU
发送彩信
MMSTO:13800138000:send mms
发送网址
http://www.hoojar.com/
拨打电话
TEL:13800138000
发送邮件
MATMSG:TO:hoojar@hoojar.com;SUB:TEST;BODY:THIS IS TEST MAIL FOR QR CODE;;
电话本
MECARD:N:李,林;ADR:湖南省株洲市;TEL:+8313800138000;EMAIL:hoojar@hoojar.com;URL:http://m.hoojar.com/;;
书签
MEBKM:TITLE:慧佳生活网;URL:http://www.hoojar.com/;;

使用pst-barcode宏包生成QR码,QR码中包含的信息为电子名片信息,格式为MECARD。虽然Vcard是通用的电子名片标准,但由于二维码的信息量有限,Vcard的条目过于详细,所以在用QR码存储名片信息时,更常用MeCard格式

模板如下:

\documentclass[11pt,a4paper]{memoir}

\setstocksize{55mm}{85mm} % UK Stock size
\setpagecc{55mm}{85mm}{*}
\settypeblocksize{45mm}{75mm}{*}
\setulmargins{5mm}{*}{*}
\setlrmargins{5mm}{*}{*}

\setheadfoot{0.1pt}{0.1pt}
\setheaderspaces{1pt}{*}{*}
\checkandfixthelayout[fixed]

\pagestyle{empty}

\usepackage{pstricks}
\usepackage{pst-barcode}

\begin{document}
%\pagecolor[cmyk]{.22,.36,.51,.08}%
\begin{Spacing}{0.75}%
\noindent
\textbf{Andrew~Brampton~Ph.D.}\\
\rule{75mm}{1mm}\\
\begin{minipage}[t]{30mm}
\vspace{-1mm}%
\begin{pspicture}(30mm,30mm)
% The MECARD format is used to exchange contact information. More information at:
% http://www.nttdocomo.co.jp/english/service/imode/make/content/barcode/function/application/addressbook/index.html
\psbarcode{MECARD:N:Brampton,Andrew;EMAIL:a.bramptonATlancs.ac.uk;URL:http://bramp.net;;}{eclevel=L width=1.181 height=1.181}{qrcode}
\end{pspicture}
\end{minipage}
\hspace{1mm}
\begin{minipage}[t]{42mm}
\vspace{-1mm}%
\begin{flushright}
{\scriptsize
\begin{Spacing}{1.5}%
% \textbf{Research Associate}\\
\textbf{Network Researcher}\\
Computing Department\\
Lancaster University\vspace{9mm}\\
\end{Spacing}
}
{\tiny
\textbf{email:} a.brampton AT lancs.ac.uk\\
\textbf{web:} http://bramp.net/\\
\vspace*{2mm}
}
\end{flushright}
\end{minipage}
\rule{75mm}{1mm}
\end{Spacing}
\end{document}

2011年9月30日星期五

还真是浪费了一天

分区

安装ubuntu时分了两个区,大部分划给/,剩下4g划给swap。后来想重新装个win7,发现已没空间。fdisk并不能调整分区大小,也就是说,它可以删除分区,在未使用空间中建新分区,但是它不能既保证原有分区正常使用,又能划出新的剩余空间建新分区。

GParted才可以满足上述需求,类似于Windows下的分区软件。其项目主页上有具体的使用方法,要点是要调整已挂载系统分区时(我面临的情况)需要使用GParted live cd,通过grub2启动GParted live cd。

安装Win7 后修复grub

主要步骤有以下几步:
  1. live cd 启动系统
  2. 在live 系统中,挂载需要修复grub的硬盘。
    这里假设硬盘为sda:
    sudo mount /dev/sda1 /mnt
  3. 修复grub
    grub-install --root-directory=/mnt /dev/sda
  4. 重启系统,不出意外grub菜单应重新出现,选择Ubuntu(但选择Ubuntu后,会提示什么参数不正确,不过仍能启动)
  5. 进入Ubuntu后,执行
    sudo grub-install /dev/sda (是的,重新执行一遍)
    sudo update-grub2 (会把windows启动一起找到的)
最后一步很重要,可以使第4步出现的参数不正确错误消除掉。

设置grub菜单timeout

修改 /etc/default/grub 中设置GRUB_TIMEOUT。

Gnome桌面显示回收站等

gconfig-editor打开配置管理器,到/apps/nautilus/desktop 中修改。

Ubuntu版本10.04,Grub2