C #, сохранение кадров с веб-камеры непрерывно каждые 5 секунд - PullRequest
0 голосов
/ 09 ноября 2019

Когда запускается код, основанный на графическом интерфейсе, и он фактически открывает канал веб-камеры и сохраняет кадр каждые 5 секунд, после случайного времени работы (это может быть 5 минут или 20 минут), возникает исключение, возникающее при использовании объектадругой процесс, возможно связанный с сохранением текущего кадра. Любые идеи, что вызывает проблему и какие модификации кода должны быть сделаны?

public partial class Form1 : Form
    {
        private FilterInfoCollection CaptureDevices;
        private VideoCaptureDevice videoSource;
        bool blncapturing;

        public Form1()
        {
            InitializeComponent();
            timer1.Enabled = false;
            timer1.Interval = 5000;
            blncapturing = false;
        }



        private void button1_Click(object sender, EventArgs e)
        {
            videoSource = new VideoCaptureDevice(CaptureDevices[comboBox1.SelectedIndex].MonikerString);
            videoSource.NewFrame += new NewFrameEventHandler(VideoSource_NewFrame);
            videoSource.Start();
            timer1.Enabled = true;
        }


        private void VideoSource_NewFrame(object sender, NewFrameEventArgs eventArgs)
        {
            if (pictureBox1.Image != null)
                ((IDisposable)pictureBox1.Image).Dispose();


            pictureBox1.Image = (Bitmap)eventArgs.Frame.Clone();
        }

        private void Form1_Load_1(object sender, EventArgs e)
        {

            CaptureDevices = new FilterInfoCollection(FilterCategory.VideoInputDevice);
            foreach (FilterInfo Device in CaptureDevices)
            {
                comboBox1.Items.Add(Device.Name);
            }
            comboBox1.SelectedIndex = 0;
            videoSource = new VideoCaptureDevice();
        }

        private void button2_Click(object sender, EventArgs e)
        {
            videoSource.Stop();
            pictureBox1.Image = null;
            pictureBox1.Invalidate();
            pictureBox2.Image = null;
            pictureBox2.Invalidate();

        }

        private void button3_Click(object sender, EventArgs e)
        {
            Capturing();
        }

        private void button4_Click(object sender, EventArgs e)
        {
            if (videoSource.IsRunning == true)
            {
                videoSource.Stop();
            }
            Application.Exit(null);
        }

        private void timer1_Tick(object sender, EventArgs e)
        {
            timer1.Stop();
            Capturing();
            timer1.Start();
        }

        private void Capturing()
        {
            try
            {
                if (blncapturing == false)
                {
                    blncapturing = true;

                    pictureBox2.Image = (Bitmap)pictureBox1.Image.Clone();

                    string strGrabFileName = String.Format("C:\\Users\\echristo\\Desktop\\trial\\Snapshot_{0:yyyyMMdd_hhmmss.fff}.bmp", DateTime.Now);
                    pictureBox2.Image.Save(strGrabFileName, System.Drawing.Imaging.ImageFormat.Jpeg) ;
                    blncapturing = false;

                }
            }
            catch (Exception e)
            {
                MessageBox.Show(e.Message);                
            }


        }

1 Ответ

0 голосов
/ 09 ноября 2019

Если вы обращаетесь к объектам из разных потоков, вам нужна блокировка.
Также используйте invoke при доступе к элементу управления из другого потока.

private readonly object myLock = new object();

private void VideoSource_NewFrame(object sender, NewFrameEventArgs eventArgs)
{
    lock (myLock) {
        this.Invoke((Action)(() => {
            if (pictureBox1.Image != null)
                ((IDisposable)pictureBox1.Image).Dispose();


            pictureBox1.Image = (Bitmap)eventArgs.Frame.Clone();        

        }));
    }
}

private void timer1_Tick(object sender, EventArgs e)
{
    timer1.Stop();

    // you can try with timer1 inside the lock
    lock (myLock) {
        // you don't need invoke here, because the winforms timer runs on the main thread.
        Capturing();    
    }

    timer1.Start();
}

Также, как сказал @TheGeneral, утилизируйте pictureBox2.Imageпосле сохранения.

...