nginx服务器搭建完成以后,就可以着手客户端的开发了,首先是直播端,这一节主要是说明音频和视频数据的采集
视频数据采集思路
调用摄像头,前置和后置都需要,因此需要摄像头切换
视频开始直播和结束直播,需要在本地获得预览,选择surfaceview进行绘制
将获取到的视频信息传递给native,在底层进行视频推流
因此,视频数据采集只需要调用Android自带方法就可完成
音频数据采集思路
调用麦克风,采集收到的数据
当开始直播的时候将数据传递给native层进行推流
因此,音频推流也只需要调用Android自带方法即可
布局实现
从分析来看,需要两个按钮,一个发送开始结束信号,一个切换摄像头
还有一个SurfaceView布局1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<android.view.SurfaceView
android:id="@+id/surface_view"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_gravity="center" />
<Button
android:id="@+id/btn_camera_switch"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_alignParentLeft="true"
android:gravity="center"
android:text="切换摄像头"
android:onClick="mSwitchCamera"/>
<Button
android:id="@+id/btn_push"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_alignParentRight="true"
android:gravity="center"
android:text="开始直播"
android:onClick="mStartLive" />
</RelativeLayout>
Java文件
由于要传递音视频参数,这实现两个GetSet类AudioParam.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27public class AudioParam {
private int sampleRateInHz = 44100;
private int channel = 1;
public AudioParam(int sampleRateInHz, int channel) {
super();
this.sampleRateInHz = sampleRateInHz;
this.channel = channel;
}
public int getSampleRateInHz() {
return sampleRateInHz;
}
public void setSampleRateInHz(int sampleRateInHz) {
this.sampleRateInHz = sampleRateInHz;
}
public int getChannel() {
return channel;
}
public void setChannel(int channel) {
this.channel = channel;
}
}
VideoParam.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37public class VideoParam {
private int width;
private int height;
private int cameraId;
public VideoParam(int width, int height, int cameraId) {
super();
this.width = width;
this.height = height;
this.cameraId = cameraId;
}
public int getWidth() {
return width;
}
public void setWidth(int width) {
this.width = width;
}
public int getHeight() {
return height;
}
public void setHeight(int height) {
this.height = height;
}
public int getCameraId() {
return cameraId;
}
public void setCameraId(int cameraId) {
this.cameraId = cameraId;
}
}
定义抽象类Pusher,让Video和Audio去实现其方法,其方法包括开始,结束,释放资源Pusher.java
1
2
3
4
5
6
7
8
9public abstract class Pusher {
public abstract void startPush();
public abstract void stopPush();
public abstract void release();
}
Video实现类VideoPusher.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98import java.io.IOException;
import com.cj5785.livegetvideoaudio.params.VideoParam;
import android.hardware.Camera;
import android.hardware.Camera.CameraInfo;
import android.hardware.Camera.PreviewCallback;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
public class VideoPusher extends Pusher implements Callback, PreviewCallback{
private Camera mCamera;
private boolean isPushing = false;
private SurfaceHolder surfaceHolder;
private VideoParam videoParam;
private byte[] buffers;
public VideoPusher(SurfaceHolder surfaceHolder, VideoParam videoParam) {
this.surfaceHolder = surfaceHolder;
this.videoParam = videoParam;
surfaceHolder.addCallback(this);
}
public void startPush() {
isPushing = true;
}
public void stopPush() {
isPushing = false;
}
public void release() {
stopPreview();
}
public void switchCamera() {
if(videoParam.getCameraId() == CameraInfo.CAMERA_FACING_BACK){
videoParam.setCameraId(CameraInfo.CAMERA_FACING_FRONT);
}else{
videoParam.setCameraId(CameraInfo.CAMERA_FACING_BACK);
}
//重新预览
stopPreview();
startPreview();
}
private void startPreview() {
try {
//SurfaceView初始化完成,开始相机预览
mCamera = Camera.open(videoParam.getCameraId());
mCamera.setPreviewDisplay(surfaceHolder);
//获取预览图像数据
buffers = new byte[videoParam.getWidth() * videoParam.getHeight() * 5 * 10];
mCamera.addCallbackBuffer(buffers);
mCamera.setPreviewCallbackWithBuffer(this);
mCamera.startPreview();
} catch (IOException e) {
e.printStackTrace();
}
}
private void stopPreview() {
if(mCamera != null){
mCamera.stopPreview();
mCamera.release();
mCamera = null;
}
}
public void onPreviewFrame(byte[] data, Camera camera) {
if(mCamera != null){
mCamera.addCallbackBuffer(buffers);
}
if(isPushing){
//回调函数中获取图像数据,然后给Native代码编码
Log.d("cj5785","start video");
}
}
public void surfaceCreated(SurfaceHolder holder) {
startPreview();
}
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {}
public void surfaceDestroyed(SurfaceHolder holder) {}
}
Audio实现类AudioPusher.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66import com.cj5785.livegetvideoaudio.params.AudioParam;
import android.media.AudioFormat;
import android.media.AudioRecord;
import android.media.MediaRecorder.AudioSource;
import android.util.Log;
public class AudioPusher extends Pusher{
private AudioRecord audioRecord;
private boolean isPushing = false;
private int minBufferSize;
public AudioPusher(AudioParam audioParam) {
int channelConfig = audioParam.getChannel() == 1 ?
AudioFormat.CHANNEL_IN_MONO : AudioFormat.CHANNEL_IN_STEREO;
//最小缓冲区大小
minBufferSize = AudioRecord.getMinBufferSize(audioParam.getSampleRateInHz(),
channelConfig, AudioFormat.ENCODING_PCM_16BIT);
audioRecord = new AudioRecord(AudioSource.MIC, audioParam.getSampleRateInHz(),
channelConfig, AudioFormat.ENCODING_PCM_16BIT, minBufferSize);
}
public void startPush() {
isPushing = true;
//启动一个录音子线程
new Thread(new AudioRecordTask()).start();
}
public void stopPush() {
isPushing = false;
audioRecord.stop();
}
public void release() {
if(audioRecord != null){
audioRecord.release();
audioRecord = null;
}
}
class AudioRecordTask implements Runnable{
public void run() {
//开始录音
audioRecord.startRecording();
while(isPushing){
//通过AudioRecord不断读取音频数据
byte[] buffer = new byte[minBufferSize];
int len = audioRecord.read(buffer, 0, buffer.length);
if(len > 0){
//传给Native代码,进行音频编码
Log.d("cj5785","start record");
}
}
}
}
}
另外还需要一个实例化Video和Audio的类LivePusher.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60import com.cj5785.livegetvideoaudio.params.AudioParam;
import com.cj5785.livegetvideoaudio.params.VideoParam;
import android.hardware.Camera.CameraInfo;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
public class LivePusher implements Callback{
private SurfaceHolder surfaceHolder;
private VideoPusher videoPusher;
private AudioPusher audioPusher;
public LivePusher(SurfaceHolder surfaceHolder) {
this.surfaceHolder = surfaceHolder;
surfaceHolder.addCallback(this);
prepare();
}
private void prepare() {
//实例化视频推流器
VideoParam videoParam = new VideoParam(320, 240, CameraInfo.CAMERA_FACING_BACK);
videoPusher = new VideoPusher(surfaceHolder,videoParam);
//实例化音频推流器
AudioParam audioParam = new AudioParam(44100, 1);
audioPusher = new AudioPusher(audioParam);
}
public void switchCamera() {
videoPusher.switchCamera();
}
public void startPush(String url) {
videoPusher.startPush();
audioPusher.startPush();
}
public void stopPush() {
videoPusher.stopPush();
audioPusher.stopPush();
}
private void release() {
videoPusher.release();
audioPusher.release();
}
public void surfaceCreated(SurfaceHolder holder) {}
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {}
public void surfaceDestroyed(SurfaceHolder holder) {
stopPush();
release();
}
}
最后是主活动类MainActivity.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36import com.cj5785.livegetvideoaudio.pusher.LivePusher;
import android.app.Activity;
import android.os.Bundle;
import android.view.SurfaceView;
import android.view.View;
import android.widget.Button;
public class MainActivity extends Activity {
public static final String URL = "";
private LivePusher live;
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
SurfaceView surfaceView = (SurfaceView) findViewById(R.id.surface_view);
live = new LivePusher(surfaceView.getHolder());
}
public void mStartLive(View view) {
Button btn = (Button)view;
if(btn.getText().equals("开始直播")){
live.startPush(URL);
btn.setText("停止直播");
}else{
live.stopPush();
btn.setText("开始直播");
}
}
public void mSwitchCamera(View view) {
live.switchCamera();
}
}
效果查看
在发布到手机上以后,摄像头前后切换正常
在点击开始直播按钮的时候,会不断打印出start video
和start audio
,在数据采集这一块到这里就可以继续接下来的数据编码了