用sed实现wc -w的功能
版权申明:本文为博主窗户(Colin Cai)原创,欢迎转帖。如要转贴,必须注明原文网址 http://www.cnblogs.com/Colin-Cai/p/7663831.html 作者:窗户 QQ:6679072 E-mail:6679072@qq.com
sed是所谓的流编辑器,我们经常用它来做一些文本替换的事情,这是sed最擅长的事情,如sed 's/Bob/Tom/g'就是把文章中所有的Bob改成Tom。
sed是图灵完备的,作为sed的粉丝,喜欢用sed做各种sed不擅长的事情,这里实现一下wc -w的功能,也就是统计文章单词数量。
我习惯喜欢加上n和r,n表示每行结束时不会自动打印,r表示正则表达式的扩展方式,我实在很讨厌写那么多\,所以sed基本上我是一定加这两个东西的。
先从sed擅长的开始,先用s命令做替换,把每个单词都替换为单个1。这一步其实很简单,s/[^ \t\r]+/1/g即可,也就是把不是空格的连续匹配替换为1,g是表示对一行中所有满足这样的模式都替换为1,再考虑到正则表达式的贪婪,其实我们的[^ \t\r]+实际上就是指完整的一个单词,熟悉regex替换的应该不难理解。
然后为了整齐,替换为1之后,再把空格都去掉,其实也就是把不是1的去掉,那么紧接着一条s/[^1]+//g即可,然后再用p打印一下。
一口吃不成胖子,先从简单的来,我们可以看一下效果。在此之前先找篇文章,就节选一下google的pixel buds新闻吧。
linux-p94b:/tmp/testhere # cat 1.txt
American company Google recently announced the release of its Google Pixel 2 phone and other products that work together with the phone.
One of the new products is a pair of wireless earphones Google calls Pixel Buds.
The earphones are seen as the company's answer to competitor Apple's popular AirPod headphones.
At a launch event on October 4, Google said its Pixel Buds were built to provide high-quality sound and hands-free use. All of their operations can be controlled by simply touching the right earphone.
Once the headphones are paired with a Pixel phone, its many features can be used through the Pixel Buds.
One example is Google Assistant, the company's artificial intelligence, or AI, service. Users can now talk directly to Pixel Buds to ask Google Assistant questions, get information or other help. This can all be done without touching the telephone.
The Pixel Buds also can work with Google Translate, the service that provides words and expressions in over 100 languages.
Google product manager Juston Payne demonstrated this feature during the launch event. He was able to talk with someone whose native language is Swedish.
When the person spoke Swedish into the Pixel Buds, the phone's speakers provided the translation in English. The English speaker's response was then translated in real time into Swedish and heard through the Pixel Buds.
linux-p94b:/tmp/testhere # cat wc-w.sed
#!/usr/bin/sed -nrf
s/[^ \t\r]+/1/g
s/[^1]+//g
p
linux-p94b:/tmp/testhere # ./wc-w.sed <1.txt
1111111111111111111111
111111111111111
11111111111111
1111111111111111111111111111111111
1111111111111111111
111111111111111111111111111111111111111
11111111111111111111
111111111111111111111111
11111111111111111111111111111111111
对一下,确实没有错,只是出来了一堆1,而且还是分行的,那么第二步,把这个分行给去掉。当然,加个管道,tr -d '\n'就去掉了,不过我们要的是单个sed解决,那么需要再动一点点脑筋。
我们可以在上面的基础上稍微改动改动,把这些1先缓存进保持空间(hold space),最后再从保持空间中取出,然后用s/\n//g去掉所有的回车符,再打印。
linux-p94b:/tmp/testhere # cat wc-w.sed
#!/usr/bin/sed -nrf
s/[^ \t\r]+/1/g
s/[^1]+//g
H
$ {
g
s/\n//g
p
}
linux-p94b:/tmp/testhere # ./wc-w.sed <1.txt
111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111
H命令就是放在保持空间的最后,$是判定输入结束,g是用保持空间的内容替换模式空间。
上面打印出了222个1,离结果222已经很近了。
最后就是如何整合成222了,这里的确是需要一点点技巧了。我们建立以下计数方法:
1..1;1..1;1..1...
每一堆1的个数假设为nk,nk-1,...,n0
允许数量为0的堆
每一堆1之间用分号隔开,如果看到有多个分号在一起,那么中间实际上有数量为0的堆
整个计数表示的是nk*10k+nk-1*10k-1+...+n0
很明显,我们十进制表示方法和整个很类似,只是,十进制表示里,每一堆都小于10而已。
于是我们可以创立一个算法,也就是,当我们发现一堆里有10个1,那么我们就可以往高位进1。
很容易证明这个算法可以结束。
假设{nk,nk-1,...,n0}有限序列是非负整数num的一个表示,序列里的每一个数字是一个非负整数,最高位nk大于0,除非num等于0。
显然,一个具体整数的表示方法是有限的,实际上,这个k不可能大于num对10取对数,序列中的每一项不可能大于num。
序列可以比较大小,
{mj,mj-1,...,m0}有限序列是num的另外一个表示,那么
{nk,nk-1,...,n0} 〉{mj,mj-1,...,m0} 当且仅当 k > j 或者 k = j且nk=mk...nk-p=mk-p,nk-p-1>mk-p-1
以上比较大小的方法可以把一个非负整数的所有表示串成一个全序集。
之前的算法中,每当升位,其表示都会变的比之前大。因为所有的表示为有限个,而最大的表示则是十进制的表示方法,从而可以知道算法是可以结束得到十进制表示的。
那么我们根据这个,不停的找10个0,每当找到,就进位,最后再把每堆挨个替换为9,8,7,6,5,4,3,2,0,再去掉分号,就完成了。有点费脑子吧,我实现一下如下:
linux-p94b:/tmp/testhere # cat wc-w.sed
#!/usr/bin/sed -nrf
s/[^ \t\r]+/1/g
s/[^1]+//g
H
$ {
g
s/\n//g
:a
s/;1111111111/1;/
s/^1111111111/1;/
ta
s/111111111/9/g
s/11111111/8/g
s/1111111/7/g
s/111111/6/g
s/11111/5/g
s/1111/4/g
s/111/3/g
s/11/2/g
:b
s/;;/;0;/g
tb
s/;$/;0/
s/;//g
/^$/s/^/0/
p
}
linux-p94b:/tmp/testhere # ./wc-w.sed <1.txt
222
用sed实现wc -w的功能的更多相关文章
- 用sed实现wc -c的功能
sed是所谓的流编辑器,我们经常用它来做一些文本替换的事情,这是sed最擅长的事情,如sed 's/Bob/Tom/g'就是把文章中所有的Bob改成Tom. sed是图灵完备的,作为sed的粉丝,喜欢 ...
- 模仿WC.exe的功能实现--node.js
Github项目地址:https://github.com/102derLinmenmin/myWc WC 项目要求 wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和行数.这个项目要 ...
- WC的基本功能实现.(Java)
我的GitHub地址:https://github.com/Yuetao1219/lessons WC 项目要求 wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和行数.这个项目要求写 ...
- sed - 文本三剑客之编辑功能
sed - stream editor for filtering and transforming text Sed是一个流编辑器.流编辑器用于对输入流(文件或管道输入)执行基本的文本转换.虽然在某 ...
- 软件工程:java实现wc项目基本功能
项目相关要求 项目地址:https://github.com/xiawork/wcwork 实现一个统计程序,它能正确统计程序文件中的字符数.单词数.行数,以及还具备其他扩展功能,并能够快速地处理多个 ...
- 软件工程:Java实现WC.exe基本功能
项目相关要求 GitHub地址:https://github.com/3216004716/WC 实现一个统计程序,它能正确统计程序文件中的字符数.单词数.行数,以及还具备其他扩展功能,并能够快速地处 ...
- Sed 与 Linux 等价命令代码鉴赏(转)
参考了 http://www.chinaunix.net/jh/24/307045.html sed http://bbs.chinauni ...
- 文本处理命令--wc、sed
一.wc wc命令的功能为统计指定文件中的字节数.字数.行数,并将统计结果显示输出. 测试文件内容: (my_python_env)[root@hadoop26 ~]# cat test hnlinu ...
- 文本处理命令(sort+uniq+cut+tr+wc)+三剑客之sed
目录 文本处理命令+三剑客之sed 一.文本处理命令 1.排序命令 sort 2.检查/删除命令 uniq 3. cut 显示特定部分命令 4. 替换或删除命令 tr 5.统计 计算数字命令 wc 二 ...
随机推荐
- Java核心技术 对象与类
对象与对象变量: 要想使用对象,就必须首先构造对象,并指定其初始状态,然后,对对象应用方法. 在Java中,使用构造器构造新实例.构造器是一种特殊的方法,用来构造并初始化对象. 在实际开发中,通常需要 ...
- CNN大战验证码
介绍 爬虫江湖,风云再起.自从有了爬虫,也就有了反爬虫:自从有了反爬虫,也就有了反反爬虫. 反爬虫界的一大利器,就是验证码(CAPTCHA),各种各样的验证码让人眼花缭乱,也让很多人在爬虫的过 ...
- C#泛型。
作用: 使用泛型可以实现算法重用. class Program { static void Main(string[] args) { MyClass<string> myClass = ...
- 一个小时学会Maven
一.为什么要Maven 在开发中经常需要依赖第三方的包,包与包之间存在依赖关系,版本间还有兼容性问题,有时还里要将旧的包升级或降级,当项目复杂到一定程度时包管理变得非常重要. Maven是当前最受欢迎 ...
- Prism 学习:从配置文件中加载 Module
之前我们已经了解过,如何从指定的目录中来加载 Module(原文),现在我们来看,如何从应用程序的配置文件中来加载 Module.以这种方式来加载 Module 的优点包括:1. 被加载的 Modul ...
- struts2_Action的三种实现方式
1.普通java类 package com.ahd.action; public class HelloAction{ public String execute() throws Exception ...
- virtualbox中 清理磁盘
1. 碎片整理 windows: 下载 sdelete 工具 执行命令: sdelete –z c:\ Linux: 执行如下命令: sudo dd if=/dev/zero of=/EMPTY bs ...
- JavaScript如何正确处理Unicode编码问题!
原文:JavaScript 如何正确处理 Unicode 编码问题! 作者:前端小智 Fundebug经授权转载,版权归原作者所有. JavaScript 处理 Unicode 的方式至少可以说是令人 ...
- 现在有两个变量,分别是a = 3, b = 4,那么我们不用第三个变量来调换a和b的值。
现在有两个变量,分别是a = 3, b = 4,那么我们不用第三个变量来调换a和b的值. <!DOCTYPE html><html><head> <me ...
- 亲测:LNMP环境下,解决项目缓冲慢、502以及配置https的问题
在做的项目在nginx下访问缓冲时间过长,明显比apache下访问蛮11倍有余, 解决办法: 1增加nginx的upstream,其中upstream中为php-cgi的地址: 2利用nginx作为反 ...