Android和C#基于UDP的实时图像传输

本文介绍了一个使用UDP进行实时图像传输的项目,包括Android客户端和C#服务端的实现细节。客户端通过摄像头捕获图像并将其转换为JPEG格式,然后通过UDP发送给服务端。服务端接收这些图像帧并在PictureBox中显示。
摘要由CSDN通过智能技术生成

实时图像传输的话还是用UDP比较好,速度比TCP快,反正丢一些帧也没有关系


照例先上图

电脑端


手机端



项目:http://pan.baidu.com/s/1pLrYrij


Android端

首先修改AndroidManifest.xml文件

添加这两个权限

<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.CAMERA"/>

然后把MainActiviy设置为横屏

android:screenOrientation="landscape"


然后修改activity_main.xml文件

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <SurfaceView
        android:id="@+id/surfaceView"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</RelativeLayout>

很简单,只是一个SurfaceView


最后就是MainActivity.java了

package com.ffpy.cameratransmitclient;

import android.graphics.ImageFormat;
import android.graphics.Rect;
import android.graphics.YuvImage;
import android.hardware.Camera;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.Socket;
import java.util.LinkedList;

public class MainActivity extends AppCompatActivity implements SurfaceHolder.Callback, Camera.PreviewCallback {

    private final int TCP_PORT = 3333;                  //TCP通讯的端口号
    private final int UDP_PORT = 4444;                  //UDP通讯的端口号
    private final String SERVER_IP = "192.168.1.104";   //服务器端的IP地址

    private Camera mCamera;
    private Camera.Size previewSize;            //预览图像的宽高
    private DatagramSocket packetSenderSocket;  //发送图像帧的套接字
    private long lastSendTime;                  //上一次发送图像帧的时间
    private InetAddress serverAddress;          //服务端地址
    private final LinkedList<DatagramPacket> packetList = new LinkedList<>();   //图像数据包队列

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        final SurfaceView surfaceView = (SurfaceView) findViewById(R.id.surfaceView);
        SurfaceHolder holder = surfaceView.getHolder();
        holder.setKeepScreenOn(true);  //保持屏幕常亮
        holder.addCallback(this);

        //开启通讯连接线程,连接服务端
        new ConnectThread().start();
    }

    @Override
    public void surfaceCreated(SurfaceHolder holder) {
        //获取相机
        if (mCamera == null) {
            mCamera = Camera.open();    //打开后摄像头
            Camera.Parameters parameters = mCamera.getParameters();
            //设置预览图大小
            //注意必须为parameters.getSupportedPreviewSizes()中的长宽,否则会报异常
            parameters.setPreviewSize(960, 544);
            previewSize = parameters.getPreviewSize();
            //设置自动对焦
            parameters.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_PICTURE);
            mCamera.setParameters(parameters);
            mCamera.cancelAutoFocus();
            //设置回调
            try {
                mCamera.setPreviewDisplay(holder);
                mCamera.setPreviewCallback(this);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        //开始预览
        mCamera.startPreview();
    }

    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {

    }

    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {
        //释放相机
        if (mCamera != null) {
            mCamera.setPreviewCallback(null);
            mCamera.stopPreview();
            mCamera.release();
            mCamera = null;
        }
    }

    /**
     * 获取每一帧的图像数据
     */
    @Override
    public void onPreviewFrame(byte[] data, Camera camera) {
        long curTime = System.currentTimeMillis();
        //每20毫秒发送一帧
        if (serverAddress != null && curTime - lastSendTime >= 20) {
            lastSendTime = curTime;
            //NV21格式转JPEG格式
            YuvImage image = new YuvImage(data, ImageFormat.NV21, previewSize.width ,previewSize.height, null);
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            image.compressToJpeg(new Rect(0, 0, previewSize.width, previewSize.height), 40, bos);
            int packMaxSize = 65500;    //防止超过UDP包的最大大小
            byte[] imgBytes = bos.toByteArray();
            Log.i("tag", imgBytes.length + "");
            //打包
            DatagramPacket packet = new DatagramPacket(imgBytes, imgBytes.length > packMaxSize ? packMaxSize : imgBytes.length,
                    serverAddress, UDP_PORT);
            //添加到队尾
            synchronized (packetList) {
                packetList.addLast(packet);
            }
        }
    }

    /**
     * 连接线程
     */
    private class ConnectThread extends Thread {
        @Override
        public void run() {
            super.run();
            try {
                //创建连接
                packetSenderSocket = new DatagramSocket();
                Socket socket = new Socket(SERVER_IP, TCP_PORT);
                serverAddress = socket.getInetAddress();
                //断开连接
                socket.close();
                //启动发送图像数据包的线程
                new ImgSendThread().start();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    /**
     * 发送图像数据包的线程
     */
    private class ImgSendThread extends Thread {
        @Override
        public void run() {
            super.run();
            while (packetSenderSocket != null) {
                DatagramPacket packet;
                synchronized (packetList) {
                    //没有待发送的包
                    if (packetList.isEmpty()) {
                        try {
                            Thread.sleep(10);
                            continue;
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                    //取出队头
                    packet = packetList.getFirst();
                    packetList.removeFirst();
                }
                try {
                    packetSenderSocket.send(packet);
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
surfaceCreated()、surfaceChanged()、surfaceDestroyed()这三个方法是对应SurfaceView的,做过Android自定义相机的都知道的。需要注意的是,在surfaceCreated()中我设置了预览图像的宽高

parameters.setPreviewSize(960, 544);
这个值我根据我手机屏幕的大小设置的,因为预览图会在SurfaceView上拉伸成SurfaceView的宽高,所以设置成跟SurfaceView的宽高比差不多,这样看上去图像不会变形。
而且它不能随便设置,设置得不对的话会报异常,可以通过parameters.getSupportedPreviewSizes()来查看手机支持的宽高列表,不同的手机所支持的不一定相同,所以如果在这里报错的话可以修改成你手机所支持的值。

onPreviewFrame()就是获取每一帧图像的地方了,在这里获取摄像头的图像帧,并转为JPEG格式。还有因为UDP每个包的大小不能超过64K,减去一下包头那些所占用的,大概还有65500个字节。因为我这里是每一个包封装一帧图像,所以要注意压缩图像,不要超过65500个字节,否则在电脑端就看不到超出的那部分图像了。


C#

窗体很简单,就是一个PictureBox放在Form上,把PictrureBox的Name设置为pictureBox,并让它充满Form。设置Form的大小为978, 591,这样pictureBox的大小就是960, 544了,跟预览图的大小一致


Form1.cs

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.Drawing;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace CameraTransmitServer
{
    public partial class Form1 : Form
    {
        private const int TCP_PORT = 3333;  //TCP通讯的端口号
        private const int UDP_PORT = 4444;  //UDP通讯的端口号

        private UdpClient client;
        private IPEndPoint remote;

        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            //创建等待连接线程
            Thread thread = new Thread(new ThreadStart(waitConnect));
            thread.IsBackground = true;
            thread.Start();
        }

        //等待连接
        private void waitConnect()
        {
            Socket serverSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
            serverSocket.Bind(new IPEndPoint(IPAddress.Any, TCP_PORT));
            serverSocket.Listen(10);
            Debug.WriteLine("监听中");
            Socket clientSocket = serverSocket.Accept();
            Debug.WriteLine("连接成功");

            remote = (IPEndPoint) clientSocket.RemoteEndPoint;
            client = new UdpClient(new IPEndPoint(IPAddress.Any, UDP_PORT));
             
            //关闭套接字
            serverSocket.Close();
            clientSocket.Close();

            //启动接收线程
            Thread thread = new Thread(new ThreadStart(recvImage));
            thread.IsBackground = true;
            thread.Start();
        }

        //接收图像帧,并显示到PictureBox上
        private void recvImage()
        {
            while(true) 
            {
                //接受图像帧数据
                byte[] recvBuf = client.Receive(ref remote);
                MemoryStream ms = new MemoryStream(recvBuf);
                try
                {
                    //显示到pictureBox上
                    pictureBox.Image = Image.FromStream(ms);
                }
                catch (ArgumentException)
                { }
            }
        }
    }
}

这里就是等待手机端连接然后不断地接收图像数据并显示在PictureBox上


你可能会问,不是UDP吗,怎么这里还用了TCP?

我这里用TCP是为了保证连接的建立,确认连接成功后再通过UDP发送图像数据,否则的话服务端还没上线手机就开始不断地发送图像数据了。而且如果需要的话还可以用TCP的可靠连接来发送除了图像以外的其它数据,比如文字什么的。

这两天下了一个使用UDP传输目录内文件的程序,发出来给大家一起看看,共同进步。有问题请指教。 由于udp丢包比较厉害,因此使用了自定义的内部协议,进行双方的确认。 程序跑起来后,看网络状况,有时候会一卡一卡的。 以下是程序说明: * 本程序集成了数据导出端(服务器端)和数据导入端(客户端),使用UDP进行文件传递 * 服务器端的文件来源目录,见Tools中SOURCEPATH的设置 * 客户端的文件保存目录,见Tools中DESTINATIONPATH的设置,可以根据自己需要进行调整 * * 由于UDP存在丢包问题,因此Server和Client的通讯需要来回包进行确认,协议包头如下: * 1. "55 aa 99 01",表示客户端发起广播请求,请求服务器响应 * 2. "55 aa 99 02 + 服务器的设备名称",表示服务器接收到广播后,响应客户端请求,把此包指定IP发送客户端(此指定IP地址可以UDP广播信息包中获取) * 3. "55 aa 99 03",表示客户端接收到服务器的响应,接着向服务器指定IP请求:需要传递的文件总数目和文件总容量(单位为KB) * 4. "55 aa 99 04 + 4字节文件总数目和4字节的文件总容量",表示服务器接收到客户端的0x03请求,统计SOURCEPATH中的所有文件数目和文件总容量,发送指定IP地址的客户端 * 5. "55 aa 99 05",客户端接收到文件总数目和文件总容量,请求服务器发送文件具体内容 * 6. "55 aa 99 10 + 文件名称",服务器发送文件名称 * 7. "55 aa 99 11",客户端响应,表示接收到服务器发送的0x10包 * 8. "55 aa 99 12+文件内容",服务器端发送具体文件内容 * 9. "55 aa 99 13",客户端响应,表示接收到服务器发送的0x12包 * 10."55 aa 99 14",服务端高速客户端发送完毕 * * 注意:服务器发送0x10包后,收到客户端的0x11响应包,将把文件具体内容拆分成N个0x12包,每个包的大小见Tools.byteSize的设置,目前设置为10K, * 服务器没收到一个0x13响应包,才能继续发下一个0x12包,已放置UDP的丢包,另外每个0x12包最多发送10次而无0x13包的响应,则发送进程结束,界面提示 * * 本程序已经封装好,调用见TransportFilesActivity.java文件
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值