如何从网络摄像头连续捕获图像?
我想尝试对象识别(可能使用 java 媒体框架)。
我正在考虑创建两个线程
一个线程:
- 节点 1:捕获实时图像
- 节点2:将图像另存为“1.jpg”
- 节点 3:等待 5 秒
- 节点 4:重复...
其他线程:
- 节点 1:等到图像被捕获
- 节点 2:使用“1.jpg”从每个像素中获取颜色
- 节点 3:将数据保存在数组中
- 节点 4:重复...
这个 JavaCV 实现工作正常。
代码:
import org.bytedeco.javacv.*;
import org.bytedeco.opencv.opencv_core.IplImage;
import java.io.File;
import static org.bytedeco.opencv.global.opencv_core.cvFlip;
import static org.bytedeco.opencv.helper.opencv_imgcodecs.cvSaveImage;
public class Test implements Runnable {
final int INTERVAL = 100;///you may use interval
CanvasFrame canvas = new CanvasFrame("Web Cam");
public Test() {
canvas.setDefaultCloseOperation(javax.swing.JFrame.EXIT_ON_CLOSE);
}
public void run() {
new File("images").mkdir();
FrameGrabber grabber = new OpenCVFrameGrabber(0); // 1 for next camera
OpenCVFrameConverter.ToIplImage converter = new OpenCVFrameConverter.ToIplImage();
IplImage img;
int i = 0;
try {
grabber.start();
while (true) {
Frame frame = grabber.grab();
img = converter.convert(frame);
//the grabbed frame will be flipped, re-flip to make it right
cvFlip(img, img, 1);// l-r = 90_degrees_steps_anti_clockwise
//save
cvSaveImage("images" + File.separator + (i++) + "-aa.jpg", img);
canvas.showImage(converter.convert(img));
Thread.sleep(INTERVAL);
}
} catch (Exception e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
Test gs = new Test();
Thread th = new Thread(gs);
th.start();
}
}
JavaCV的配置也有帖子
您可以修改代码并能够定期保存图像并进行所需的其余处理。
前段时间,我创建了通用 Java 库,可用于使用 PC 网络摄像头拍照。API 非常简单,功能不过分,可以独立工作,还支持其他网络摄像头驱动程序,如 OpenIMAJ、JMF、FMJ、LTI-CIVIL 等,以及一些 IP 摄像头。
该项目的链接是https://github.com/sarxos/webcam-capture
示例代码(拍照并保存在test.jpg中):
Webcam webcam = Webcam.getDefault();
webcam.open();
BufferedImage image = webcam.getImage();
ImageIO.write(image, "JPG", new File("test.jpg"));
它也可以在 Maven 中央存储库中使用,或者作为单独的 ZIP 提供,其中包括所有必需的依赖项和 3rd 方 JAR。
JMyron 使用起来非常简单。 http://webcamxtra.sourceforge.net/
myron = new JMyron();
myron.start(imgw, imgh);
myron.update();
int[] img = myron.image();
这种使用 JavaCV 与 gt_ebuddy 的答案不同,但我的视频输出质量比他的答案高得多。我还添加了一些其他随机改进(例如在按下 和 时关闭程序ESC,CTRL+C并确保关闭程序正确使用的资源)。
import java.awt.event.ActionEvent;
import java.awt.event.KeyEvent;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.image.BufferedImage;
import javax.swing.AbstractAction;
import javax.swing.ActionMap;
import javax.swing.InputMap;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.KeyStroke;
import com.googlecode.javacv.CanvasFrame;
import com.googlecode.javacv.OpenCVFrameGrabber;
import com.googlecode.javacv.cpp.opencv_core.IplImage;
public class HighRes extends JComponent implements Runnable {
private static final long serialVersionUID = 1L;
private static CanvasFrame frame = new CanvasFrame("Web Cam");
private static boolean running = false;
private static int frameWidth = 800;
private static int frameHeight = 600;
private static OpenCVFrameGrabber grabber = new OpenCVFrameGrabber(0);
private static BufferedImage bufImg;
public HighRes()
{
// setup key bindings
ActionMap actionMap = frame.getRootPane().getActionMap();
InputMap inputMap = frame.getRootPane().getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW);
for (Keys direction : Keys.values())
{
actionMap.put(direction.getText(), new KeyBinding(direction.getText()));
inputMap.put(direction.getKeyStroke(), direction.getText());
}
frame.getRootPane().setActionMap(actionMap);
frame.getRootPane().setInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW, inputMap);
// setup window listener for close action
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.addWindowListener(new WindowAdapter()
{
public void windowClosing(WindowEvent e)
{
stop();
}
});
}
public static void main(String... args)
{
HighRes webcam = new HighRes();
webcam.start();
}
@Override
public void run()
{
try
{
grabber.setImageWidth(frameWidth);
grabber.setImageHeight(frameHeight);
grabber.start();
while (running)
{
final IplImage cvimg = grabber.grab();
if (cvimg != null)
{
// cvFlip(cvimg, cvimg, 1); // mirror
// show image on window
bufImg = cvimg.getBufferedImage();
frame.showImage(bufImg);
}
}
grabber.stop();
grabber.release();
frame.dispose();
}
catch (Exception e)
{
e.printStackTrace();
}
}
public void start()
{
new Thread(this).start();
running = true;
}
public void stop()
{
running = false;
}
private class KeyBinding extends AbstractAction {
private static final long serialVersionUID = 1L;
public KeyBinding(String text)
{
super(text);
putValue(ACTION_COMMAND_KEY, text);
}
@Override
public void actionPerformed(ActionEvent e)
{
String action = e.getActionCommand();
if (action.equals(Keys.ESCAPE.toString()) || action.equals(Keys.CTRLC.toString())) stop();
else System.out.println("Key Binding: " + action);
}
}
}
enum Keys
{
ESCAPE("Escape", KeyStroke.getKeyStroke(KeyEvent.VK_ESCAPE, 0)),
CTRLC("Control-C", KeyStroke.getKeyStroke(KeyEvent.VK_C, KeyEvent.CTRL_DOWN_MASK)),
UP("Up", KeyStroke.getKeyStroke(KeyEvent.VK_UP, 0)),
DOWN("Down", KeyStroke.getKeyStroke(KeyEvent.VK_DOWN, 0)),
LEFT("Left", KeyStroke.getKeyStroke(KeyEvent.VK_LEFT, 0)),
RIGHT("Right", KeyStroke.getKeyStroke(KeyEvent.VK_RIGHT, 0));
private String text;
private KeyStroke keyStroke;
Keys(String text, KeyStroke keyStroke)
{
this.text = text;
this.keyStroke = keyStroke;
}
public String getText()
{
return text;
}
public KeyStroke getKeyStroke()
{
return keyStroke;
}
@Override
public String toString()
{
return text;
}
}
您也可以尝试Java Webcam SDK库。SDK 演示小程序可在链接中找到。
我在视频会议应用程序中使用了 JMF,它在两台笔记本电脑上运行良好:一台带有集成网络摄像头,另一台带有旧的 USB 网络摄像头。它需要预先安装和配置 JMF,但是一旦完成,您就可以相当轻松地通过 Java 代码访问硬件。
你可以试试Marvin 框架。它提供了一个使用相机的界面。此外,它还提供了一组实时视频处理功能,如对象跟踪和过滤。
看一看!
实时视频处理演示:
http ://www.youtube.com/watch?v=D5mBt0kRYvk
您可以使用以下来源。只需每 5 秒使用MarvinImageIO.saveImage()保存一帧。
网络摄像头视频演示:
public class SimpleVideoTest extends JFrame implements Runnable{
private MarvinVideoInterface videoAdapter;
private MarvinImage image;
private MarvinImagePanel videoPanel;
public SimpleVideoTest(){
super("Simple Video Test");
videoAdapter = new MarvinJavaCVAdapter();
videoAdapter.connect(0);
videoPanel = new MarvinImagePanel();
add(videoPanel);
new Thread(this).start();
setSize(800,600);
setVisible(true);
}
@Override
public void run() {
while(true){
// Request a video frame and set into the VideoPanel
image = videoAdapter.getFrame();
videoPanel.setImage(image);
}
}
public static void main(String[] args) {
SimpleVideoTest t = new SimpleVideoTest();
t.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}
}
对于那些只想拍一张照片的人:
网络摄像头图片.java
public class WebcamPicture {
public static void main(String[] args) {
try{
MarvinVideoInterface videoAdapter = new MarvinJavaCVAdapter();
videoAdapter.connect(0);
MarvinImage image = videoAdapter.getFrame();
MarvinImageIO.saveImage(image, "./res/webcam_picture.jpg");
} catch(MarvinVideoInterfaceException e){
e.printStackTrace();
}
}
}
webcam = Webcam.getDefault();
webcam.open();
if (webcam.isOpen()) { //if web cam open
BufferedImage image = webcam.getImage();
JLabel imageLbl = new JLabel();
imageLbl.setSize(640, 480); //show captured image
imageLbl.setIcon(new ImageIcon(image));
int showConfirmDialog = JOptionPane.showConfirmDialog(null, imageLbl, "Image Viewer", JOptionPane.YES_NO_OPTION, JOptionPane.QUESTION_MESSAGE, new ImageIcon(""));
if (showConfirmDialog == JOptionPane.YES_OPTION) {
JFileChooser chooser = new JFileChooser();
chooser.setDialogTitle("Save Image");
chooser.setFileFilter(new FileNameExtensionFilter("IMAGES ONLY", "png", "jpeg", "jpg")); //this file extentions are shown
int showSaveDialog = chooser.showSaveDialog(this);
if (showSaveDialog == 0) { //if pressed 'Save' button
String filePath = chooser.getCurrentDirectory().toString().replace("\\", "/");
String fileName = chooser.getSelectedFile().getName(); //get user entered file name to save
ImageIO.write(image, "PNG", new File(filePath + "/" + fileName + ".png"));
}
}
}
http://grack.com/downloads/school/enel619.10/report/java_media_framework.html
将播放器与 Swing 一起使用
Player 也可以在 Swing 应用程序中轻松使用。以下代码创建了一个基于 Swing 的电视捕捉程序,视频输出显示在整个窗口中:
import javax.media.*;
import javax.swing.*;
import java.awt.*;
import java.net.*;
import java.awt.event.*;
import javax.swing.event.*;
public class JMFTest extends JFrame {
Player _player;
JMFTest() {
addWindowListener( new WindowAdapter() {
public void windowClosing( WindowEvent e ) {
_player.stop();
_player.deallocate();
_player.close();
System.exit( 0 );
}
});
setExtent( 0, 0, 320, 260 );
JPanel panel = (JPanel)getContentPane();
panel.setLayout( new BorderLayout() );
String mediaFile = "vfw://1";
try {
MediaLocator mlr = new MediaLocator( mediaFile );
_player = Manager.createRealizedPlayer( mlr );
if (_player.getVisualComponent() != null)
panel.add("Center", _player.getVisualComponent());
if (_player.getControlPanelComponent() != null)
panel.add("South", _player.getControlPanelComponent());
}
catch (Exception e) {
System.err.println( "Got exception " + e );
}
}
public static void main(String[] args) {
JMFTest jmfTest = new JMFTest();
jmfTest.show();
}
}
Java 通常不喜欢访问硬件,因此您需要某种驱动程序,正如 Goldenmean 所说。我已经在我的笔记本电脑上通过找到一个拍摄图片的命令行程序来完成此操作。然后和goldmean解释的一样;您在 takepicture() 例程中从您的 java 程序运行命令行程序,其余代码运行相同。
除了关于将像素值读取到数组中的部分之外,最好将文件保存为 BMP,这已经接近这种格式,然后使用标准的 java 图像库。
使用命令行程序会增加对程序的依赖并降低其可移植性,但网络摄像头也是如此,对吧?
尝试使用 JMyron如何使用 Java 使用网络摄像头。我认为使用 JMyron 是使用 java 访问网络摄像头的最简单方法。我尝试将它与 64 位处理器一起使用,但它给了我一个错误。不过,它在 32 位处理器上运行良好。
我相信网络摄像头附带的网络摄像头应用软件,或者你的原生Windows网络摄像头软件可以在打开网络摄像头后以批处理脚本(windows / dos脚本)运行(即如果它需要外部电源供应)。在 bacth 脚本中,您可以添加适当的延迟以在特定时间段后捕获。并继续循环执行捕获命令。
我想这应该是可能的
-广告
推荐使用 FMJ 进行多媒体相关的 java 应用程序。
FMJ 可以做到这一点,它使用的支持库 LTI-CIVIL 也可以。两者都在 sourceforge 上。
在processing中有一个非常好的接口,这是一种为图形设计的pidgin java。它被用于一些图像识别工作,例如那个链接。
根据您的需要,您可能能够加载在 java 中使用的视频库,或者如果您只是在玩它,您可能可以通过使用处理本身来获得。