8

現在、いくつかの HttpWebRequest 関数を変更しようとしていますが、HttpWebRequest には (逆シリアル化コンストラクター以外に) パブリック コンストラクターがないため、継承によってそれを行うことはできません。これを行うための回避策はありますか?

私の目的は、以下の例のようなコードを作成することですが、このクラス オブジェクトは HttpWebRequest のプロパティとメソッドを継承する必要があります。

using System;
using System.Net;
using System.Threading;

public class AsyncWebRequest:WebRequest
{
    private readonly AsyncCallback getResponseCallback;
    private readonly Uri uri;
    private volatile int RetriesLeft = 3;
    private volatile WebRequest request;

    public AsyncWebRequest(string uri, AsyncCallback getResponseCallback)
       :this(new Uri(uri), getResponseCallback)
    {
    }

    public AsyncWebRequest(Uri uri, AsyncCallback getResponseCallback):base()
    {
        this.uri = uri;
        this.getResponseCallback = getResponseCallback;
    }

    private IAsyncResult BeginGetResponse()
    {
        request = HttpWebRequest.CreateDefault(uri);
        ((HttpWebRequest)request).ReadWriteTimeout = Timeout;
        var result = request.BeginGetResponse(GetResponseCallback, null);
        ThreadPool.RegisterWaitForSingleObject(result.AsyncWaitHandle,
            GetResponseTimeout, null, Timeout, true);
        return result;
    }

    private void GetResponseTimeout(object state, bool timedOut)
    {
        if (timedOut)
        {
            Retry();
        }
    }

    private void Retry()
    {
        request.Abort();
        bool retry = false;
        lock (request)
        {
            if (RetriesLeft > 0)
            {
                Interlocked.Decrement(ref RetriesLeft);
                retry = true;
            }
        }
        if (retry)
        {
            BeginGetResponse();
        }
        else
        {
            getResponseCallback(null);
        }
    }

    private void GetResponseCallback(IAsyncResult AsyncResult)
    {
        try
        {
            getResponseCallback(AsyncResult);
        }
        catch(WebException webException)
        {
            Retry();
        }
    }
}
4

2 に答える 2

12

HttpWebRequest から継承することはできませんが (シリアライゼーション コンストラクターを呼び出したくない場合)、構成と委任、および WebRequest から継承することはできます (それでうまくいくかどうかはわかりませんが、機能的には非常に似ています)。WebRequest にはデフォルトのコンストラクターがあります。

この場合、クラスから拡張することはできないため、クラスを HttpWebRequest (is-a 関係のように) に「する」ことはできませんが、WebRequest に「する」ことはできます。これで十分です。

WebRequest から継承し、WebRequest 型のインスタンス メンバーを持つクラスを作成し、HttpWebRequest を作成し、新しい型のコンストラクターでインスタンス メンバーに割り当て、すべての呼び出しをその参照に委任することができます (一種のデコレーター パターン)。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Text;

namespace ClassLibrary1
{
    public class MyHttpWebRequest : WebRequest
    {
        private WebRequest request;

        public MyHttpWebRequest(string uri)
        {
            request = HttpWebRequest.Create(uri);
        }

        public override WebResponse GetResponse()
        {
            // do your extras, or just delegate to get the original code, as long
            // as you still keep to expected behavior etc.
            return request.GetResponse();
        }

    }
}
于 2008-12-30T16:32:40.497 に答える
4

シリアライゼーション コンストラクターをだまして入札を行わせることができない限り、いいえ、方法はありません。

そのクラスのコンストラクターは内部であるため、それらを呼び出す方法はありません。

于 2008-12-30T15:35:37.913 に答える