用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 二 ...
随机推荐
- 18.Class 的基本语法
Class 的基本语法 Class 的基本语法 简介 JavaScript 语言中,生成实例对象的传统方法是通过构造函数.下面是一个例子. function Point(x, y) { this.x ...
- JavaScript Date 对象的异常现象-new Date('0001-01-01 00:00:00')
Date 对象 Date 对象用于处理日期和时间. new Date() :Date 对象会自动把当前日期和时间保存为其初始值. 打开chrome的开发者工具,在Console敲下new Date() ...
- 注册asp.net 4.0版本到IIS服务器中
在IIS服务器的运维的过程中,有时候部署asp.net网站发现未安装.net framework对应版本信息,此时就需要重新将.net framework对应的版本注册到IIS中,此处以重新注册.ne ...
- windows 下 nginx 配置文件路径
nginx在windowns下路径 http{ #虚拟主机1 server{ listen 80; #监听端口,基于IP配置的时候变更此处,比如192.168.1.100:8080; server_n ...
- c#连接oracle数据库 DBHelper
闲着没事自己写了一个OracleHelper类,希望大神给点建议优化 using System; using System.Collections.Generic; using System.Linq ...
- [PHP] PHP闭包(closures)
1.闭包函数也叫匿名函数,一个没有指定名称的函数,一般会用在回调部分 2.闭包作为回调的基本使用, echo preg_replace_callback('~-([a-z])~', function ...
- [android] 服务的生命周期(混合方式)
绑定服务:可以调用服务里面的方法, 如果调用者activity销毁了,服务也会跟着销毁 单独解除绑定的时候,服务也会被销毁 开启服务:不可以调用服务里面的方法 如果调用者activity退出了,服务还 ...
- laravel的时间日期处理包Carbon用法
时间日期处理包--Carbon Carbon – 是继承自 PHP DateTime 类的 API 扩展,它使得处理日期和时间更加简单.Laravel 中默认使用的时间处理类就是 Carbon. La ...
- 7.QT-Qt对象间的父子关系
Qt对象之间可以存在父子关系 继承于QObject类或者其子类的对象,都称为Qt对象 当指定Qt对象的父对象时 需要通过setParent()成员函数来设置对象间的父子关系 子对象将会把自己的指针地址 ...
- python基础学习(二)注释和算术运算符
注释 1. 注释的作用 注释就是对某些代码进行标注说明,以增强代码的可读性.我们在写程序的时候,编写的某一部分代码的意图不太明显,这时候就需要对这一部分代码加以说明,来明确这一部分到的意图.一般的编程 ...