1

form1 top で私がした:

private volatile bool _workersEnabled;
private ConcurrentQueue<Bitmap> _imageBuffer;
Thread threadA;
Thread threadB;

次に、コンストラクターで次のようにします。

_workersEnabled = false;
_imageBuffer = new ConcurrentQueue<Bitmap>();

threadA = new Thread(CaptureScreensEntryPoint);
threadB = new Thread(ConsumeScreensEntryPoint);

次に、ボタンクリックイベントで:

private void StartRecording_Click(object sender, EventArgs e)
{
    ffmp.Start("test.avi", 25);
    _workersEnabled = true;
    threadA.Start();
    threadA.Start();

    //Disable the button, so we eliminate the possibility to start this twice (would throw an exception anyway).
    StartRecording.Enabled = false;
}

その後、次のように追加しました。

private void CaptureScreensEntryPoint()
{
    while(_workersEnabled)
    {
        Bitmap bitmap = (Bitmap)ScreenCapture.CaptureScreen(true);

        //Just add it to the queue.
        _imageBuffer.Enqueue(bitmap);

        //Wait a bit
        Thread.Sleep(40);
    }
}

private void ConsumeScreensEntryPoint()
{
    while (_workersEnabled)
    {
        Bitmap workItem = null;
        if (_imageBuffer.TryDequeue(out workItem))
        {
            ffmp.PushFrame(workItem);
            workItem.Dispose();
        }

        //Also wait a bit here. Don't want to eat up the entire processor.
        Thread.Sleep(10);
    }
}

例外は次のthreadA.Start();とおりです。

ThreadStateException : スレッドが実行中または終了しています。再起動できません

System.Threading.ThreadStateException was unhandled
  HResult=-2146233056
  Message=Thread is running or terminated; it cannot restart.
  Source=mscorlib
  StackTrace:
       at System.Threading.Thread.StartupSetApartmentStateInternal()
       at System.Threading.Thread.Start(StackCrawlMark& stackMark)
       at System.Threading.Thread.Start()
       at ScreenVideoRecorder.Form1.StartRecording_Click(Object sender, EventArgs e) in d:\C-Sharp\ScreenVideoRecorder\ScreenVideoRecorderWorkingVersion\Form1.cs:line 152
       at System.Windows.Forms.Control.OnClick(EventArgs e)
       at System.Windows.Forms.Button.OnClick(EventArgs e)
       at System.Windows.Forms.Button.OnMouseUp(MouseEventArgs mevent)
       at System.Windows.Forms.Control.WmMouseUp(Message& m, MouseButtons button, Int32 clicks)
       at System.Windows.Forms.Control.WndProc(Message& m)
       at System.Windows.Forms.ButtonBase.WndProc(Message& m)
       at System.Windows.Forms.Button.WndProc(Message& m)
       at System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
       at System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
       at System.Windows.Forms.NativeWindow.DebuggableCallback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)
       at System.Windows.Forms.UnsafeNativeMethods.DispatchMessageW(MSG& msg)
       at System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(IntPtr dwComponentID, Int32 reason, Int32 pvLoopData)
       at System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner(Int32 reason, ApplicationContext context)
       at System.Windows.Forms.Application.ThreadContext.RunMessageLoop(Int32 reason, ApplicationContext context)
       at System.Windows.Forms.Application.Run(Form mainForm)
       at ScreenVideoRecorder.Program.Main() in d:\C-Sharp\ScreenVideoRecorder\ScreenVideoRecorderWorkingVersion\Program.cs:line 18
       at System.AppDomain._nExecuteAssembly(RuntimeAssembly assembly, String[] args)
       at System.AppDomain.ExecuteAssembly(String assemblyFile, Evidence assemblySecurity, String[] args)
       at Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()
       at System.Threading.ThreadHelper.ThreadStart_Context(Object state)
       at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
       at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
       at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
       at System.Threading.ThreadHelper.ThreadStart()
  InnerException:

どうすれば解決できますか?

4

2 に答える 2

1

どちらが時間がかかるかわかりません:ScreenCapture.CaptureScreenまたはffmp.PushFrame(bitmap). これはビットマップなので、ffmp はそれほど高速ではなく、時間がかかると思います。

プロデューサー/コンシューマーのアプローチを採用します。スクリーンショットを実行するスレッド A があり、それらを a にプッシュし、ConcurrentQueueそこからそれらを ffmp にプッシュするスレッド B によってピックアップされます。

いくつかのサンプル コード (テストされておらず、完全に動作することが保証されていないため、おそらくいくつかの調整を行う必要があります):

//We need this thread for cross-thread access, so we don't want it cached. 
private volatile bool _workersEnabled;
private ConcurrentQueue<Bitmap> _imageBuffer;

public Form1()
{
  InitializeComponent();
  ffmp = new Ffmpeg();
  sc = new ScreenCapture();

  _workersEnabled = false; 
  _imageBuffer = new ConcurrentQueue<Bitmpap>();

  threadA = new Thread(CaptureScreensEntryPoint);
  threadB = new Thread(ConsumeScreensEntryPoint);
}

private void StartRecording_Click(object sender, EventArgs e)
{
  _workersEnabled = true; 
  threadA.Start();
  threadB.Start();

  //Disable the button, so we eliminate the possibility to start this twice (would throw an exception anyway).
  StartRecording.Enabled = false;
}

private void CaptureScreensEntryPoint() 
{
  while(_workersEnabled)
  {
    Bitmap bitmap = (Bitmap)ScreenCapture.CaptureScreen(true);

    //just add it to the queue.
    _imageBuffer.Enqueue(bitmap);

    //wait a bit
    Thread.Sleep(40);
  }
}

private void ConsumeScreensEntryPoint() 
{
  while(_workersEnabled)
  {
    Bitmap workItem = null;
    if(_imageBuffer.TryDequeue(out workItem))
    { 
      ffmp.PushFrame(workItem);
      workItem.Dispose();
    }

    //Also wait a bit here. Don't want to eat up the entire processor.
    Thread.Sleep(10);
  }
}

ワーカーを停止したいとき、たとえばフォームが閉じたとき、または専用のボタンがあるときなどに設定_workersEnabledしてください。falseエラー処理を追加して、両方のスレッドのスリープ期間を構成可能にすることもできます。


ノート

私は TPL でそれを行うことができたと思いますが、これは投稿するのがいくらか速かったです。ソリューションを調整するか、新しいソリューションを投稿してください。

于 2013-05-30T21:12:29.537 に答える
0

キャプチャ コードを別のスレッドに配置してみてください。デスクトップ キャプチャ アプリケーションで同様のことを行ったところ、約 60 FPS を得ることができました。

于 2013-05-30T20:29:16.203 に答える