版权申明:本文为博主窗户(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 -c的功能的更多相关文章

  1. 用sed实现wc -w的功能

    版权申明:本文为博主窗户(Colin Cai)原创,欢迎转帖.如要转贴,必须注明原文网址 http://www.cnblogs.com/Colin-Cai/p/7663831.html 作者:窗户 Q ...

  2. 模仿WC.exe的功能实现--node.js

    Github项目地址:https://github.com/102derLinmenmin/myWc WC 项目要求 wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和行数.这个项目要 ...

  3. WC的基本功能实现.(Java)

    我的GitHub地址:https://github.com/Yuetao1219/lessons WC 项目要求 wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和行数.这个项目要求写 ...

  4. 软件工程:java实现wc项目基本功能

    项目相关要求 项目地址:https://github.com/xiawork/wcwork 实现一个统计程序,它能正确统计程序文件中的字符数.单词数.行数,以及还具备其他扩展功能,并能够快速地处理多个 ...

  5. 软件工程:Java实现WC.exe基本功能

    项目相关要求 GitHub地址:https://github.com/3216004716/WC 实现一个统计程序,它能正确统计程序文件中的字符数.单词数.行数,以及还具备其他扩展功能,并能够快速地处 ...

  6. sed - 文本三剑客之编辑功能

    sed - stream editor for filtering and transforming text Sed是一个流编辑器.流编辑器用于对输入流(文件或管道输入)执行基本的文本转换.虽然在某 ...

  7. 文本处理命令--wc、sed

    一.wc wc命令的功能为统计指定文件中的字节数.字数.行数,并将统计结果显示输出. 测试文件内容: (my_python_env)[root@hadoop26 ~]# cat test hnlinu ...

  8. 用c语言基本实现wc.exe功能

    网址:https://github.com/3216005214/wc.exe wc项目要求 wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和行数.这个项目要求写一个命令行程序,模仿 ...

  9. 软件工程—WC功能实现 (JAVA)

    软件工程-WC功能实现(JAVA) Github项目地址:https://github.com/Ousyoung/wc 项目要求 ​ wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和 ...

随机推荐

  1. java基础学习——集合

    -------|List: 有存储顺序, 可重复-----------|ArrayList: 数组实现, 查找快, 增删慢,由于是数组实现, 在增和删的时候会牵扯到数组增容, 以及拷贝元素. 所以慢. ...

  2. VB与C#语言部分不用的地方Part1

    1. 数据类型: ① 日期型(Date)      表示日期和时间用两个“#”符号把日期和时间的值括起来,如:#08/20/2001#.#2001-08-20#. ② 变体型(Variant)    ...

  3. IOS学习[Swift中跳转与传值]

    Swift中页面跳转与传值: 1.简单方式 首先,Swift的跳转可分为利用xib文件跳转与storyboard跳转两种方法,我这里选择使用storyboard的界面跳转方法. 1.通过在storyb ...

  4. tkinter第二章(添加图片,背景图片)

    #插入文件图片import tkinter as tk root = tk.Tk() #创建一个标签类, [justify]:对齐方式textLabel = tk.Label(root,text=&q ...

  5. 软工+C(2017第9期) 助教指南

    //上一篇:提问与回复 [备注]:请优先阅读 Handshake/点评/评分 三部分. 0x00 Handshake 了解<构建之法>作者参与软件工程改革的一些背景: http://www ...

  6. 201521123102 《Java程序设计》第6周学习总结

    1. 本周学习总结 2. 书面作业 Q1.clone方法 1.1 Object对象中的clone方法是被protected修饰,在自定义的类中覆盖clone方法时需要注意什么? 子类要实现Clonea ...

  7. MD格式示例

    一个例子: 例子开始 1. 本章学习总结 今天主要学习了三个知识点 封装 继承 多态 2. 书面作业 Q1. java HelloWorld命令中,HelloWorld这个参数是什么含义? 今天学了一 ...

  8. JavaSE(十)之Map总结

    上一篇是总结了Collection接口的各种实现类,这一篇我将分享的是Map的总结,希望大家点评! 一.Map接口 1.1.为什么Collection不能满足集合的所有功能? Collection接口 ...

  9. Winfrom 简单的安卓手机屏幕获取和安卓简单操作

    为啥我要做这个东西了,是因为经常要用投影演示app ,现在有很多这样的软件可以把手机界面投到电脑上 ,但都要安装,比如说360的手机助手,我又讨厌安装,于是就自己捣鼓了下 做了这个东西, 实现了以下简 ...

  10. Spring Boot Maven Plugin(一):repackage目标

    简介 Spring Boot Maven Plugin插件提供spring boot在maven中的支持.允许你打包可运行的jar包或war包. 插件提供了几个maven目标和Spring Boot ...