Android 远程监控摄像头 移动端+PC端 旧手札变废为宝

本帖是基于网上资源的重新整合与更正:

先上图:


代码如下:

Android移动端代码:共为两部分CameraTest.java和GetIP.java

第一部分:CameraTest.java

package org.wanghai.CameraTest;
import android.app.Activity;
import android.content.Intent;
import android.graphics.ImageFormat;
import android.graphics.Rect;
import android.graphics.YuvImage;
import android.hardware.Camera;
import android.hardware.Camera.Size;
import android.os.Bundle;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
import android.view.SurfaceView;
import android.view.Window;
import android.view.WindowManager;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.Socket;

public class CameraTest extends Activity {
	SurfaceView sView;
	SurfaceHolder surfaceHolder;
	int screenWidth, screenHeight;
	Camera camera;                    // 定义系统所用的照相机
	boolean isPreview = false;        //是否在浏览中
	private String ipname;

	@SuppressWarnings("deprecation")
	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		// 设置全屏
		requestWindowFeature(Window.FEATURE_NO_TITLE);
		getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN);
		setContentView(R.layout.main);

		// 获取IP地址
		Intent intent = getIntent();
		Bundle data = intent.getExtras();
		ipname = data.getString("ipname");

		screenWidth = 640;
		screenHeight = 480;
		sView = (SurfaceView) findViewById(R.id.sView);                  // 获取界面中SurfaceView组件
		surfaceHolder = sView.getHolder();                               // 获得SurfaceView的SurfaceHolder

		// 为surfaceHolder添加一个回调监听器
		surfaceHolder.addCallback(new Callback() {
			@Override
			public void surfaceChanged(SurfaceHolder holder, int format, int width,int height) {
			}
			@Override
			public void surfaceCreated(SurfaceHolder holder) {
				initCamera();                                            // 打开摄像头
			}
			@Override
			public void surfaceDestroyed(SurfaceHolder holder) {
				// 如果camera不为null ,释放摄像头
				if (camera != null) {
					if (isPreview)
						camera.stopPreview();
					camera.release();
					camera = null;
				}
				System.exit(0);
			}
		});
		// 设置该SurfaceView自己不维护缓冲
		surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);

	}

	private void initCamera() {
		if (!isPreview) {
			camera = Camera.open();
		}
		if (camera != null && !isPreview) {
			try{
				Camera.Parameters parameters = camera.getParameters();
				parameters.setPreviewSize(screenWidth, screenHeight);    // 设置预览照片的大小
				parameters.setPreviewFpsRange(20,30);                    // 每秒显示20~30帧
				parameters.setPictureFormat(ImageFormat.NV21);           // 设置图片格式
				parameters.setPictureSize(screenWidth, screenHeight);    // 设置照片的大小
				//camera.setParameters(parameters);                      // android2.3.3以后不需要此行代码
				camera.setPreviewDisplay(surfaceHolder);                 // 通过SurfaceView显示取景画面
				camera.setPreviewCallback(new StreamIt(ipname));         // 设置回调的类
				camera.startPreview();                                   // 开始预览
				camera.autoFocus(null);                                  // 自动对焦
			} catch (Exception e) {
				e.printStackTrace();
			}
			isPreview = true;
		}
	}

}

class StreamIt implements Camera.PreviewCallback {
	private String ipname;
	public StreamIt(String ipname){
		this.ipname = ipname;
	}

	@Override
	public void onPreviewFrame(byte[] data, Camera camera) {
		Size size = camera.getParameters().getPreviewSize();
		try{
			//调用image.compressToJpeg()将YUV格式图像数据data转为jpg格式
			YuvImage image = new YuvImage(data, ImageFormat.NV21, size.width, size.height, null);
			if(image!=null){
				ByteArrayOutputStream outstream = new ByteArrayOutputStream();
				image.compressToJpeg(new Rect(0, 0, size.width, size.height), 80, outstream);
				outstream.flush();
				//启用线程将图像数据发送出去
				Thread th = new MyThread(outstream,ipname);
				th.start();
			}
		}catch(Exception ex){
			Log.e("Sys","Error:"+ex.getMessage());
		}
	}
}

class MyThread extends Thread{
	private byte byteBuffer[] = new byte[1024];
	private OutputStream outsocket;
	private ByteArrayOutputStream myoutputstream;
	private String ipname;

	public MyThread(ByteArrayOutputStream myoutputstream,String ipname){
		this.myoutputstream = myoutputstream;
		this.ipname = ipname;
		try {
			myoutputstream.close();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}

	public void run() {
		try{
			//将图像数据通过Socket发送出去
			Socket tempSocket = new Socket(ipname, 6000);
			outsocket = tempSocket.getOutputStream();
			ByteArrayInputStream inputstream = new ByteArrayInputStream(myoutputstream.toByteArray());
			int amount;
			while ((amount = inputstream.read(byteBuffer)) != -1) {
				outsocket.write(byteBuffer, 0, amount);
			}
			myoutputstream.flush();
			myoutputstream.close();
			tempSocket.close();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}

}

第二部分: GetIP.java

package org.wanghai.CameraTest;

import android.app.Activity;
import android.app.AlertDialog;
import android.app.AlertDialog.Builder;
import android.content.DialogInterface;
import android.content.DialogInterface.OnClickListener;
import android.content.Intent;
import android.os.Bundle;
import android.view.Window;
import android.view.WindowManager;
import android.widget.EditText;
import android.widget.TableLayout;

public class GetIP extends Activity {
	String ipname = null;
	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		// 设置全屏
		requestWindowFeature(Window.FEATURE_NO_TITLE);
		getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN);
		setContentView(R.layout.main);

		final Builder builder = new AlertDialog.Builder(this);   //定义一个AlertDialog.Builder对象
		builder.setTitle("登录服务器对话框");                          // 设置对话框的标题

		//装载/res/layout/login.xml界面布局
		TableLayout loginForm = (TableLayout)getLayoutInflater().inflate( R.layout.login, null);
		final EditText iptext = (EditText)loginForm.findViewById(R.id.ipedittext);
		builder.setView(loginForm);                              // 设置对话框显示的View对象
		// 为对话框设置一个“登录”按钮
		builder.setPositiveButton("登录"
				// 为按钮设置监听器
				, new OnClickListener() {
					@Override
					public void onClick(DialogInterface dialog, int which) {
						//此处可执行登录处理
						ipname = iptext.getText().toString().trim();
						Bundle data = new Bundle();
						data.putString("ipname",ipname);
						Intent intent = new Intent(GetIP.this,CameraTest.class);
						intent.putExtras(data);
						startActivity(intent);
					}
				});
		// 为对话框设置一个“取消”按钮
		builder.setNegativeButton("取消"
				,  new OnClickListener()
				{
					@Override
					public void onClick(DialogInterface dialog, int which)
					{
						//取消登录,不做任何事情。
						System.exit(1);
					}
				});
		//创建、并显示对话框
		builder.create().show();
	}
}


至此Android手机端已经完成,接下来是PC端代码,仅包含一个

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.*;
import javax.imageio.*;
import javax.swing.*;
import java.net.Socket;
import java.net.ServerSocket;

/**
*在服务器开启情况下,启动客户端,创建套接字接收图像
*/

public class ImageServer {	
    public static ServerSocket ss = null;
    
    public static void main(String args[]) throws IOException{    
    	ss = new ServerSocket(6000);
        
        final ImageFrame frame = new ImageFrame(ss);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.setVisible(true);
       
        while(true){
        	frame.panel.getimage();
            frame.repaint();
        }        
    }
       
}

/** 
    A frame with an image panel
*/
@SuppressWarnings("serial")
class ImageFrame extends JFrame{
	public ImagePanel panel;
	public JButton jb;
   
    public ImageFrame(ServerSocket ss){
   	    // get screen dimensions   	   
   	    Toolkit kit = Toolkit.getDefaultToolkit();
        Dimension screenSize = kit.getScreenSize();
        int screenHeight = screenSize.height;
        int screenWidth = screenSize.width;

        // center frame in screen
        setTitle("ImageTest");
        setLocation((screenWidth - DEFAULT_WIDTH) / 2, (screenHeight - DEFAULT_HEIGHT) / 2);
        setSize(DEFAULT_WIDTH, DEFAULT_HEIGHT);

        // add panel to frame
        this.getContentPane().setLayout(null);
        panel = new ImagePanel(ss);
        panel.setSize(640,480);
        panel.setLocation(0, 0);
        add(panel);
        jb = new JButton("拍照");
        jb.setBounds(0,480,640,50);
        add(jb);
        saveimage saveaction = new saveimage(ss);
        jb.addActionListener(saveaction);
    }

    public static final int DEFAULT_WIDTH = 640;
    public static final int DEFAULT_HEIGHT = 560;  
}

/**
   A panel that displays a tiled image
*/
@SuppressWarnings("serial")
class ImagePanel extends JPanel {     
    private ServerSocket ss;
    private Image image;
    private InputStream ins;
	Boolean a = true;
	 
    public ImagePanel(ServerSocket ss) {  
	    this.ss = ss;
    }
    
    public void getimage() throws IOException{
    	Socket s = this.ss.accept();
		if((a==true)&&(s != null)){
			
			System.out.println("连接成功!");
			
			javax.swing.JOptionPane.showMessageDialog(new javax.swing.JFrame(),
                    "已经连接成功", "提示!", javax.swing.JOptionPane.PLAIN_MESSAGE);
					a = false;
			
		}
		else if((a==true)&&(s == null)){
			javax.swing.JOptionPane.showMessageDialog(new javax.swing.JFrame(),
                    "等待连接中", "提示!", javax.swing.JOptionPane.PLAIN_MESSAGE);
		}
		
        
        this.ins = s.getInputStream();
		this.image = ImageIO.read(ins);
		this.ins.close();
    }
   
    public void paintComponent(Graphics g){  
        super.paintComponent(g);    
        if (image == null) return;
        g.drawImage(image, 0, 0, null);
    }

}

class saveimage implements ActionListener {
	RandomAccessFile inFile = null;
	byte byteBuffer[] = new byte[1024];
	InputStream ins;
	private ServerSocket ss;
	
	public saveimage(ServerSocket ss){
		this.ss = ss;
	}
	
	public void actionPerformed(ActionEvent event){
        try {
			Socket s = ss.accept();
			ins = s.getInputStream();
			
			// 文件选择器以当前的目录打开
	        JFileChooser jfc = new JFileChooser(".");
	        jfc.showSaveDialog(new javax.swing.JFrame());
	        // 获取当前的选择文件引用
	        File savedFile = jfc.getSelectedFile();
	        
	        // 已经选择了文件
	        if (savedFile != null) {
	            // 读取文件的数据,可以每次以快的方式读取数据
	            try {
					inFile = new RandomAccessFile(savedFile, "rw");
				} catch (FileNotFoundException e) {
					e.printStackTrace();
				}
	        }

            int amount;
            while ((amount = ins.read(byteBuffer)) != -1) {
                inFile.write(byteBuffer, 0, amount);
            }
            inFile.close();
            ins.close();
            s.close();
            javax.swing.JOptionPane.showMessageDialog(new javax.swing.JFrame(),
                    "已接保存成功", "提示!", javax.swing.JOptionPane.PLAIN_MESSAGE);
		} catch (IOException e) {

			e.printStackTrace();
		}
	}
}

PS:PC端使用操作说明:

1.比较了解编程的 可以使用CMD命令进行操作,进入指定文件目录执行CMD命令: javac ImageServer.java生成 class文件

2.如果不太熟悉,本人已经做好了.bat文件,可直接执行操作,不过操作之前得先设置好JAVA编译的相关环境。

这里补充一下批处理的相关知识点:

-------------快速将.java编译成.class的批处理文件------------  

方法如下:  

1、在F盘新建一个F:\study\test\puzzle的文件夹。  

2、在F:\study\test文件夹里新建一个记事本,打开,输入: set CLASSPATH=C:\Program Files\Java\jdk1.7.0_25;.;%JAVA_HOME%\lib; javac -d F:\study\test\puzzle %1 pause 然后将其保存为JavaCompiler.txt文件。说明: set CLASSPATH=C:\Program Files\Java\jdk1.7.0_25;.;%JAVA_HOME%\lib; 是我的电脑的环境变量。 F:\study\test\puzzle 是快速将.java编译成.class后所要保存的路径和文件夹。 %1 是被鼠标拖到批处理图标上的java文件名。 pause的作用是控制cmd暂停运行,方便用户有时间读取cmd图形界面上的信息。  

3、将JavaCompiler.txt另存为JavaCompiler.bat批处理文件,并保存在F:\study\test或F:\study\test\puzzle文件夹。  

4、将事先制作好的.java导入F:\study\test或F:\study\test\puzzle文件夹。  

5、用鼠标将.java文件拖到JavaCompiler.bat批处理文件的图标上,松开,批处理文件开始运行,如果顺利的话,很快就能编译生成同名.class文件,并被保存到F:\study\test\puzzle或F:\study\test文件夹里。  

6、打开F:\study\test\puzzle或F:\study\test,你会发现生成的扩展名为.class的java文件。


3.关于部分人反映在Android 客户端输入IP后,程序没有进行摄像监控的问题的一些解决方法。

WIN+R(进入DOS界面

--------输入cmd  

--------输入 ipconfig/all 查看IPv4的地址即可。 


4. 还有部分人提出,部分手机不能运行的问题的解决方法。

注意Android 2.3.3以下的Android手机要initCamera()函数里添加多一条:camera.setParameters(parameters); 否则手机有可能会出现**显示小**或者**显示不出画面**的情况。

本人创建的Android兴趣交流群群里有大量资料可供大家学习。

QQ交流群:580535711

附件:

项目资料



  • 3
    点赞
  • 36
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,你的问题是关于Python函数第二次运行报错和retry的简单用法。 首先,如果一个Python函数在第二次运行时出现了错误,可能是由于函数内部的变量状态发生了变化,导致程序出错。为了解决这个问题,可以使用retry功能来多次尝试运行该函数,直到它成功为止。 使用retry功能可以在函数出现错误时自动重试,可以避免手动重复运行该函数,从而提高效率。 下面是一个简单的示例代码,演示了如何使用retry功能: ```python import time from retry import retry @retry(delay=1, backoff=2, max_delay=4) def my_func(): print("Running my_func...") # 模拟函数出错 if time.time() % 2 == 0: raise ValueError("Oops! Something went wrong...") else: print("Success!") my_func() ``` 在这个示例代码中,我们使用了retry库中的retry装饰器来装饰my_func函数,该函数会运行多次,直到成功为止。在retry装饰器中,我们设置了最大延迟时间为4秒,每次重试前等待1秒,重试间隔时间呈指数级增长,即第一次重试后等待1秒,第二次重试后等待2秒,第三次重试后等待4秒,以此类推,直到达到最大延迟时间为止。 当my_func函数第一次运行时,它会输出“Running my_func...”和“Success!”,并正常结束。当函数第二次运行时,它会抛出一个ValueError异常,并重试多次,直到成功为止。 希望这个简单的示例可以帮助你理解如何使用retry功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值