问题 实现派生类的TextWriter


我有两个课程,我都不能以任何方式改变:

1级:取得成绩 TextWriter 作为构造函数参数并将其用作输出流。

第2类:提供方法 WriteLine(string)

我需要一个适配器,这样Class1的所有输出都写入Class2。因此我启动了一个扩展的适配器 TextWriter 并缓冲传入的文本,一旦新行到达就将其刷新到class2实例。

但是,TextWriter中有很多方法 - 我应该实现哪些方法? Class1中的输出仅为字符串。

根据MSDN,应该至少覆盖Write(char),但是,这迫使我自己完成所有\ r \ n新行处理...

Q1:你知道更好的方法来实现我的目标吗? Q2:如果不是,我应该覆盖哪些TextWriter方法以实现最小的实现工作。


5127
2017-07-18 00:35


起源



答案:


实施 Write(char) 在你的 TextWriter 派生类就是你需要做的。如果有人打电话 WriteLine 在你的新类,基类 WriteLine 方法被调用。它会做正确的事情:打电话给你 Write 与个人的方法 \r 和 \n 字符。

其实, WriteLine(string) 看起来像这样:

void WriteLine(string s)
{
    Write(s);
    Write("\r\n");
}

Write(string) 实际上是:

foreach (char c in s)
{
    Write(c);
}

全部 Write 方法 TextWriter 解决问题 Write(char) 在一个循环中。

你真的不需要实现其他任何东西。只是覆盖 Write(char) 并插入它。它会工作。

能够 覆盖那些其他方法。这样做会使您的课程更有效(更快)。但这不是必需的。我说你做的最简单的事情。然后,如果您在分析自定义编写器太慢后确定,请根据需要覆盖其他方法。

这是一个最小的 TextWriter 后裔:

public class ConsoleTextWriter: TextWriter
{
    public override void Write(char value)
    {
        Console.Write(value);
    }

    public override Encoding Encoding
    {
        get { return Encoding.Default; }
    }
}

如果我然后写:

using (var myWriter = new ConsoleTextWriter())
{
    myWriter.Write("hello, world");
    myWriter.WriteLine();
    myWriter.WriteLine();
    myWriter.WriteLine("Goodbye cruel world.");
    myWriter.Write("Fee fie foe foo!");
}

输出是:

hello, world

Goodbye cruel world.
Fee fie foe foo!

16
2017-07-18 00:46



感谢您的回答,但是,我何时知道何时通过WriteLine()将其刷新到基础类2?我只能在“全线”做到这一点。例如。我必须在Write(char)中寻找NewLines吗?是\ n一个NewLine? \ r \ n?也许在不同的平台上别的什么? - D.R.
“你可以覆盖那些其他方法。这样做可以使你的课程更有效率(更快)”你的意思是哪种方法?只有压倒一切 Write(char) 导致打印速度非常慢。 - Protector one
@Protectorone:覆盖任何虚拟方法都会提高性能。你是对的,最小的实现 - 仅覆盖 Write(char) - 导致打印速度非常慢。但它 作品,这是一个非常方便的事情。 - Jim Mischel
@ bikeman868: WriteLine(string)填充字符缓冲区并调用 Write(Char[], int buffer, int count),反过来打电话 Write(Char) 在一个循环中。看到 referencesource.microsoft.com/#mscorlib/system/io/...。我很确定它总是这样,但我可能会弄错。 - Jim Mischel
在Visual Studio中按F12并查看反编译的源代码。 WriteLine非常清楚地不在当前版本的.Net 4.0中使用Write(Char)。 - bikeman868


答案:


实施 Write(char) 在你的 TextWriter 派生类就是你需要做的。如果有人打电话 WriteLine 在你的新类,基类 WriteLine 方法被调用。它会做正确的事情:打电话给你 Write 与个人的方法 \r 和 \n 字符。

其实, WriteLine(string) 看起来像这样:

void WriteLine(string s)
{
    Write(s);
    Write("\r\n");
}

Write(string) 实际上是:

foreach (char c in s)
{
    Write(c);
}

全部 Write 方法 TextWriter 解决问题 Write(char) 在一个循环中。

你真的不需要实现其他任何东西。只是覆盖 Write(char) 并插入它。它会工作。

能够 覆盖那些其他方法。这样做会使您的课程更有效(更快)。但这不是必需的。我说你做的最简单的事情。然后,如果您在分析自定义编写器太慢后确定,请根据需要覆盖其他方法。

这是一个最小的 TextWriter 后裔:

public class ConsoleTextWriter: TextWriter
{
    public override void Write(char value)
    {
        Console.Write(value);
    }

    public override Encoding Encoding
    {
        get { return Encoding.Default; }
    }
}

如果我然后写:

using (var myWriter = new ConsoleTextWriter())
{
    myWriter.Write("hello, world");
    myWriter.WriteLine();
    myWriter.WriteLine();
    myWriter.WriteLine("Goodbye cruel world.");
    myWriter.Write("Fee fie foe foo!");
}

输出是:

hello, world

Goodbye cruel world.
Fee fie foe foo!

16
2017-07-18 00:46



感谢您的回答,但是,我何时知道何时通过WriteLine()将其刷新到基础类2?我只能在“全线”做到这一点。例如。我必须在Write(char)中寻找NewLines吗?是\ n一个NewLine? \ r \ n?也许在不同的平台上别的什么? - D.R.
“你可以覆盖那些其他方法。这样做可以使你的课程更有效率(更快)”你的意思是哪种方法?只有压倒一切 Write(char) 导致打印速度非常慢。 - Protector one
@Protectorone:覆盖任何虚拟方法都会提高性能。你是对的,最小的实现 - 仅覆盖 Write(char) - 导致打印速度非常慢。但它 作品,这是一个非常方便的事情。 - Jim Mischel
@ bikeman868: WriteLine(string)填充字符缓冲区并调用 Write(Char[], int buffer, int count),反过来打电话 Write(Char) 在一个循环中。看到 referencesource.microsoft.com/#mscorlib/system/io/...。我很确定它总是这样,但我可能会弄错。 - Jim Mischel
在Visual Studio中按F12并查看反编译的源代码。 WriteLine非常清楚地不在当前版本的.Net 4.0中使用Write(Char)。 - bikeman868


我正在添加另一个答案,因为我无法得到上述答案!

根据我的经验,我必须覆盖 WriteLine() 和 WriteLine(string) 为了使这些功能起作用。

这是一个可用于在长时间运行的任务中编写网页的示例。顺便说一句,HttpResponse对象来自ASP.net MVC。

public class WebConsoleWriter : TextWriter
{
    HttpResponseBase Response { get; set; }
    bool FlushAfterEveryWrite { get; set; }
    bool AutoScrollToBottom { get; set; }
    Color BackgroundColor { get; set; }
    Color TextColor { get; set; }

    public WebConsoleWriter(HttpResponseBase response, bool flushAfterEveryWrite, bool autoScrollToBottom)
    {
        Response = response;
        FlushAfterEveryWrite = flushAfterEveryWrite;
        AutoScrollToBottom = autoScrollToBottom;
        BackgroundColor = Color.White;
        TextColor = Color.Black;
    }

    public WebConsoleWriter(HttpResponseBase response, bool flushAfterEveryWrite,  bool autoScrollToBottom, Color backgroundColor, Color textColor)
    {
        Response = response;
        FlushAfterEveryWrite = flushAfterEveryWrite;
        AutoScrollToBottom = autoScrollToBottom;
        BackgroundColor = backgroundColor;
        TextColor = textColor;
    }

    public virtual void WritePageBeforeStreamingText()
    {
        string headerFormat =
@"<!DOCTYPE html>
<html>
<head>
    <title>Web Console</title>
    <style>
        html {{
            background-color: {0};
            color: {1};
        }}
    </style>        
</head>
<body><pre>";
        string backgroundColorHex = ColorTranslator.ToHtml(BackgroundColor);
        string foregroundColorHex = ColorTranslator.ToHtml(TextColor);
        Response.Write(string.Format(headerFormat, backgroundColorHex, foregroundColorHex));

        // Add a 256 byte comment because I read that some browsers will automatically buffer the first 256 bytes.
        Response.Write("<!--");
        Response.Write(new string('*', 256));
        Response.Write("-->");
        Response.Flush();
    }

    public virtual void WritePageAfterStreamingText()
    {
        Response.Write("</pre></body></html>");
    }

    public override void Write(string value)
    {
        string encoded = Encode(value);
        Response.Write(encoded);            
        if (FlushAfterEveryWrite)
            Response.Flush();
    }

    public override void WriteLine(string value)
    {
        Response.Write(Encode(value) + "\n");
        if (AutoScrollToBottom)
            ScrollToBottom();
        if (FlushAfterEveryWrite)
            Response.Flush();
    }

    public override void WriteLine()
    {
        Response.Write('\n');
        if (AutoScrollToBottom)
            ScrollToBottom();
        if (FlushAfterEveryWrite)
            Response.Flush();
    }

    private string Encode(string s)
    {
        return s.Replace("&", "&amp;").Replace("<", "&lt;").Replace(">", "&gt;");
    }

    public override void Flush()
    {
        Response.Flush();
    }

    public void ScrollToBottom()
    {
        Response.Write("<script>window.scrollTo(0, document.body.scrollHeight);</script>");
    }

    public override System.Text.Encoding Encoding
    {
        get { throw new NotImplementedException(); }
    }
}

0
2017-10-23 16:30