Xiangism

从一个无知角落里开始,蹒跚学步,一个未知到另一个未知,在跌跌撞撞中越走越快,越走越远,最后宇宙也为之开源。对于探索者来说,最后他们的思想总是变得和自己的足迹一样伟大。
  博客园  :: 首页  :: 联系 :: 管理

打磨程序员的专属利器——文本

Posted on 2015-06-15 21:55  Xiangism  阅读(3133)  评论(17编辑  收藏  举报

打磨程序员的专属利器分三个专题展示--

1. 命令行&文件

2. 快捷键

3. 文本 (本文)

 

1. 记笔记

好记性不如烂笔头,对于程序员更是如此。学习某种新语言或者某个库,若事先不记点笔记,几年后再拣起来会非常地慢。

之前自己一直用“为知笔记”,但慢慢发现了几个问题。

a. 这个软件喜欢升级,而每次升级都将界面改得面目全非(其实现在绝大多数软件都喜欢没事升级)。我只是想一个简洁的软件界面,只想专注于笔记,而为知的每一次更新后都得花精力在熟悉界面上。

b. 为知的笔记不是文本,这样笔记内容几乎只能用为知才能打开。如果想在ubuntu或者Mac上看,只有期待它出对应的客户端或者用网页查看(但如果没网时也不能查看本地数据了)。不过现在我打开wiz的网页版笔记,只能另我大失所望,本人只是想要一个左边是树形控件展示笔记的结构,右边可以显示笔记内容的界面,而wiz网站的在线笔记做得太复杂了。

c. 有时不知是服务器还是网络问题,同步居然出现问题。在家里电脑写的东西,来到公司后发现居然没有同步过来,fuck!!!!

忍无可忍后,就动了自己实现个笔记软件的想法。最开始是用c#写,笔记保存成.mht格式,用IE控件进行预览,用word进行编辑(这是从wiz中得到的启发)。不过后来也发现不是文本的笔记根本不能跨平台。于是又重写了一遍,用c++实现,下面是软件的界面,很简洁,但完全够自己用了。

这个程序从底层反锯齿的2d图形库,到界面库,到Markdown的解析,都是自己重新造轮子实现(虽说在软件界不鼓励重新造轮子,但通过造轮子可以将图形库、界面库理解得更透彻)。由于太过简陋就不发布出来,等过段时间有空了用QT再重写一个相对漂亮点的界面再公布源码和程序。

2. Markdown

上面的笔记中提到了Markdown。《程序员修炼之道》中提到笔记得到文本来记录。最开始本人一直认为“文本”就是“纯文本”,直到在开源软件中发现有.md文件,于是知道了有Markdown这种东东,然后忽然识别到文本也可以做得很漂亮(html,css就是文本)。

Markdown是html的一个子集。可以用少量的标示符就可以构造出漂亮的文本排版。下面简单介绍下Markdown的标示符:

a. 行首的#可以指明段落,一个表示第一级,二个表示第二级,以此类推

b. 两行```之间的内容可以被解析为code格式,就像上图中有灰色背景的区域,这种可以方便放代码。

c. 行首的 * 可以被解析成html的 <li>

d. 用[name](http://www.test.com "test") 这种格式可以指定超链接

e. 用下面的格式可以构造出如下的表格

name | age
---|---
man1 | 45
man2 |2

.... 

更多Markdown语法可以参数其官网。反正都挺好学,也比较易用,可以在记笔记时不因格式问题而花太多的精力,让我们专注于内容

3. 备份、同步

有了上面DIY的笔记软件后,对于多地办公、多台电脑办公就得考虑如何同步笔记了。

开始本人是使用金山网盘,但也是因为同步问题,有时真是金山的服务器挂了,导致无法同步。还一个严重的问题,网盘没有像svn,git那样明确的update,commit,pull,push操作,这样在同步时真就会出现问题。然后本人试了几乎所有的网盘,发现在某些情况下100%的情况会出现同步失败的问题。

************************

在写这篇博文时为了再次证实下,于是本人又装了次金山快盘。

新建一个文件夹,在这个文件夹下再新建两个文本文档,往其中一个文本文档中随便写点东西,这时去网页版的金山快盘上,就发现真就没有把刚才的东西同步上来。比吃了蟑螂还恶心!!!

qq云盘和360网盘以前还可以在Explorer中查看本地数据,但现在只有用它自己的程序来操作本地数据了。

百度云盘同样也有这样问题,没有本地路径。

************************

同样是忍无可忍

开始使用 http://svn.jundie.net/,但由于本人小气,不想花钱买私有项目,只有把自己的笔记弄成开源项目,虽说没什么重要东西,但总感觉不靠谱。

于是找到git.oschina.net,并且发现基于git有非常多的服务器,并且都可以免费申请私有项目。于是爱上了git,为了保险,将自己所有的代码都放到git上,当然是私有项目(遥想当年,由于本人折腾电脑、折腾分盘,将大一下刚学程序半年写的vb6俄罗斯方块程序源码弄丢了,多么地心疼啊)~~

于是妈妈再也不用担心代码弄丢了

为了方便在Explorer中访问笔记,可以使用前文介绍的subst命令,将笔记路径映射成一个虚拟磁盘以便访问。 

4. 学习一门文本语言

《程序员修炼之道》中说程序员应该每一年学习一门新语言,我们打个折扣1~2年学习一门新语言。并且非常有必要学习一门文本操作语言,比如Python,Ruby或者是Perl。虽然用C#,java,甚至是c/c++都可以进行文本操作,但编写代码的速度肯定不及脚本语言。

本人最开始学习的是Perl语言。这门语言咋一看觉得不好学——充满了各种奇奇怪怪的符号。不过如果沉下心来将《Perl语言入门》、《Perl语言编程》两本书看完,然后再动手自己写几个实用的小程序,Perl的学习之路基本上就算是毕业了。

下面展示几个本人所写的perl程序。也是比较简陋(本人侧重于实用):

a. ls程序

linux下的ls命令可以将当前路径下的文件、文件夹都显示出来。windows下没有这样的程序,于是本人实现了一个。

  1 use 5.010;
  2 use strict;
  3 use warnings;
  4 
  5 use Win32::Console;
  6 use Win32::Console::ANSI;
  7 use Term::ANSIColor;
  8 use POSIX();
  9 
 10 
 11 use utf8;
 12 use Encode::Locale;
 13 
 14 binmode STDIN, ":encoding(console_in)";
 15 binmode STDOUT, ":encoding(console_out)";
 16 binmode STDERR, ":encoding(console_out)";
 17 
 18 sub Main;
 19 
 20 Main @ARGV;
 21 
 22 my $is_dir_show;
 23 my $is_file_show;
 24 
 25 my $is_show_size;
 26 my $is_sort_by_size;
 27 
 28 my @folds;
 29 my @files;
 30 
 31 my $max;
 32 
 33 my $col_count;
 34 my $col_width;
 35 
 36 my $con_w;
 37 my $con_h;
 38 
 39 sub Help {
 40     #system("chcp 936");
 41     my $str = "ls 显示当前目录下的文件和文件夹名\n".
 42     "默认情况下文件和文件夹都将显示。-f 显示文件   -d 显示文件夹\n".
 43     "默认情况下不显示文件和文件夹的大小。   -s 将显示其大小\n".
 44     "如果文件名中有中文,则一定要确保控制台的代码页为936,用chcp 936可以改变代码页";
 45     print $str;
 46 }
 47 
 48 sub IsHelp {
 49     foreach (@_) {
 50         if (lc($_) eq '?' ||
 51         lc($_) eq '/?' ||
 52         lc($_) eq '-help' ||
 53         lc($_) eq '--help'
 54         ) {
 55             return 1;
 56         }
 57     }
 58     return 0;
 59 }
 60 
 61 #---------------
 62 # 判断命令行中是否有-d,如果有则表示显示文件夹
 63 #---------------
 64 sub IsDir {
 65     foreach (@_) {
 66         if (lc($_) eq '-d') {
 67             return 1;
 68         }
 69     }
 70     return 0;
 71 }
 72 #---------------
 73 # 多年命令行中是否有-f,如果有则表示显示文件
 74 #---------------
 75 sub IsFild {
 76     foreach (@_) {
 77         if (lc($_) eq '-f'){
 78             return 1;
 79         }
 80     }
 81     return 0;
 82 }
 83 #---------------
 84 # 判断命令行中是否有-s, 如果有则要显示大小
 85 #---------------
 86 sub IsSize {
 87     foreach (@_) {
 88         if (lc($_) eq '-s') {
 89             return 1;
 90         }
 91     }
 92     return 0;
 93 }
 94 
 95 #---------------
 96 # 判断命令行是否有-S,如果有则要按照文件大小排序,并且显示文件大小
 97 #     即-S包含-s
 98 #---------------
 99 sub IsSortBySize {
100     foreach (@_) {
101         if ($_ eq '-S') {
102             return 1;
103         }
104     }
105     return 0;
106 }
107 #---------------
108 # 获取文件夹的大小
109 # par_1: path
110 #---------------
111 sub GetFoldSize {
112     my $path = shift;
113     my $size = 0;
114 
115     if (-d $path) {
116         my $r = opendir(DIR, $path);
117 
118         if (!$r) {
119             return 0;
120         }
121 
122         my @files = readdir(DIR);
123         closedir(DIR);
124 
125         foreach (@files) {
126             next if $_ eq '.' or $_ eq '..';
127             my $file = "$path\\$_";
128 
129             if (-d $file) {
130                 $size += GetFoldSize($file);
131             } else {
132                 my $s = (-s $file);
133                 $size = $size + $s;
134 
135             }
136         }
137     }
138     return $size;
139 }
140 
141 #---------------
142 # 将字节数转化为G M K 显示
143 # par_1:byte_size;
144 #---------------
145 sub ByteToString {
146     my $b = shift;
147     my $k=0;
148     my $m=0;
149     my $g=0;
150 
151     if ($b >= 1024) {
152         $k = int($b/1024);
153         $b -= $k * 1024;
154     }
155 
156     if ($k >= 1024) {
157         $m = int($k/1024);
158         $k -= $m * 1024;
159     }
160 
161     if ($m >= 1024) {
162         $g = int($m/1024);
163         $m -= $g * 1024;
164     }
165 
166 #    my $str="";
167 #    my $temp;
168 #    if ($g != 0) {
169 #        $str .= "$g"."G";
170 ##        $str .= $temp." "x(6 - length($temp)) ;
171 #    }
172 #    if ($m != 0) {
173 #        $str .= "$m"."M ";
174 ##        $str .= $temp." "x(6 - length($temp)) ;
175 #    }
176 #    if ($k != 0) {
177 #        $str .= "$k"."K ";
178 ##        $str .= $temp." "x(6 - length($temp)) ;
179 #    }
180 #    if ($b != 0) {
181 #        $str .= "$b"."B ";
182 ##        $str .= $temp." "x(6 - length($temp)) ;
183 #    }
184     
185     if ($g) {
186         sprintf("%4dG %4dM %4dK %4dB", $g, $m, $k, $b);
187     } elsif ($m) {
188         sprintf("%10dM %4dK %4dB", $m, $k, $b);
189     } elsif ($k) {
190         sprintf("%16dK %4dB", $k, $b);
191     } else {
192         sprintf("%22dB", $b);
193     }
194     
195 
196 #    if ($str eq "") {
197 #        return "0B";
198 #    }
199 #
200 #    return $str;
201 }
202 
203 #---------------
204 # 获取数组中最大的字符串长度
205 # par:@_ 数组
206 #---------------
207 sub MaxLength {
208     my $max = 0;
209 
210     foreach (@_) {
211         my $len = length $_;
212         if ($len > $max) {
213             $max = $len;
214         }
215     }
216     return $max;
217 }
218 
219 #---------------
220 # 根据获取的col_count, col_width打印文件和文件夹名
221 # 在没有-s选项中这样显示,只显示文件夹名和文件名
222 # par:@_ 文件或文件夹名数组
223 #---------------
224 sub PrintFileName {
225     my $c = 0;
226 
227     foreach (@_) {
228         my $s = $_;
229         print Encode::decode("gb2312", $s);
230 
231         my $len = length $_;
232         my $diff = $col_width - $len -1; #-1 是为了防止窗口宽等于缓冲区宽时引起的空白行问题
233         my $black = " " x $diff;
234         print $black;
235 
236         ++$c;
237         if ($c == $col_count) {
238             print "\n";
239             $c = 0;
240         }
241     }
242 }
243 
244 #---------------
245 # 当不指定显示大小,或者不按大小排序时显示文件
246 #---------------
247 sub PrintNoSize {
248     my $console = Win32::Console->new();
249     ($con_w, $con_h) = $console->Size();
250 
251     $col_count = POSIX::floor($con_w / ($max+4));
252     if ($col_count == 0) {
253         $col_count = 1;
254     }
255     $col_width = POSIX::floor($con_w / $col_count);
256 
257     #print "screen_width: $con_w \n";
258     #print "max_width: $max \n";
259     #print "col_cout: $col_count\n";
260     #print "col_width: $col_width \n";
261 
262     if ($is_dir_show && scalar(@folds)) {
263         my $c = $#folds+1;
264         my $s = "文件夹: $c 个\n";
265         print $s;
266         print color 'bold yellow';
267         PrintFileName @folds;
268         print color 'reset';
269         print "\n";
270     }
271 
272     if ($is_file_show && scalar(@files)) {
273         my $c = $#files+1;
274         my $s = "文件: $c 个\n";
275         print $s;
276         print color 'bold green';
277         PrintFileName @files;
278         print color 'reset';
279     }
280 }
281 
282 #---------------
283 # 显示文件或文件夹的大小
284 #---------------
285 sub PrintBySize {
286     
287     my $width = $max + 4;
288     
289     if ($is_dir_show) {
290         
291         my %hash_dir;
292         foreach (@folds) {
293             $hash_dir{$_} = GetFoldSize($_);
294         }
295         
296         my @k;
297         if ($is_sort_by_size) {
298             @k = sort { $hash_dir{$b} <=> $hash_dir{$a} } keys %hash_dir;
299         } else {
300             @k = sort keys %hash_dir;
301         }
302 
303         foreach (@k) {
304             print color 'bold yellow';
305             print $_;
306             print color 'reset';
307 
308             my $len = length $_;
309             my $diff = $width - $len;
310             my $black = " " x $diff;
311             print $black;
312 
313             print ByteToString($hash_dir{$_});
314             print "\n";
315         }
316     }
317     if ($is_file_show) {
318         my %hash_file;
319         
320         foreach (@files) {
321             $hash_file{$_} = -s $_;
322         }
323         
324         my @k;
325         if($is_sort_by_size) {
326             @k = sort { $hash_file{$b} <=> $hash_file{$a}} keys %hash_file;
327         } else {
328             @k = sort keys %hash_file;
329         }
330         
331         foreach (@k) {
332             print color 'bold green';
333             print $_;
334             print color 'reset';
335 
336             my $len = length $_;
337             my $diff = $width - $len;
338             my $black = " " x $diff;
339             print $black;
340             print ByteToString($hash_file{$_});
341             print "\n";
342         }
343     }
344 }
345 
346 sub List {
347     $is_dir_show = IsDir @_;
348     $is_file_show = IsFild @_;
349     $is_show_size = IsSize @_;
350     $is_sort_by_size = IsSortBySize @_;
351 
352     #如果即没有指定-f,也没有指定-d,则默认为都显示
353     if (!$is_dir_show && !$is_file_show) {
354         $is_dir_show = 1;
355         $is_file_show = 1;
356     }
357 
358     opendir(DIR, '.');
359     my @file = readdir(DIR);
360     closedir(DIR);
361 
362     foreach (@file) {
363         next if $_ eq '.' or $_ eq '..';
364 
365         if (-d $_) {
366             push(@folds, $_);
367 
368         } else {
369             push(@files, $_);
370         }
371     }
372 
373     my $len1 = MaxLength @folds;
374     my $len2 = MaxLength @files;
375     $max = $len1 > $len2 ? $len1 : $len2;
376 
377     if ($is_show_size) {
378         PrintBySize;
379 
380     } else {
381         PrintNoSize;
382     }
383 
384 }
385 
386 sub Test {
387     #$s = "d:\\1.jpg";
388     #my $size = GetFoldSize "d:\\lib";
389     #print ByteToString $size;
390 
391     #print length($str);
392 
393 #    opendir(DIR, "e:\\");
394 #    my @files = readdir(DIR);
395 #    closedir(DIR);
396 
397 #    foreach (@files) {
398 #        print $_, "\n";
399 #    }
400 
401     my $len = -s "D:\\pagefile.sys";
402     print "$len \n";
403     print ByteToString($len) , "\n";
404 }
405 
406 sub Main {
407     if (IsHelp @_) {
408         Help;
409     } else {
410         List @_;
411 #        Test;
412     }
413 }
View Code

将上面的代码保存为ls.pl,放在系统路径中。然后在命令行中就可以用 perl ls.pl 来执行它。如果嫌这个命令太复杂,就想用ls来执行它,可以这样,在系统路径中建立一个ls.bat文件,内容为

@echo off
perl e:\Tools\ls.pl %*

这样输入ls时,先执行ls.bat文件,然后在这个文件中再真正执行perl代码(这里指定ls.pl的路径得用绝对路径)。其中%*是将 ls后面所有的命令行选项传递给ls.pl。

ls所支持的命令行选项有:

  • ls  -f 只显示文件
  • ls -d 只显示文件夹
  • ls -s 显示文件或文件夹的大小
  • ls -S 按文件大小排序后再显示

 

b. grep程序

当我们阅读某个开源项目时往往想找一个类名、或者一个变量名在哪些地方使用过,如果这时没有将项目组织用IDE组织起来,想要查找就比较麻烦。在linux下有个grep可以对文本进行搜索,然后本人就用perl写了一个类似grep的程序,可以在文本文件中查找字符串,并且可以指定在特定的文件上查找。

  1 use 5.010;
  2 use strict;
  3 use warnings;
  4 use Cwd;
  5 
  6 use Win32::Console::ANSI;
  7 use Term::ANSIColor;
  8 
  9 sub Main;
 10 sub MainTest;
 11 
 12 Main @ARGV;
 13 #MainTest;
 14 
 15 #--------------
 16 # return 返回是否只查看一级目录,bool
 17 #--------------
 18 sub OnlyLocal {
 19     foreach (@_) {
 20         if (lc($_) eq '-l') {
 21             return 1;
 22         }
 23     }
 24     return 0;
 25 }
 26 
 27 #--------------
 28 # 获取命令行参数的一般方法
 29 # 用法 :GetPar('-f', @_);
 30 # return 返回第一个参数后面的字符串
 31 #--------------
 32 sub GetPar {
 33 
 34     my $next = 0;
 35     my $str = $_[0];
 36     shift @_;
 37     #say @_;
 38     foreach (@_) {
 39         if ($next) {
 40             return $_;    
 41         }
 42         if ($_ =~ /\Q$str/i) {
 43             $next = 1;
 44         }
 45     }
 46     return '';
 47 }
 48 
 49 #--------------------
 50 # param 输入的命令行
 51 # return 返回 -f后面的值
 52 #--------------------
 53 sub FileExp {
 54 
 55     my $s = &GetPar('-f', @_);
 56     if ($s ne '') {
 57         return $s;    
 58     }    
 59     return '';
 60 }
 61 
 62 #--------------------
 63 # return 返回 -t 后面的值
 64 #--------------------
 65 sub TxtExp {
 66     return &GetPar('-t' ,@_);
 67 }
 68 
 69 my $is_loc;
 70 my $file_exp;
 71 my $txt_exp;
 72 my $count;
 73 my @scan_files;
 74 
 75 #--------------
 76 # 用 $txt_exp查找文件中的每一行,返回所有符合条件的行
 77 #--------------
 78 sub ScanFile {
 79     my $file = shift;
 80     my @text;
 81     
 82     open(FILE, "$file");
 83     while (<FILE>) {
 84         if (/$txt_exp/i) {
 85             #return 1;
 86             push(@text, $_);
 87         }
 88     }
 89     close FILE;
 90     #return 0;
 91     return @text;
 92 }
 93 
 94 sub PrintColor;
 95 #--------------
 96 # 用不同颜色来突出显示匹配到的文本
 97 # par_0@:line_text
 98 #--------------
 99 sub PrintColor {
100     my $text = shift;
101     
102     if ($text =~ /$txt_exp/i) {
103     
104         print $`;
105         
106         print color 'bold red';
107         print $&;
108         print color 'reset';
109         
110         
111         PrintColor $';   #'为了消除$后面的影响
112         
113     } else {
114         print $text;
115     }
116 
117 }
118 
119 #--------------
120 # 显示文件信息
121 # par_0: $file
122 # par_1: @text
123 #--------------
124 sub PrintFile {
125     my $file = shift;
126     
127     print color 'bold cyan'; 
128 
129     print "$count  $file\n";
130     print color 'reset';
131         
132     ++$count;
133     push(@scan_files, $file);
134     
135     #现在一个文件中只显示一条信息
136     
137     $_[0] =~ s/^\s+|\s+$//g; #去掉前后的空格
138     $_[0] =~ s/\s+/ /g; #将多个空格替换成一个空格
139 
140     print "    ";
141     PrintColor $_[0];
142     print color 'reset';
143     print "\n\n";
144     
145 }
146 
147 #--------------
148 # 查看一级目录中的文件
149 # param: $dir
150 #--------------
151 sub ScanFold {
152     
153     my $dir = shift;
154     opendir(DIR, "$dir");
155     my @files = readdir(DIR);
156     closedir(DIR);
157     
158     foreach (@files) {
159         next if $_ eq '.' or $_ eq '..';
160         my $file = "$dir\\$_";
161         
162         if (-d $file) {
163             if (!$is_loc) {
164                 ScanFold($file);
165             }
166             
167         } else {
168             
169             my $file_valid = 0;
170             
171             #如果文件表达式为空,则判断文件是否为文本文件
172             #否则用文件表达式去匹配 $_
173             if ($file_exp eq '') {
174                 $file_valid = -T $file;
175             } else {
176                 $file_valid = /$file_exp/i;
177             }
178         
179              if ($file_valid) {
180                 
181                 if ($txt_exp ne '') {
182                     my @text = &ScanFile($file);
183                     
184                     if ($#text >= 0 ) {  # $#text是最后一个数组元素的index
185                         PrintFile($file, @text);
186                     }
187                     
188                 } else {
189                     print color 'bold cyan';
190                     print "$count $file\n";
191                     print color 'reset';
192                                         
193                     ++$count;
194                     push(@scan_files, $file);
195                 }
196              }
197         }
198         
199     }
200 }
201 
202 sub isnumeric {
203     my $val = shift;
204     ($val ^ $val) eq '0';    
205 }
206 
207 sub Scan {
208 
209     $is_loc = &OnlyLocal(@_);
210     
211     $file_exp = &FileExp(@_);
212     $txt_exp = &TxtExp(@_);
213     
214     $count = 0;
215     
216     if ($file_exp eq '' && $txt_exp eq '' && $#_==0) {
217         $txt_exp = $_[0];
218     }
219     
220     #print "file_exp:", $file_exp, "\n";
221     #print "txt_exp:", $txt_exp, "\n";
222     #print "###============================\n";
223     
224     &ScanFold(".");
225     
226     print "如果想打开文件,则输入对应编号按回车\n";
227     
228     my $num = <STDIN>;
229     chomp $num;
230  
231     return if $num eq "";
232         
233     if ($num>=0 && $num<=$#scan_files) {
234         my $path = getcwd;    #获取当前路径,用于获取文件的完整路径
235         $path =~ s%/%\\%g; #将/替换成\
236         
237         my $file = $scan_files[$num];
238         $file = substr($file, 1);  #去掉$file前面的.
239         my $f = $path . $file;
240          
241         system "OpenInExplorer.exe $f";
242     }    
243     
244 }
245 
246 sub Main {
247     Scan @_;
248 }
grep

同样为了方便,建立一个grep2.bat文件,其内容为(这里只所有取名为grep2,是因为本人在widnows中装了linux的一些软件后有grep程序了,为了不引起冲突而取名grep2)

@echo off
perl e:\Tools\grep.pl %*

grep2 ClassName 查找所有文本文件中的ClassName

grep2 -f file  只查找文件名(不查找内容)

grep2 -f \.cpp$ -t ClassName 在以.cpp结尾的文件中查找ClassName

 

c. 从开源代码构建vs项目

分析linux下的开源项目时往往没有IDE的支持,这里想要定位到变量的定义处,或者想查找在哪些地方有使用这个变量就很不方便。于是可以建立一个不能编译的vs项目,将所有源码添加到vs后,仅仅用来浏览代码。但基本上开源项目的结构都非常复杂,子文件夹中包括子文件夹,而vs也只支持批量添加同一个文件夹中的所有文件。如果一个个文件夹手动去添加代码文件,是个非常重复的工作。——于是程序的天性,消除重复劳动在此得到体现

如果是建立c/c++项目,其项目拓展名是.vcproj,可以发现它就是一个xml文件。

其包括文件的关键节点如下:

<Files>
<Filter Name="app">
<File ReleativePath=".\app\about.h">
<File Releativepath=".\app\cpp.c">
</Filter>
<Files>

知道了这个结构后,就可以用程序自动生成了~~

文本操作,本人依然用Perl !!

 1 #
 2 #使用方法:将些文件复制到要建立VS项目的文件夹内,用perl create_vs.pl运行此文件。
 3 #           在运行前先vs_pre.txt复制到与create_vs.pl同目录下,这样就可以生成直接生成vcproj文件
 4 #
 5 
 6 use Win32::OLE;
 7 use warnings;
 8 use 5.010;
 9 
10 use XML::Simple;
11 use XML::DOM;
12 
13 #use XML::Tidy;
14 use strict;
15 use Encode;
16 
17 sub addFile;
18 
19 my $parser = new XML::DOM::Parser;
20 my $doc    = $parser->parse('<Files/>');
21 
22 #addFile( 2, 4 );
23 addFile( $doc->getDocumentElement(), '.' );
24 $doc->printToFile('file.xml');
25 $doc->dispose;
26 
27 open file_out, ">temp.vcproj";
28 open file_in1, "vs_pre.txt";
29 open file_in2, "file.xml";
30 
31 while(<file_in1>) {
32     chomp($_);
33     say file_out $_;
34 }
35 while(<file_in2>) {
36     chomp($_);
37     say file_out $_;
38 }
39 my $end ="     <Globals>
40     </Globals>
41 </VisualStudioProject>";
42 say file_out $end;
43 
44 
45 #下面的格式正则代码有问题
46 #my $tidy = XML::Tidy->new( 'filename' => 'file.xml' );
47 #$tidy->tidy();
48 #$tidy->write();
49 
50 sub addFile {
51     my $node = shift;
52     my $path = shift;
53 
54     opendir( DIR, "$path" );
55     my @file = readdir(DIR);
56     closedir(DIR);
57     foreach (@file) {
58         next if $_ eq '.' or $_ eq '..';
59         my $file = "$path\\$_";
60         if ( -d $file ) {
61 
62             #print $file,"\n";
63             my $n = $doc->createElement('Filter');
64             my $att = $doc->createAttribute( 'Name', $_ );
65             $n->setAttributeNode($att);
66             $node->appendChild($n);
67             addFile( $n, $file );
68         }
69         else {
70             my $s = substr( $file, rindex( $file, "." ) );
71             print $s,"\n";
72             #只添加h,c文件
73             if ( $s ~~ /[h|hpp|cpp]/) {
74                 my $n = $doc->createElement('File');
75                 my $att = $doc->createAttribute( 'RelativePath', $file );
76                 $n->setAttributeNode($att);
77                 $node->appendChild($n);
78             }
79 
80         }
81     }
82     #如果这个节点上没有子节点,则将其去掉
83     if ( !$node->hasChildNodes() ) {
84         $node->getParentNode()->removeChild($node);
85     }
86 }
create.pl
<?xml version="1.0" encoding="gb2312"?>
<VisualStudioProject
    ProjectType="Visual C++"
    Version="9.00"
    Name="coreutils"
    ProjectGUID="{D5524811-7F7F-49A7-80E0-85645B1D2B68}"
    RootNamespace="coreutils"
    Keyword="Win32Proj"
    TargetFrameworkVersion="196613"
    >
    <Platforms>
        <Platform
            Name="Win32"
        />
    </Platforms>
    <ToolFiles>
    </ToolFiles>
    <Configurations>
        <Configuration
            Name="Debug|Win32"
            OutputDirectory="$(SolutionDir)$(ConfigurationName)"
            IntermediateDirectory="$(ConfigurationName)"
            ConfigurationType="1"
            CharacterSet="1"
            >
            <Tool
                Name="VCPreBuildEventTool"
            />
            <Tool
                Name="VCCustomBuildTool"
            />
            <Tool
                Name="VCXMLDataGeneratorTool"
            />
            <Tool
                Name="VCWebServiceProxyGeneratorTool"
            />
            <Tool
                Name="VCMIDLTool"
            />
            <Tool
                Name="VCCLCompilerTool"
                Optimization="0"
                PreprocessorDefinitions="WIN32;_DEBUG;_CONSOLE"
                MinimalRebuild="true"
                BasicRuntimeChecks="3"
                RuntimeLibrary="3"
                UsePrecompiledHeader="2"
                WarningLevel="3"
                DebugInformationFormat="4"
            />
            <Tool
                Name="VCManagedResourceCompilerTool"
            />
            <Tool
                Name="VCResourceCompilerTool"
            />
            <Tool
                Name="VCPreLinkEventTool"
            />
            <Tool
                Name="VCLinkerTool"
                LinkIncremental="2"
                GenerateDebugInformation="true"
                SubSystem="1"
                TargetMachine="1"
            />
            <Tool
                Name="VCALinkTool"
            />
            <Tool
                Name="VCManifestTool"
            />
            <Tool
                Name="VCXDCMakeTool"
            />
            <Tool
                Name="VCBscMakeTool"
            />
            <Tool
                Name="VCFxCopTool"
            />
            <Tool
                Name="VCAppVerifierTool"
            />
            <Tool
                Name="VCPostBuildEventTool"
            />
        </Configuration>
        <Configuration
            Name="Release|Win32"
            OutputDirectory="$(SolutionDir)$(ConfigurationName)"
            IntermediateDirectory="$(ConfigurationName)"
            ConfigurationType="1"
            CharacterSet="1"
            WholeProgramOptimization="1"
            >
            <Tool
                Name="VCPreBuildEventTool"
            />
            <Tool
                Name="VCCustomBuildTool"
            />
            <Tool
                Name="VCXMLDataGeneratorTool"
            />
            <Tool
                Name="VCWebServiceProxyGeneratorTool"
            />
            <Tool
                Name="VCMIDLTool"
            />
            <Tool
                Name="VCCLCompilerTool"
                Optimization="2"
                EnableIntrinsicFunctions="true"
                PreprocessorDefinitions="WIN32;NDEBUG;_CONSOLE"
                RuntimeLibrary="2"
                EnableFunctionLevelLinking="true"
                UsePrecompiledHeader="2"
                WarningLevel="3"
                DebugInformationFormat="3"
            />
            <Tool
                Name="VCManagedResourceCompilerTool"
            />
            <Tool
                Name="VCResourceCompilerTool"
            />
            <Tool
                Name="VCPreLinkEventTool"
            />
            <Tool
                Name="VCLinkerTool"
                LinkIncremental="1"
                GenerateDebugInformation="true"
                SubSystem="1"
                OptimizeReferences="2"
                EnableCOMDATFolding="2"
                TargetMachine="1"
            />
            <Tool
                Name="VCALinkTool"
            />
            <Tool
                Name="VCManifestTool"
            />
            <Tool
                Name="VCXDCMakeTool"
            />
            <Tool
                Name="VCBscMakeTool"
            />
            <Tool
                Name="VCFxCopTool"
            />
            <Tool
                Name="VCAppVerifierTool"
            />
            <Tool
                Name="VCPostBuildEventTool"
            />
        </Configuration>
    </Configurations>
    <References>
    </References>
vs_pre.txt

将create.pl vs_pre.txt复制到需要构建项目的地方,然后执行perl create.pl,这样就会生成一个temp.vcproj的项目,其中就包括了这个目录下所有的.h, .hpp, .cpp文件,在create.pl 的73行可以指定添加文件的类型。

 之前本人分析gimp和ffmpeg的源码就是用这种方法建立了vs项目,于是极大地提高了分析代码的效率

如果以后有心情了,分析linux系统源码时也会用这种方式来构建项目。

 d. git项目的批量pull, push

自从爱上了git.oschina.net后,将所有代码都放到git的私有项目中。c++代码都放在本地的 e:\cpp_app\ 文件夹下

如图所示,左下角有小绿圆圈的标志都是有提交git的项目。

这么多项目,如果一个个手动地进入文件夹然后pull,push,非常繁琐。

于是先建立下面的git_syn.pl文件

 1 my $one = $ARGV[0];
 2 my $path = "e:\\cpp_app\\";
 3 opendir(DIR, $path);
 4 my @files = readdir(DIR);
 5 
 6 foreach (@files) {
 7     next if $_ eq '.' or $_ eq '..';
 8         
 9     my $str = $path . $_ . "\\" . ".git";
10     if (-d $str) {
11         my $s = $path . $_;
12 
13         say "\n\n=============== now directory : $s";
14         chdir $s;
15 
16         if ($one == 0) {
17             system "git add .";
18             system "git commit -a";
19             system "git push";
20         } elsif ($one == 1) {
21             system "git pull";
22         }
23     }
24 }

 

建立pull.bat文件

1 echo off
2 perl git_syn.pl 1
3 pause

 

建立push.bat文件

1 echo off
2 perl git_syn.pl 0
3 pause

 git_syn.pl遍历e:\cpp_app文件夹下的所有文件夹,如果发现有.git子文件夹,则根据传入参数判断是push还是pull操作。

有了pull.bat, push.bat后,妈妈再也不会担心有代码没更新了。使用svn的项目也这样弄,只要将git 改成svn,pull,push换成update,commit即可。参见《命令行&界面

5. 正则表达式

个人认为程序员应该将正则表达式融入自己的血液

如果不沉下心来学习,或者学了后不用,都会出现想使用正则表达式而很茫然的情况。

本人借着学习Perl的机会,将正则表达式重新好好学了一遍,并且在日常的工作中有意识地强迫自己去使用,几个月下来后基本上可以称得上熟练掌握了正则表达式。

下面就介绍几个在vs中用正则表达式进行查找/替换的例子(vs的正则和标准的正则有所不同,不过思想是一样的,只是在表示符号上不所不同):

用如下图所示的方法打开vs的正则表达式查找、替换功能:

  • 查找被注释掉的代码   .*//.*GetTickCount
  • 将for,if, while后面的 { 放在与其同一行。查找部分为 ^{:b*[for|if|while].*[^\{]}\n:b*\{  替换为 \1{
  • 给所有逗号后面都加上一个空格。查找部分为,{[^ ]},替换为, \1

 其中{}表示分组(相当于标准正则中的()),[]表示一类字符集(相当于标准正则中的{})

关于正则表达式的具体使用情况还有很多,结合上面介绍的grep2,可以用\.cpp$来查找所有后缀是cpp的文件。

总之,学习正则表达式的回报会随着时间的累积越来越多,随着对其掌握的熟练程序而越来越多。