在C#使用Console.SetOut方法轻松灵活地操作控制台

作者:V君 发布于:2015-9-24 23:04 Thursday 分类:挖坑经验

标题说得很清楚, 这次不TL;DR了 ,强行听我扯吧 (=゚ω゚)=     (´∀((☆ミつ

起初是这样: (这回只讲控制台,例子细节就别在意啦(在意细节的都是笨蛋哦(´∀((☆ミつ

点击查看原图

用Linux的家伙都应该知道能用转义符去改变控制台的字体颜色,还能变粗变(),

然而windows原生cmd就没这么多玩法了, 我大井似乎只能一次次调用Console的颜色改变方法

泥萌会觉得这是要一次次切换Console前景背景色或者是自己封装一层再在里面一次次切换么?


部队! 某很懒的, 先放狗了, 这种东西一定会有人蛋过 ☆_,☆


看, edokan桑在爆栈上的回答, 用了他的奇怪代码居然能写得这么愉快!

点击查看原图

泥煤看错, 就是直接写控制台, 只是调用了一些奇怪的扩展方法!


用爽了之后, 该F12进去看看胖次, 这货是如何做到那些神奇的事情了.

要用这货的功能首先要调用它的静态方法Install, 之后就能穿插它奇怪的扩展方法了.

Install方法里面就简单的一句Console.SetOut(new EscapeSequencer(Console.Out));

这是个 TextWriter, 也是关键所在, 重新指定Console的Write系列方法的写入目标

也就是说 Console.WriteLine 方法先写到你的 TextWriter,

你自个处理高兴了再接着Console.Out往系统标准输出里面倒

(小声:还能串起来咧 ... 后面说(=゚ω゚)= (´∀((☆ミつ

EscapeSequencer就是接管你往Console里面写的家伙 

配合使用扩展方法, 生成一些奇怪的包裹字符, 再经由它挨个字符处理,这样就能从中作怪了,

比如解析到特定字符序列时先捅一下Console的颜色设置方法再往下输出.


自从用了这种方式以后妈妈再也不用担心我整天重复造轮子封装Console了!

用高兴了么?


别太得意了, 这时候忽然来了个要求, 

要你把所有在控制台输出的东西写到文本文件, 每行还要有时间戳在前面.


嗯这时候泥会怎样鸡汁的处理变更呢?

是在每次输出控制台的语句后面多复制一行改成写到文件? (´∀((☆ミつ

还是写个封装, 包住控制台输出语句, 每当调用就文件和控制台一起写? (´∀((☆ミつ

不玩你了, 回到主题吧 (´∀((☆ミつ


其实Console.SetOut不仅能改变Console的写入目标, 还能玩钩子链

依葫芦画瓢,咱也定义一个叫做Logger的TextWriter实现,

也整一个Install静态方法在里面调用 Console.SetOut

并在 EscapeSequencer.Install() 之后追加一行调用它


嗯嗯, Console.SetOut(new Logger(Console.Out))

这时候的Console.Out已经是edokan桑的转义解析过后输出

理想的情况, Console.WriteLine("ba{0}ba","la".Red()); 

这句代码应该是先把la用转义序列包裹以指示红色, 然后丢进Console.Out

因为我们之前Install里面用SetOut改变了它,

进到EscapeSequencer里面, 挨个处理序列,该变色/还原时自动调用,

接着输出到咱们定义的Logger,一边往控制台丢,一边写文件

因为重写TextWriter的是Write(char)

一个个字符来处理的, 所以颜色的改变/还原十分有序, 不会错乱

所以处理换行需要留个心眼, 泥堪我这个:

点击查看原图

_needNewLine是成员bool变量,初始值是true

_baseWrlter是构造时传入的Console.Out

_file不解释 (=゚ω゚)=


然后放心的F5吧!你会发现和理想的情况一样控制台和文件都顺利得到了你的控制台输出

....只不过你的文件里面得到的是带有奇怪转义包裹的乱七八糟的字符wwwwwww


其实只要把Logger.Install()放到EscapeSequencer.Install()上面就好 (=゚ω゚)= 


那是因为理解错了, 在钩子思想里, 最后一个加挂的钩子总是首个被调用

这时候被转义序列包裹住的内容先到你的Logger,写完文件才到EscapeSequencer

于是就得到了控制台输出好好地,日志却乱七八糟的窘境.


//EOF

标签: 软件开发 C# 控制台 钩子链

评论(0) 引用(0) 浏览(903)

Powered by emlog 去你妹的备案 sitemap