免费试用

中文化、本土化、云端化的在线跨平台软件开发工具,支持APP、电脑端、小程序、IOS免签等等

app付费视频30秒试看开发技术

在App开发中,实现付费视频功能是一个非常常见的需求,其中又会有很多用户希望能够提供30秒的试看时间,以便更好的决定是否进行购买。

下面我们来介绍一下实现这个功能的原理和方法。具体实现方式分为两种:客户端实现和服务器实现。

一、 客户端实现

1.客户端控制播放时间

客户端可以通过设置一个时间段,比如说30秒,然后在这个时间到达时暂停播放,等待用户操作去选择是否购买。

如下示例代码:

```

//设置一个30秒的定时器,时间到了暂停播放

CGRect videoViewFrame = CGRectMake(0, 0, self.view.frame.size.width, 200);

AVPlayerItem *playerItem = [[AVPlayerItem alloc] initWithURL:[NSURL URLWithString:@"http://example.com/video.mp4"]];

AVPlayer *player = [[AVPlayer alloc] initWithPlayerItem: playerItem];

//设置定时器时间为30秒

[player addPeriodicTimeObserverForInterval:CMTimeMakeWithSeconds(30, NSEC_PER_SEC) queue:nil usingBlock:^(CMTime time) {

//定时器时间到,暂停播放

[player pause];

}];

AVPlayerLayer *playerLayer = [AVPlayerLayer playerLayerWithPlayer: player];

playerLayer.frame = videoViewFrame;

[self.view.layer addSublayer: playerLayer];

[player play];

```

2.播放前缓存

在客户端播放视频之前,先缓存一定量的视频数据,同时控制播放进度和缓存进度的比例,当播放时间达到30秒时,则停止播放,并提示用户可以付费购买。此种方式需要做大量的本地缓存处理,对于流量比较高的视频来说显得有些低效。

如下示例代码:

```

//计算需要缓存的视频时长

NSUInteger duration = 30;

NSRange range = NSMakeRange(0, duration); //需要缓存的时长范围

AVPlayerItem *playerItem = [[AVPlayerItem alloc] initWithURL: [NSURL URLWithString: @"http://example.com/video.mp4"]];

AVAsset *asset = playerItem.asset;

AVAssetResourceLoader *resourceLoader = asset.resourceLoader;

AVURLAsset *urlAsset = (AVURLAsset *)asset;

NSURLComponents *components = [NSURLComponents componentsWithURL: [urlAsset.URL absoluteURL] resolvingAgainstBaseURL: NO];

components.scheme = @"custom";

//获取缓存区域的数据并存储到本地文件中

AVAssetResourceLoader *resourceLoader = asset.resourceLoader;

[resourceLoader setDelegate: self queue: dispatch_get_main_queue()];

__weak typeof(self) weakSelf = self;

self.resourceLoaderDelegate = [ResourceLoaderDelegate new];

[self.resourceLoaderDelegate addCompletionHandler: ^BOOL(NSData *data, NSError *__autoreleasing *error) {

BOOL success = NO;

if (range.length > 0 && self.requestTask) {

if (resourceLoader.localCachePath.length > 0) {

success = YES;

}

}

return success;

} forRequestTask: self.requestTask];

//计算缓存时长

int length = 1;

NSURL *customSchemeURL = [components URL];

AVURLAsset *urlAsset = [[AVURLAsset alloc] initWithURL: customSchemeURL options: nil];

NSArray *keys = @[@"duration"];

[urlAsset loadValuesAsynchronouslyForKeys:keys completionHandler:^{

NSError *error = nil;

AVKeyValueStatus durationStatus = [urlAsset statusOfValueForKey:@"duration" error:&error];

switch (durationStatus) {

case AVKeyValueStatusLoaded:

dispatch_async(dispatch_get_global_queue(0, DISPATCH_QUEUE_PRIORITY_DEFAULT), ^{

//1.计算需要缓存的数据字节

unsigned long long offset = range.location;

unsigned long long length = range.length;

//2.向服务端请求需要缓存的数据

NSData *requestData = [self sendRequestWithOffset: offset length: length];

if (requestData) {

//3.将请求来的数据存储到本地缓存文件中

BOOL result = [self writeData: requestData atOffset:offset];

if (result) {

//异步回调

dispatch_async(dispatch_get_main_queue(), ^{

[urlAsset.resourceLoader setDelegate:nil queue: NULL];

//创建播放器

weakSelf.playerItem = [[AVPlayerItem alloc] initWithURL: [NSURL URLWithString: @"http://example.com/video.mp4"]];

weakSelf.player = [AVPlayer playerWithPlayerItem:weakSelf.playerItem];

weakSelf.playerLayer = [AVPlayerLayer playerLayerWithPlayer:weakSelf.player];

weakSelf.playerLayer.frame = CGRectMake(0, 0, weakSelf.view.frame.size.width, 200);

[weakSelf.view.layer addSublayer: weakSelf.playerLayer];

[weakSelf.player play];

});

}

}

});

break;

default:

break;

}

}];

```

二、服务器实现

1.前端只需调用API

在服务器实现的方式中,前端只需调用服务端提供的API,即可在客户端以流的方式播放视频内容。在API中,可以设置视频内容试看的时间长度为30秒,超过30秒部分的内容需要付费才可以继续观看。

如下示例代码:

```

//前端根据用户ID和视频ID向API发起请求获取播放URL

NSString *userId = @"user-123456";

NSString *videoId = @"video-123456";

NSString *urlStr = [NSString stringWithFormat:@"api/v1/playUrl?userId=%@&videoId=%@", userId, videoId];

NSURLSession *session = [NSURLSession sharedSession];

NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:urlStr]];

[request setHTTPMethod:@"GET"];

NSURLSessionDataTask *task = [session dataTaskWithRequest:request

completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {

if(error) {

//请求出现错误,处理异常

}else {

//解析返回结果

NSDictionary *result = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];

NSString *playUrl = result[@"playUrl"];

//根据返回的URL展示视频播放器

AVPlayer *player = [[AVPlayer alloc] initWithURL:[NSURL URLWithString:playUrl]];

AVPlayerLayer *playerLayer = [AVPlayerLayer playerLayerWithPlayer:player];

playerLayer.frame = CGRectMake(0, 0, self.view.width, 200);

[self.view.layer addSublayer:playerLayer];

[player play];

}

}];

[task resume];

//api/v1/playUrl接口的实现,服务端根据用户ID和视频ID获取播放URL

public String getPlayUrl(String userId, String videoId) {

//1.根据用户ID和视频ID获取对应的视频信息

Video video = videoMapper.getVideoInfo(userId, videoId);

if(video == null) {

throw new BusinessException("该视频不存在!");

}

//2.检查该用户是否已经购买了该视频

boolean buyed = userService.checkUserBuyedVideo(userId, videoId);

if(buyed) {

//已购买,返回完整播放URL

return video.getUrl();

}else {

//未购买,返回试看URL(试看时间为30秒)

return video.getTrialUrl();

}

}

```

2.服务端自动截取前30秒

另外,服务端也可以通过自动截取视频前30秒,来实现视频试看功能。具体实现原理与上述两种方式类似。

如下示例代码:

```

//客户端请求服务端播放视频

NSString *userId = @"user-123456";

NSString *videoId = @"video-123456";

NSString *urlStr = [NSString stringWithFormat:@"api/v1/playUrl?userId=%@&videoId=%@", userId, videoId];

NSURLSession *session = [NSURLSession sharedSession];

NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:urlStr]];

[request setHTTPMethod:@"GET"];

NSURLSessionDataTask *task = [session dataTaskWithRequest:request

completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {

if(error) {

//请求出现错误,处理异常

}else {

//解析返回结果

NSDictionary *result = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];

NSString *playUrl = result[@"playUrl"];

//根据返回的URL展示视频播放器

AVPlayer *player = [[AVPlayer alloc] initWithURL:[NSURL URLWithString:playUrl]];

AVPlayerLayer *playerLayer = [AVPlayerLayer playerLayerWithPlayer:player];

playerLayer.frame = CGRectMake(0, 0, self.view.width, 200);

[self.view.layer addSublayer:playerLayer];

[player play];

}

}];

[task resume];

//api/v1/playUrl接口截取前30秒的实现

public String getPlayUrl(String userId, String videoId) {

//1.根据用户ID和视频ID获取对应的视频信息

Video video = videoMapper.getVideoInfo(userId, videoId);

if(video == null) {

throw new BusinessException("该视频不存在!");

}

//2.检查该用户是否已经购买了该视频

boolean buyed = userService.checkUserBuyedVideo(userId, videoId);

if(buyed) {

//已购买完整视频,返回完整播放URL

return video.getUrl();

}else {

//未购买,返回截取前30秒试看,并给出购买提示

//先获取完整视频URL

String fullVideoUrl = video.getUrl();

//调用接口截取视频前30秒,并获取新的URL

String trialUrl = videoService.cutVideoToUrl(fullVideoUrl, 0, 30);

return trialUrl;

}

}

//VideoService中截取视频前30秒的实现

public String cutVideoToUrl(String videoUrl, int startSecond, int endSecond) {

FFmpegFrameGrabber grabber = new FFmpegFrameGrabber(videoUrl);

try {

grabber.start();

if(startSecond >= grabber.getLengthInTime()/1000000) {

throw new BusinessException("截取起始时间不能超过总时长!");

}

FFmpegFrameRecorder recorder = new FFmpegFrameRecorder("out.mp4", grabber.getImageWidth(), grabber.getImageHeight(), grabber.getAudioChannels());

recorder.setVideoCodec(avcodec.AV_CODEC_ID_MPEG4);

recorder.setFormat("mp4");

recorder.setFrameRate(grabber.getFrameRate());

recorder.setSampleRate(grabber.getSampleRate());

recorder.setAudioChannels(grabber.getAudioChannels());

recorder.start();

long start = startSecond * 1000000;

long end = endSecond * 1000000;

Frame frame = null;

while((frame = grabber.grab()) != null) {

if(frame.timestamp > start && frame.timestamp <= end) { //在截取的范围内

recorder.record(frame);

}

if(frame.timestamp > end) { //超出截取范围

break;

}

}

grabber.stop();

grabber.release();

recorder.stop();

recorder.release();

return "http://example.com/out.mp4";

} catch (Exception e) {

throw new BusinessException("截取视频失败!");

}

}

```

综上所述,客户端实现和服务端实现都可以实现给用户提供30秒试看的功能,实现的方式各不相同,具体实现方案应根据实际情况来选择。


相关知识:
如何开发非原生app
随着移动互联网的发展,移动应用程序已经成为人们日常生活中不可或缺的一部分。当我们需要开发移动应用程序时,我们通常会想到原生应用程序。然而,随着技术的发展和用户需求的不断变化,非原生应用程序也越来越受到欢迎。那么,如何开发非原生应用程序呢?本文将为您介绍非原
2024-01-10
【本地部署】下载PHP本地托管文件,轻松实现自有域名下载APP,适用于PHP网站环境
很多开发者希望测试地址为自己的域名! 这里提供一种快捷搭建分发下载的方法,只需要在分发平台下载对应分发部署文件上传自有服务器即可!
2024-01-09
app开发的技术威胁是什么
移动应用程序(App)已成为人们生活的重要组成部分。然而,随着移动应用程序的普及和依赖性的增加,安全风险也在增加。这些风险与应用程序本身和后台服务器之间的通信有关,这使得移动应用程序容易受到黑客攻击。本文将讨论一些主要的移动应用程序安全威胁,并介绍防范这些
2023-06-29
app开发的学习
在现代社会,移动应用程序已成为人们生活中必不可少的一部分。因此,随着人们对移动应用程序的需求不断增长,会发现越来越多的人想学习app开发。本文将介绍app开发的基础知识和原理,以帮助初学者了解app开发。首先,什么是app开发?app是指应用程序,通常是在
2023-06-29
app开发教程doc
APP(Application)即应用程序,是指在电子设备上运行的软件程序,具有特定的功能,通常与特定的硬件或操作系统相关。APP在现代人们的生活中扮演着越来越重要的角色,无论是工作、娱乐还是社交等方面,都有广泛的应用。随着智能手机的普及,APP开发也成为
2023-06-29
appnative开发
Native app开发是指在移动设备上直接安装的应用程序,它们通过设备的操作系统(如iOS、Android等)进行交互。App Native开发就是针对不同的平台开发原生应用程序。App Native开发需要使用针对特定设备操作系统的软件开发工具包(SD
2023-05-06