Like spacemacs, but for vim.
SpaceVim 放在 github 上, 3 天获取 700+ 星,该项目不仅适合 vim 以及 neovim 老用户, 对于新用户来说,也是非常适合的。
@Shougo 大神 以及 @mhinz 大神对此项目也很支持, 另外,该项目在 Hacker News 首页也出现过。拥有 200+ points 。
在 gittter #neovim 内, 也有用户支持该项目。
目前该项目 80% 收藏者来自国外,国内确实没有找到很好的 post 平台, 现尝试在 V2EX 上发布下, 希望有更多的 vim 用户支持,让 vim 中文社区更加活跃。
该项目地址 github 地址是 : SpaceVim/SpaceVim
官网 : http://spacevim.org
SpaceVim 是高度可定制的模块化 Vim 配置集合,适合开发各种语言, 尤其是 Java c c++ python php 等常用语言,并且拥有非常优秀的 UI 界面, 用户可以根据自己需求载入需要的模块。如果有兴趣,欢迎进群大家一起讨论。 本群为非常专业的技术讨论群,请勿水!
在装了 arch 之后,没有庄 gnome 等桌面,而是采用了 I3-WM 这样的平铺桌面, 平时有聊天的需求,又无法忍受手机打字那么麻烦, 于是做了一个 Linux 系统下 借助 Vim 的聊天工具。分享给大家:
call dein#add('wsdjeg/vim-chat')
安装完成之后,可以使用 call chat#chatting#OpenMsgWin(),打开聊天窗口。
cpanm -v Mojo::IRC::Server::Chinese, 详见: irc.md首先启动 QQ 服务器 call chat#qq#start(), 然后会自动弹出一个二维码,手机扫描下就可以登录了。neovim 默认使用 Alt + x 打开/关闭聊天窗口。
其实关于这个问题,至今我也没有完全想明白。可能是因为上了年纪的原因,有些怀旧,但是发现过往的记忆似乎很难完全想起来了。 因此,我觉得有必要日常做一些简单的记录。
其实早在 2004 年 QQ 出现的时候,就开始写 QQ 空间日志,再到后来的新浪博客、网易博客、百度空间、博客园、CSDN 等等, 这些平台都有使用过。但是都存在各种各样的问题,包括:
独立站点的优势在于:
邮件更加私密,也显得更加正式。同时,也可以避免广告、spam 等内容直接在评论区显示。
之前用过的社交平台软件,各种各样问题都有。以至于有很长一段时间,很想彻底摆脱这些社交软件。
Reddit 账号已删除,留张图片纪念一下:

这是一个值得记忆的日子,今天我们家迎来了一位新的小生命。因为自己的工作原因,在老婆最辛苦的这段日子里, 没能够时常陪伴身边。
《华严经》里说:“不忘初心,方得始终。”
记得很长时段时间,签名是:感叹那两个少年,一个温柔了岁月,一个经验了流年。 已经无从考证这句话到底来自哪里了。当时第一次看到这段话就觉得深受感触。
每个人,在最初的时候可能会有各种各样的梦想。但是随着时间的推移,生活的磨砺。 会让我们越来越执着于眼前,淡忘了曾经的梦想。梦想是每个人出发的起点,不忘初心, 才会让自己明确从哪里来,到哪里去。
初心,给人强大的勇气,把一切困难都踩在脚下。人生最大的困厄,就是被人驱使。 被父母催促着去上学,被领导吩咐着去做事,被生活逼迫着去赚钱。 看似是出于无奈,实则缺少了情愿,总像背负着巨石,压得喘不过气来。 转换一下思维,调整一下状态,从自己的初心开始,想自己想做的事, 想着自己该肩负的责任,不后悔,肯坚持,虽难免生长的阵痛,却会得到成长的快乐。
初心,给了人积极进取的人生状态,随时能回到最初的原点,重新开始。 人生的经历,在我们大脑里塞满了各种各样的东西。 有时我们满足于鲜花和掌声,喜欢在美好的过去里追忆; 有时我们则沉浸于挫折的苦痛,在灰暗的心空里踽踽独行。 其实我们最需要的,就是不忘初心,倾空自己,回到充满渴望的婴儿般的状态, 永远在拼命地汲取,永远在奋力前行的路上。
在这个时代,初心常常被我们遗忘,正如纪伯伦所说:“我们已经走的太远,以至于忘记了为什么出发。” 人生只有一次,生命无法重来,要记得自己的初心。 经常回头望一下自己的来路,回忆起当初为什么起程; 经常让自己回到起点,给自己鼓足从头开始的勇气; 经常纯净自己的内心,给自己一双澄澈的眼睛, 不忘初心,才会找对人生的方向,才会坚定我们的追求, 抵达自己的初衷,也更容易抵达人生的目的地。
:global 和 :substitute 命令是 Vim 最强大的命令之一,
将其摸透用熟可以事半功倍,在这里我总结了一些网上的网上的经典问题,
结合自己的使用和理解,通过实例详细介绍一下其用法。示例难度不一,
其中有些可能并没有多少实用性,仅仅是为了展示功能。
阅读 :help :g可知,该命令的使用形式为:
:[range]global/{pattern}/{command}
global 命令在 [range] 指定的文本范围内(缺省为整个文件)查找 {pattern},
然后对匹配到的行执行命令{command},如果希望对没匹配上的行执行命令,
则使用 global! 或 vglobal 命令。
先来看 Vim 用户手册里的一个经典例子。
【例1】倒序文件行(即unix下的tac命令)
:g/^/m 0
这条命令用行首标记 ^ 匹配文件的所有行(这是查找的一个常用技巧,如果用 .则是匹配非空行,不满足本例要求),
然后用 move 命令依次将每行移到第一行(第0行的下一行),从而实现了倒序功能。
global命令实际上是分成两步执行:首先扫描[range]指定范围内的所有行,
给匹配{pattern}的行打上标记;然后依次对打有标记的行执行{command}命令,
如果被标记的行在对之前匹配行的命令操作中被删除、移动或合并,则其标记自动消失,而不对该行执行{command}命令。
标记的概念很重要,以例说明。
【例2】删除偶数行
:g/^/+1 d
这条命令也是匹配所有行,然后隔行删除(其中+1用以定位于当前行的下一行)。 为什么是隔行呢?因为在对第一行执行+1 d命令时删除的是第二行,而第二行虽然也被标记了,但已不存在了, 因此不会执行删除第三行的命令。
本例也可以用:normal命令实现:
:%normal! jdd
% 指定整个文件,然后依次执行普通模式下的 jdd,即下移删除一行。与 global 命令不同之处在于,
%normal! jdd 是按照行号顺序执行,在第一行时删除了第二行,后面的所有行号都减一,
因此在第二行执行 jdd 时删除的是原来的第四行。也就是说,global 命令是通过偶数行标记的消失实现的,
而 normal 命令是通过后续行的自动前移实现的。
【例3】删除奇数行
:g/^/d|m.
光是:g/^/d显然不行,这会删除所有行,我们需要用move命令把偶数行的标记去掉。当然,本例可以很简单的转换成【例2】,在此只是用来强调标记的概念。
本例若想用 normal 命令实现比较有意思,%normal! dd也同样会删除整个文件,%normal! jkdd就可以,我不知道两者为什么不同,可能和normal命令内部的运行机制有关。
不少人觉得这两个命令差不多,的确,它们的形式很相似,都是要进行查找匹配,
只不过 :substitute 执行的是替换而 :global 执行的其它命令(当然,substitute 缺省的 [range] 是当前行,这点也不同)。
先看两个例子,体会一下:s和:g不同的思维方式。
【例4】double所有行
:%s/.*/&\r&/
:g/^/t.
substitue是查找任意行,然后替换为两行夹回车;global是将每一行复制(:t就是:copy)到自己下面,更加清晰明了。
【例5】把以回车排版、以空行分段的文本变成以回车分段的文本
很多txt格式的ebook,以及像vim help这样的文本,每行的字符数受限,段之间用空行分隔。若把它们拷贝到word里,那些硬回车和空行就比较讨厌了,虽然word里也有自动调整格式的功能,不过在Vim里搞定更是小菜一碟。先看看用替换如何实现。
:%s/\n\n\@!//
\n\n\@!是查找后面不跟回车的回车(关于\@!的用法请:h /\@!,在此不多说了),然后替换为空,也就是去掉用于排版的回车。
注:如果等宽段之间无空行,则合并命令变为:%s/\n\%(\s\{2,}\|\n\)\@!//g
global命令则完全是另一种思路。
:g/./,/^$/j
/./标记非空行,/^$/查找其后的空行,然后对二者之间的行进行合并操作。也许有人会问,段中的每一行会不会都执行了j命令?前面已经说过,在之前操作中消失掉的标记行不执行操作命令,在处理每段第一行时已经把段内的其余行都合并了,所以每段只会执行一次j命令。这条命令使用global标记做为[range]的起始行,这样的用法后面还会详述。
global经常与substitute组合使用,用前者定位满足一定条件的行,用后者在这些行中进行查找替换。如:
【例6】将aaa替换成bbb,除非该行中有ccc或者ddd
:v/ccc\|ddd/s/aaa/bbb/g
【例7】将aaa替换成bbb,条件是该行中有ccc但不能有ddd
如何写出一个匹配aaa并满足行内有ccc但不能有ddd的正则表达式?我不知道。即便能写出来,也必定极其复杂。用global命令则并不困难:
:g/ccc/if getline('.') !~ 'ddd' | s/aaa/bbb/g
该命令首先标记匹配ccc的行,然后执行if命令(if也是ex命令!),getline函数取得当前行,然后判断是否匹配ddd,如果不匹配(!~的求值为true)则执行替换。要掌握这样的用法需要对ex命令、Vim函数和表达式有一定了解才行,实际上,这条命令已经是一个快捷版的脚本了。可能有人会想,把g和v连起来用不就行了么,可惜global命令不支持(恐怕也没法支持)嵌套。
:h range
在global命令第一步中所设的标记,可以被用来为{command}命令设定各种形式的[range]。在【例2】和【例5】中都已使用了这一技巧,灵活使用[range],是一项重要的基本功。先看看【例2】和【例3】的一般化问题。
【例8】每n行中,删除前/后m行(例如,每10行删除前/后3行)
:g/^/,+2 d | ,+6 m -1
:g/^/,+6 m -1 | +1,+3 d
这两个命令还是利用move来清除保留行的标志,需要注意的是执行第二个命令时的当前行是第一个命令寻址并执行后的位置。再看两个更实用点的例子。
感谢 gqqnb2005 提醒,C++ preprocessor 已經是個很複雜很强大的語言了,下面的两个示例仅供参考理解
:g命令配合/命令搜索确定range, 实际上,语法级别的range确定,使用正则表达式是完全不够的。
【例9】提取条件编译内容。例如,在一个多平台的C程序里有大量的条件编译代码:
#ifdef WIN32
XXX1
XXX2
#endif
...
#ifdef WIN32
XXX3
XXX4
#else
YYY1
YYY2
#endif
现在用global命令把Win32平台下代码提取出来,拷贝到文件末:
:g/#ifdef WIN32/+1,/#else\|#endif/-1 t $
t命令的[range]是由逗号分隔,起始行是/#ifdef WIN32/标记行的下一行,结束行是一个查找定位,是在起始行后面出现的#endif或#else的上一行,t将二者间的内容复制到末尾。
【例10】提取上述C程序中的非Win32平台的代码(YYY部分)
首先说明一下,这个例子比前例要复杂的多,主要涉及的是[range]的操作,已经和global命令没多少关系,大可不看。加到这的目的是把问题说完,供喜欢细抠的朋友参考。本例的复杂性在于:首先,不能简单的用#else和#endif定位,因为代码中可能有其它的条件编译,我们必须要将查找范围限定在#ifdef WIN32的block中;另外,在block中可能并没有#else部分,这会给定位带来很大麻烦。解决方法是:
:try | g/#ifdef WIN32//#else/+1, /#endif/-1 t $ | endtry
先不管try和endtry,只看中间的global部分:找到WIN32,再向后找到#else,将其下一行作为[range]的起始行,然后从当前的光标(WIN32所在行,而非刚找到的#else的下一行)向下找到#endif,将其上一行作为[range]的结束行,然后执行t命令。但对于没有#else的block,如第一段代码,[range]的起始行是YYY1,而结束行是XXX2(因为查找#endif时是从第一行开始的,而不是从YYY1开始),这是一个非法的[range],会引起exception,如果不放在try里面global命令就会立刻停止。
与逗号(,)不同,如果[range]是用分号(;)分隔的,则会使得当前光标移至起始行,在查找#endif时是从#else的下一行开始,这样就产生非法[range],用不着try,但带来的问题是:没有#else的block会错误的把后面block中的#else部分找出来。