Android VideoView RTSP 延迟

Posted

技术标签:

【中文标题】Android VideoView RTSP 延迟【英文标题】:Android VideoView RTSP delay 【发布时间】:2012-05-17 14:13:29 【问题描述】:

我正在通过 VideoView 在我的 android 应用程序中打开 IP 摄像机的 RTSP 流。有个问题是延迟很大,达到20多秒!虽然通过浏览器从普通 PC 上查看相机,但没有这样的延迟。请问有什么想法吗?

我的代码:

        String path="rtsp://192.168.1.20/3gpp";
        myVideoView.setVideoURI(Uri.parse(path));
        myVideoView.setMediaController(new MediaController(this));
        myVideoView.requestFocus();
        myVideoView.start();

【问题讨论】:

这篇文章可能会有所帮助:***.com/questions/3937241/reduce-video-buffering 【参考方案1】:

使用 MJPEG 流,而不是 RTSP。经过大量研究,我发现了一个 MJPEG 查看器类,可以近乎实时地播放直播(取决于您的网络连接)。

这是代码Android and MJPEG

【讨论】:

【参考方案2】:

您可以为此执行以下操作

MjpegSample 类

package de.mjpegsample;

import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.Window;
import android.view.WindowManager;
import de.mjpegsample.MjpegView.MjpegInputStream;
import de.mjpegsample.MjpegView.MjpegView;

public class MjpegSample extends Activity 

private MjpegView mv;
private static final int MENU_QUIT = 1;

/* Creates the menu items */
public boolean onCreateOptionsMenu(Menu menu)     
menu.add(0, MENU_QUIT, 0, "Quit");
return true;


/* Handles item selections */
public boolean onOptionsItemSelected(MenuItem item)     
    switch (item.getItemId()) 
        case MENU_QUIT:
            finish();
            return true;    
            
    return false;


public void onCreate(Bundle icicle) 
    super.onCreate(icicle);
    //sample public cam
    String URL = "http://gamic.dnsalias.net:7001/img/video.mjpeg";

    requestWindowFeature(Window.FEATURE_NO_TITLE);
    getWindow().setFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN, WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);
    mv = new MjpegView(this);
    setContentView(mv);        
    mv.setSource(MjpegInputStream.read(URL));
    mv.setDisplayMode(MjpegView.SIZE_BEST_FIT);
    mv.showFps(false);


public void onPause() 
    super.onPause();
    mv.stopPlayback();


Mjepg 视图

import java.io.IOException;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.PorterDuff;
import android.graphics.PorterDuffXfermode;
import android.graphics.Rect;
import android.graphics.Typeface;
import android.util.AttributeSet;
import android.view.SurfaceHolder;
import android.view.SurfaceView;

public class MjpegView extends SurfaceView implements SurfaceHolder.Callback 
public final static int POSITION_UPPER_LEFT  = 9;
public final static int POSITION_UPPER_RIGHT = 3;
public final static int POSITION_LOWER_LEFT  = 12;
public final static int POSITION_LOWER_RIGHT = 6;

public final static int SIZE_STANDARD   = 1; 
public final static int SIZE_BEST_FIT   = 4;
public final static int SIZE_FULLSCREEN = 8;

private MjpegViewThread thread;
private MjpegInputStream mIn = null;    
private boolean showFps = false;
private boolean mRun = false;
private boolean surfaceDone = false;    
private Paint overlayPaint;
private int overlayTextColor;
private int overlayBackgroundColor;
private int ovlPos;
private int dispWidth;
private int dispHeight;
private int displayMode;

public class MjpegViewThread extends Thread 
    private SurfaceHolder mSurfaceHolder;
    private int frameCounter = 0;
    private long start;
    private Bitmap ovl;

    public MjpegViewThread(SurfaceHolder surfaceHolder, Context context)  mSurfaceHolder = surfaceHolder; 

    private Rect destRect(int bmw, int bmh) 
        int tempx;
        int tempy;
        if (displayMode == MjpegView.SIZE_STANDARD) 
            tempx = (dispWidth / 2) - (bmw / 2);
            tempy = (dispHeight / 2) - (bmh / 2);
            return new Rect(tempx, tempy, bmw + tempx, bmh + tempy);
        
        if (displayMode == MjpegView.SIZE_BEST_FIT) 
            float bmasp = (float) bmw / (float) bmh;
            bmw = dispWidth;
            bmh = (int) (dispWidth / bmasp);
            if (bmh > dispHeight) 
                bmh = dispHeight;
                bmw = (int) (dispHeight * bmasp);
            
            tempx = (dispWidth / 2) - (bmw / 2);
            tempy = (dispHeight / 2) - (bmh / 2);
            return new Rect(tempx, tempy, bmw + tempx, bmh + tempy);
        
        if (displayMode == MjpegView.SIZE_FULLSCREEN) return new Rect(0, 0, dispWidth, dispHeight);
        return null;
    

    public void setSurfaceSize(int width, int height) 
        synchronized(mSurfaceHolder) 
            dispWidth = width;
            dispHeight = height;
        
    

    private Bitmap makeFpsOverlay(Paint p, String text) 
        Rect b = new Rect();
        p.getTextBounds(text, 0, text.length(), b);
        int bwidth  = b.width()+2;
        int bheight = b.height()+2;
        Bitmap bm = Bitmap.createBitmap(bwidth, bheight, Bitmap.Config.ARGB_8888);
        Canvas c = new Canvas(bm);
        p.setColor(overlayBackgroundColor);
        c.drawRect(0, 0, bwidth, bheight, p);
        p.setColor(overlayTextColor);
        c.drawText(text, -b.left+1, (bheight/2)-((p.ascent()+p.descent())/2)+1, p);
        return bm;           
    

    public void run() 
        start = System.currentTimeMillis();
        PorterDuffXfermode mode = new PorterDuffXfermode(PorterDuff.Mode.DST_OVER);
        Bitmap bm;
        int width;
        int height;
        Rect destRect;
        Canvas c = null;
        Paint p = new Paint();
        String fps = "";
        while (mRun) 
            if(surfaceDone) 
                try 
                    c = mSurfaceHolder.lockCanvas();
                    synchronized (mSurfaceHolder) 
                        try 
                            bm = mIn.readMjpegFrame();
                            destRect = destRect(bm.getWidth(),bm.getHeight());
                            c.drawColor(Color.BLACK);
                            c.drawBitmap(bm, null, destRect, p);
                            if(showFps) 
                                p.setXfermode(mode);
                                if(ovl != null) 
                                    height = ((ovlPos & 1) == 1) ? destRect.top : destRect.bottom-ovl.getHeight();
                                    width  = ((ovlPos & 8) == 8) ? destRect.left : destRect.right -ovl.getWidth();
                                    c.drawBitmap(ovl, width, height, null);
                                
                                p.setXfermode(null);
                                frameCounter++;
                                if((System.currentTimeMillis() - start) >= 1000) 
                                    fps = String.valueOf(frameCounter)+"fps";
                                    frameCounter = 0; 
                                    start = System.currentTimeMillis();
                                    ovl = makeFpsOverlay(overlayPaint, fps);
                                
                            
                         catch (IOException e) 
                    
                 finally  if (c != null) mSurfaceHolder.unlockCanvasAndPost(c); 
            
        
    


private void init(Context context) 
    SurfaceHolder holder = getHolder();
    holder.addCallback(this);
    thread = new MjpegViewThread(holder, context);
    setFocusable(true);
    overlayPaint = new Paint();
    overlayPaint.setTextAlign(Paint.Align.LEFT);
    overlayPaint.setTextSize(12);
    overlayPaint.setTypeface(Typeface.DEFAULT);
    overlayTextColor = Color.WHITE;
    overlayBackgroundColor = Color.BLACK;
    ovlPos = MjpegView.POSITION_LOWER_RIGHT;
    displayMode = MjpegView.SIZE_STANDARD;
    dispWidth = getWidth();
    dispHeight = getHeight();


public void startPlayback()  
    if(mIn != null) 
        mRun = true;
        thread.start();         
    


public void stopPlayback()  
    mRun = false;
    boolean retry = true;
    while(retry) 
        try 
            thread.join();
            retry = false;
         catch (InterruptedException e) 
    


public MjpegView(Context context, AttributeSet attrs)  super(context, attrs); init(context); 
public void surfaceChanged(SurfaceHolder holder, int f, int w, int h)  thread.setSurfaceSize(w, h); 

public void surfaceDestroyed(SurfaceHolder holder)  
    surfaceDone = false; 
    stopPlayback(); 


public MjpegView(Context context)  
    super(context); 
    init(context); 
        
public void surfaceCreated(SurfaceHolder holder)  
    surfaceDone = true; 
    
public void showFps(boolean b)  
    showFps = b; 
    
public void setSource(MjpegInputStream source)  
    mIn = source; 
    startPlayback();
    
public void setOverlayPaint(Paint p)  
    overlayPaint = p; 
    
public void setOverlayTextColor(int c)  
    overlayTextColor = c; 
    
public void setOverlayBackgroundColor(int c)  
    overlayBackgroundColor = c; 
    
public void setOverlayPosition(int p)  
    ovlPos = p; 
    
public void setDisplayMode(int s)  
    displayMode = s; 
    

MjpegInputStream

package de.mjpegsample.MjpegView;

import java.io.BufferedInputStream;
import java.io.ByteArrayInputStream;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.util.Properties;

import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;

import android.graphics.Bitmap;
import android.graphics.BitmapFactory;

public class MjpegInputStream extends DataInputStream 
private final byte[] SOI_MARKER =  (byte) 0xFF, (byte) 0xD8 ;
private final byte[] EOF_MARKER =  (byte) 0xFF, (byte) 0xD9 ;
private final String CONTENT_LENGTH = "Content-Length";
private final static int HEADER_MAX_LENGTH = 100;
private final static int FRAME_MAX_LENGTH = 40000 + HEADER_MAX_LENGTH;
private int mContentLength = -1;

public static MjpegInputStream read(String url) 
    HttpResponse res;
    DefaultHttpClient httpclient = new DefaultHttpClient();     
    try 
        res = httpclient.execute(new HttpGet(URI.create(url)));
        return new MjpegInputStream(res.getEntity().getContent());              
     catch (ClientProtocolException e) 
     catch (IOException e) 
    return null;


public MjpegInputStream(InputStream in)  super(new BufferedInputStream(in, FRAME_MAX_LENGTH)); 

private int getEndOfSeqeunce(DataInputStream in, byte[] sequence) throws IOException 
    int seqIndex = 0;
    byte c;
    for(int i=0; i < FRAME_MAX_LENGTH; i++) 
        c = (byte) in.readUnsignedByte();
        if(c == sequence[seqIndex]) 
            seqIndex++;
            if(seqIndex == sequence.length) return i + 1;
         else seqIndex = 0;
    
    return -1;


private int getStartOfSequence(DataInputStream in, byte[] sequence) throws IOException 
    int end = getEndOfSeqeunce(in, sequence);
    return (end < 0) ? (-1) : (end - sequence.length);


private int parseContentLength(byte[] headerBytes) throws IOException, NumberFormatException 
    ByteArrayInputStream headerIn = new ByteArrayInputStream(headerBytes);
    Properties props = new Properties();
    props.load(headerIn);
    return Integer.parseInt(props.getProperty(CONTENT_LENGTH));
   

public Bitmap readMjpegFrame() throws IOException 
    mark(FRAME_MAX_LENGTH);
    int headerLen = getStartOfSequence(this, SOI_MARKER);
    reset();
    byte[] header = new byte[headerLen];
    readFully(header);
    try 
        mContentLength = parseContentLength(header);
     catch (NumberFormatException nfe)  
        mContentLength = getEndOfSeqeunce(this, EOF_MARKER); 
    
    reset();
    byte[] frameData = new byte[mContentLength];
    skipBytes(headerLen);
    readFully(frameData);
    return BitmapFactory.decodeStream(new ByteArrayInputStream(frameData));


【讨论】:

以上是关于Android VideoView RTSP 延迟的主要内容,如果未能解决你的问题,请参考以下文章

Android videoview不播放rtsp

如何使用 videoview 播放带有身份验证的 rtsp 流?

将RTSP流保存到android中的mp4文件

编辑 android VideoView 帧

在 VideoView 中播放流,将 url 转换为 rtsp

视频的 RTSP 网址在 VideoView 中不起作用