zoukankan      html  css  js  c++  java
  • iOS 音视频播放

     2018.9.18

    容器云强势上线!快速搭建集群,上万Linux镜像随意使用

    自定义控制层的播放器 SJVideoPlayer

    项目介绍:

    SJVideoPlayer

    一个可自定义控制层的播放器(如果缺少API, 可以邮箱联系我 changsanjiang@gmail.com )

    安装

    # 有默认控制层的播放器
    pod 'SJVideoPlayer' 
    
    # 基础播放器, 不含控制层, 如果需要自定义控制层, 可以使用它.
    pod 'SJBaseVideoPlayer'
    • 有默认控制层的播放器

    • 基础播放器, 不含控制层

    _

    播放

        Player.asset = [[SJVideoPlayerAssetCarrier alloc] initWithAssetURL:[NSURL URLWithString:@"http://....."] beginTime:10];

    _

    cell中播放

        Player.asset =
        [[SJVideoPlayerAssetCarrier alloc] initWithAssetURL:[NSURL URLWithString:cell.model.playURLStr]
                                                 scrollView:self.tableView
                                                  indexPath:[self.tableView indexPathForCell:cell]
                                               superviewTag:playerParentView.tag];

    _

    在嵌套(tableView嵌套collectionView或者反之)的视图中播放

        Player.asset =
        [[SJVideoPlayerAssetCarrier alloc] initWithAssetURL:playURL
                                                  indexPath:indexPath
                                               superviewTag:playerParentView.tag
                                        scrollViewIndexPath:embeddedScrollViewIndexPath
                                              scrollViewTag:embeddedScrollView.tag
                                             rootScrollView:self.tableView];

    _

    播放相关

    #pragma mark - 播放
    
    @interface SJBaseVideoPlayer (Play)
    
    @property (nonatomic, strong, readwrite, nullable) NSURL *assetURL;
    
    @property (nonatomic, strong, readwrite, nullable) SJVideoPlayerURLAsset *URLAsset;
    
    - (void)playWithURL:(NSURL *)playURL;
    
    - (void)playWithURL:(NSURL *)playURL jumpedToTime:(NSTimeInterval)time;
    
    - (void)refresh;
    
    @end

    _

    时间相关

    #pragma mark - 时间
    
    @interface SJBaseVideoPlayer (Time)
    
    - (NSString *)timeStringWithSeconds:(NSInteger)secs; // format: 00:00:00
    
    @property (nonatomic, readonly) float progress;
    
    @property (nonatomic, readonly) NSTimeInterval currentTime;
    @property (nonatomic, readonly) NSTimeInterval totalTime;
    
    @property (nonatomic, strong, readonly) NSString *currentTimeStr;
    @property (nonatomic, strong, readonly) NSString *totalTimeStr;
    
    - (void)jumpedToTime:(NSTimeInterval)secs completionHandler:(void (^ __nullable)(BOOL finished))completionHandler; // unit is sec. 单位是秒.
    
    - (void)seekToTime:(CMTime)time completionHandler:(void (^ __nullable)(BOOL finished))completionHandler;
    
    @end

    _

    控制相关

    #pragma mark - 控制
    
    @interface SJBaseVideoPlayer (Control)
    
    @property (nonatomic, readwrite) BOOL mute; // default is no. 静音.
    
    @property (nonatomic, readwrite, getter=isLockedScreen) BOOL lockedScreen; // 锁定播放器. 所有交互事件将不会触发.
    
    @property (nonatomic, readwrite, getter=isAutoPlay) BOOL autoPlay; // 自动播放. default is YES.
    
    - (BOOL)play;
    
    - (BOOL)pause;                                           // 调用此方法, 表示开发者暂停.
    - (void)pauseForUser;                                    // 调用此方法, 表示用户暂停.
    @property (nonatomic, assign, readonly) BOOL userPaused; // 区分是用户暂停的, 还是开发者暂停的
    
    - (void)stop;
    
    - (void)stopAndFadeOut; // 停止播放并淡出
    
    - (void)replay;
    
    @property (nonatomic, readwrite) float volume;
    
    @property (nonatomic, readwrite) float brightness;
    
    @property (nonatomic, readwrite) float rate; // 0.5...2
    
    @property (nonatomic, copy, readwrite, nullable) void(^rateChanged)(__kindof SJBaseVideoPlayer *player);
    
    - (void)resetRate;
    
    @property (nonatomic, copy, readwrite, nullable) void(^playDidToEnd)(__kindof SJBaseVideoPlayer *player); // 播放完毕
    
    @end

    _

    控制层管理器相关

    #pragma mark - 控制层
    
    @interface SJBaseVideoPlayer (ControlLayer)
    
    @property (nonatomic, readwrite) BOOL enableControlLayerDisplayController; // default is YES. 是否开启控制层[显示/隐藏]的管理器
    @property (nonatomic, readonly) BOOL controlLayerAppeared; // 控制层是否显示
    @property (nonatomic, copy, readwrite, nullable) void(^controlLayerAppearStateChanged)(__kindof SJBaseVideoPlayer *player, BOOL state);
    
    - (void)controlLayerNeedAppear;
    - (void)controlLayerNeedDisappear;
    
    // 控制层是否显示
    @property (nonatomic, readonly) BOOL controlViewDisplayed NS_DEPRECATED(2_0, 2_0, 2_0, 2_0, "use `controlLayerAppeared`");
    
    /*!
     *  Call when the control view is appear or disappear.
     *
     *  控制视图隐藏或显示的时候调用.
     **/
    @property (nonatomic, copy, readwrite, nullable) void(^controlViewDisplayStatus)(__kindof SJBaseVideoPlayer *player, BOOL displayed) NS_DEPRECATED(2_0, 2_0, 2_0, 2_0, "use `controlLayerAppearStateChanged`");
    
    @end

    _

    屏幕旋转相关

    #pragma mark - 屏幕旋转
    
    @interface SJBaseVideoPlayer (Rotation)
    
    - (void)rotation; // 旋转
    
    @property (nonatomic, assign, readwrite) BOOL disableRotation; // 禁止播放器旋转
    
    @property (nonatomic, copy, readwrite, nullable) void(^willRotateScreen)(__kindof SJBaseVideoPlayer *player, BOOL isFullScreen); // 将要旋转的时候调用
    
    @property (nonatomic, copy, readwrite, nullable) void(^rotatedScreen)(__kindof SJBaseVideoPlayer *player, BOOL isFullScreen);    // 已旋转
    
    @property (nonatomic, assign, readonly) BOOL isFullScreen;  // 是否全屏
    
    @end

    截图相关

    #pragma mark - 截图
    
    @interface SJBaseVideoPlayer (Screenshot)
    
    @property (nonatomic, copy, readwrite, nullable) void(^presentationSize)(__kindof SJBaseVideoPlayer *videoPlayer, CGSize size);
    
    - (UIImage * __nullable)screenshot;
    
    - (void)screenshotWithTime:(NSTimeInterval)time
                    completion:(void(^)(__kindof SJBaseVideoPlayer *videoPlayer, UIImage * __nullable image, NSError *__nullable error))block;
    
    - (void)screenshotWithTime:(NSTimeInterval)time
                          size:(CGSize)size
                    completion:(void(^)(__kindof SJBaseVideoPlayer *videoPlayer, UIImage * __nullable image, NSError *__nullable error))block;
    
    - (void)generatedPreviewImagesWithMaxItemSize:(CGSize)itemSize
                                       completion:(void(^)(__kindof SJBaseVideoPlayer *player, NSArray<id<SJVideoPlayerPreviewInfo>> *__nullable images, NSError *__nullable error))block;
    
    @end
    

    _

    提示相关

    #pragma mark - 提示
    
    @interface SJBaseVideoPlayer (Prompt)
    
    /*!
     *  prompt.update(^(SJPromptConfig * _Nonnull config) {
            config.cornerRadius = 4;                    // default cornerRadius.
            config.font = [UIFont systemFontOfSize:12]; // default font.
        });
     *
     **/
    @property (nonatomic, strong, readonly) SJPrompt *prompt;
    
    - (void)showTitle:(NSString *)title; // duration default is 1.0
    
    - (void)showTitle:(NSString *)title duration:(NSTimeInterval)duration; // duration if value set -1, promptView will always show.
    
    - (void)hiddenTitle;
    
    @end

    _

    控制层协议

    #pragma mark - Protocol
    
    @protocol SJVideoPlayerControlLayerDataSource <NSObject>
    
    @required
    
    - (UIView *)controlView;
    
    /// 控制层需要隐藏之前会调用这个方法, 如果返回NO, 将不调用`controlLayerNeedDisappear:`.
    - (BOOL)controlLayerDisappearCondition;
    
    /// 触发手势之前会调用这个方法, 如果返回NO, 将不调用水平手势相关的代理方法.
    - (BOOL)triggerGesturesCondition:(CGPoint)location;
    
    @optional
    /// 安装完控制层的回调.
    - (void)installedControlViewToVideoPlayer:(SJBaseVideoPlayer *)videoPlayer;
    
    @end
    
    
    @protocol SJVideoPlayerControlLayerDelegate <NSObject>
    
    @optional
    
    #pragma mark - 播放之前/状态
    /// 当设置播放资源时调用.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer prepareToPlay:(SJVideoPlayerURLAsset *)asset;
    
    /// 播放状态改变.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer stateChanged:(SJVideoPlayerPlayState)state;
    
    /// 播放报错
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer playFailed:(NSError *)error;
    
    #pragma mark - 进度
    /// 播放进度回调.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer
            currentTime:(NSTimeInterval)currentTime currentTimeStr:(NSString *)currentTimeStr
              totalTime:(NSTimeInterval)totalTime totalTimeStr:(NSString *)totalTimeStr;
    
    /// 缓冲的进度.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer loadedTimeProgress:(float)progress;
    
    /// 开始缓冲.
    - (void)startLoading:(SJBaseVideoPlayer *)videoPlayer;
    
    /// 缓冲完成.
    - (void)loadCompletion:(SJBaseVideoPlayer *)videoPlayer;
    
    #pragma mark - 显示/消失
    /// 控制层需要显示.
    - (void)controlLayerNeedAppear:(SJBaseVideoPlayer *)videoPlayer;
    
    /// 控制层需要隐藏.
    - (void)controlLayerNeedDisappear:(SJBaseVideoPlayer *)videoPlayer;
    
    ///  在`tableView`或`collectionView`上将要显示的时候调用.
    - (void)videoPlayerWillAppearInScrollView:(SJBaseVideoPlayer *)videoPlayer;
    
    ///  在`tableView`或`collectionView`上将要消失的时候调用.
    - (void)videoPlayerWillDisappearInScrollView:(SJBaseVideoPlayer *)videoPlayer;
    
    #pragma mark - 锁屏
    /// 播放器被锁屏, 此时将不旋转, 不触发手势相关事件.
    - (void)lockedVideoPlayer:(SJBaseVideoPlayer *)videoPlayer;
    
    /// 播放器解除锁屏.
    - (void)unlockedVideoPlayer:(SJBaseVideoPlayer *)videoPlayer;
    
    #pragma mark - 屏幕旋转
    /// 播放器将要旋转屏幕, `isFull`如果为`YES`, 则全屏.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer willRotateView:(BOOL)isFull;
    
    /// 旋转完毕.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer didEndRotation:(BOOL)isFull;
    
    #pragma mark - 音量 / 亮度 / 播放速度
    /// 静音开关变更
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer muteChanged:(BOOL)mute;
    
    /// 声音被改变.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer volumeChanged:(float)volume;
    
    /// 亮度被改变.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer brightnessChanged:(float)brightness;
    
    /// 播放速度被改变.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer rateChanged:(float)rate;
    
    #pragma mark - 水平手势
    /// 水平方向开始拖动.
    - (void)horizontalDirectionWillBeginDragging:(SJBaseVideoPlayer *)videoPlayer;
    
    /// 水平方向拖动中. `translation`为此次增加的值.
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer horizontalDirectionDidDrag:(CGFloat)translation;
    
    /// 水平方向拖动结束.
    - (void)horizontalDirectionDidEndDragging:(SJBaseVideoPlayer *)videoPlayer;
    
    #pragma mark - size
    - (void)videoPlayer:(SJBaseVideoPlayer *)videoPlayer presentationSize:(CGSize)size;
    
    @end

    _

    抽离出的控件, 可独立使用

    加载视图

    全屏返回手势

    • 手势在UIScrollView和UIPageViewController中完美处理。

    • 可指定盲区。指定的区域不会触发手势。它不会影响其他ViewControllers。

    • 可在指定页面禁用手势。指定ViewController禁用手势。它不会影响其他ViewControllers。

    • WKWebView返回上一个网页。

    亮度和音量调整

    播放资源载体

    屏幕旋转观察者

    滑动条

    提示

    便捷创建UI的工厂

    播放类型

    https://blog.csdn.net/weixin_33825683/article/details/91374810

    按公司需求需要对音频文件进行后台播放,借此机会对音频播放做了个总结.主要针对 AVPlayer 进行详细说明.

    iOS 各播放器比较

    名称使用环境优点缺点
    System Sound Services AVFoundation C语言的底层写法,节省内存 支持的格式有限,音量无法通过音量键控制,而且播放方式单一。
    AVAudioPlayer AVFoundation 抒写效率更高,基本上支持所有的音频格式,对播放的控制,如循环播放,声音大小,暂停等比较方便。 对内存的消耗会多些。不支持流式,即无法播放在线音乐。
    AVPlayer AVFoundation 可以播放音视频,可播放在线音乐,使用灵活  
    MPMoviePlayerController MediaPlayer 简单易用 不可定制
    AVPlayerViewController AVKit 简单易用 不可定制
    IJKPlayer IJKMediaFramework 定制度高,支持流媒体播放 使用稍复杂

    AVPlayer 使用

    简介

    AVPlayer 是iOS上常用的视频播放器组件,支持常见的音视频格式,支持流播放,可以播放在线音乐. 支持视频格式: WMV,AVI,MKV,RMVB,RM,XVID,MP4,3GP,MPG等。 支持音频格式:MP3,WMA,RM,ACC,OGG,APE,FLAC,FLV等。

    相关类

    • AVPlayer:播放器,控制播放器的播放,暂停,播放速度.
    • AVURLAsset : AVAsset 的一个子类,使用 URL 进行实例化,实例化对象包换 URL 对应视频资源的所有信息.
    • AVPlayerItem:管理资源对象,提供播放数据源.
    • AVPlayerLayer:负责显示视频,如果没有添加该类,只有声音没有画面.

    简单使用

    使用 url 创建 AVPlayer

    1.  
      let player = AVPlayer(url: URL(string: "http://www.xxxx.mp3"))
    2.  
      复制代码

    使用 AVPlayerItem 创建 AVPlayer

    1.  
      if let url = URL(string: "http://www.***.mp3") {
    2.  
      let asset = AVAsset(url: url)
    3.  
      guard asset.isPlayable else{
    4.  
      // 检测文件是否可播放
    5.  
      return
    6.  
      }
    7.  
      let playItem = AVPlayerItem(asset: asset)
    8.  
      let player = AVPlayer(playerItem: playItem)
    9.  
      player.play()
    10.  
      }
    11.  
      复制代码

    AVPlayer 控制播放

    1.  
      player.play() // 播放
    2.  
      player.pause() //暂停
    3.  
      player.rate = 1.0 // 播放速度
    4.  
      复制代码

    通过通知监听播放状态变化

    //播放完成
    AVPlayerItemDidPlayToEndTimeNotification
    //播放失败
    AVPlayerItemFailedToPlayToEndTimeNotification
    //异常中断
    AVPlayerItemPlaybackStalledNotification
     
    // eg: 播放结束通知
    NotificationCenter.default.addObserver(self, selector: #selector(finish(_:)), name: NSNotification.Name.AVPlayerItemDidPlayToEndTime, object: nil)
    复制代码

    监听播放进度

    // 添加周期时间观察者 一秒执行一次 block
    let timeObserver = player.addPeriodicTimeObserver(forInterval: CMTime(seconds: 1, preferredTimescale: 1), queue: DispatchQueue.main, using: { [weak self] (cmTime) in
        if let totalTime = self?.currentPlayItem?.duration {
            self?.delegate?.player(self!, currentTime: cmTime.seconds, totalTime: totalTime.seconds)
        }
    })
    // 不要忘记移除
    player.removeTimeObserver(observer)

    AVPlayerItem 创建

     
    // 监听 playerItem 状态变化
    playItem.addObserver(self, forKeyPath: "status", options: .new, context: nil)
    // 监听缓存时间
    playItem.addObserver(self, forKeyPath: "loadedTimeRanges", options: .new, context: nil)
     
    // 移除监听
    currentPlayItem?.removeObserver(self, forKeyPath: "status")
    currentPlayItem?.removeObserver(self, forKeyPath: "loadedTimeRanges")
    复制代码
    override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
        
        if object is AVPlayerItem {
            if keyPath == "status" {
                if let playerItem = object as? AVPlayerItem {
                    switch playerItem.status {
                    case .readyToPlay:
                        // 准备播放
                    case .failed:
                        // 加载失败                    
                    default:
                        // 未知状态
                    }
                }
            }
            
            if keyPath == "loadedTimeRanges" {
                if let playerItem = object as? AVPlayerItem {
                    if let timeRange = playerItem.loadedTimeRanges.first as? CMTimeRange {
                        let cache = timeRange.start.seconds + timeRange.duration.seconds // 缓存总时长
                    }
                }
            }
        }

    音频后台播放

    开启所需后台模式

    利用 AVAudioSession 申请后台播放权限

    let session = AVAudioSession.sharedInstance()
    do {
        try session.setActive(true)
        try session.setCategory(AVAudioSessionCategoryPlayback)
    } catch {
        print(error)
    }
    1. 在播放控制界面接受远程控制(Remote Control)

    2. 开启远程控制

     

    // 声明接收Remote Control事件

    UIApplication.shared.beginReceivingRemoteControlEvents()
    1. 设置 Remote Control 响应

    // 响应 Remote Control事件
    MPRemoteCommandCenter.shared().playCommand.addTarget(self, action: #selector(play))
    MPRemoteCommandCenter.shared().nextTrackCommand.addTarget(self, action: #selector(next))
    MPRemoteCommandCenter.shared().pauseCommand.addTarget(self, action: #selector(pause))
    MPRemoteCommandCenter.shared().previousTrackCommand.addTarget(self, action: #selector(previous))

    移除 Remote Control 响应

     
    // 在关闭播放页面时记得移除
    MPRemoteCommandCenter.shared().playCommand.removeTarget(self, action: #selector(play))
    MPRemoteCommandCenter.shared().nextTrackCommand.removeTarget(self, action: #selector(next))
    MPRemoteCommandCenter.shared().pauseCommand.removeTarget(self, action: #selector(pause))
    MPRemoteCommandCenter.shared().previousTrackCommand.removeTarget(self, action: #selector(previous))
    // 停止响应 Remote Control
    UIApplication.shared.endReceivingRemoteControlEvents()
    1. 通过重写父类方法响应外部事件

    2. 开启接受远程控制

    3. 使当前页面成为第一响应者

    4. 重写 remoteControlReceivedWithEvent 方法. UIEvent Type 取值:

      1. UIEventSubtypeRemoteControlTogglePlayPause // 暂停
      2. UIEventSubtypeRemoteControlPreviousTrack // 上一首
      3. UIEventSubtypeRemoteControlNextTrack // 下一首
      4. UIEventSubtypeRemoteControlPlay // 播放
      5. UIEventSubtypeRemoteControlPause // 暂停
    5. 关闭接受远程控制

    6. 锁屏页面显示播放信息(Now Playing Center)

    使用 MPNowPlayingInfoCenter 设置锁屏页面音乐信息.

    func setLockScreenPlayingInfo(_ info: YTTMediaInfo) {
        // Now Playing Center可以在锁屏界面展示音乐的信息,也达到增强用户体验的作用。
        // https://www.jianshu.com/p/458b67f84f27
        var infoDic: [String : Any] = [:]
        infoDic[MPMediaItemPropertyTitle] = info.title // 歌曲名
        infoDic[MPMediaItemPropertyArtist] = info.singer // 歌手
        if let img = info.image {
            infoDic[MPMediaItemPropertyArtwork] = MPMediaItemArtwork(image: img) // 专辑图片
        }
        infoDic[MPMediaItemPropertyPlaybackDuration] = info.totalTime // 歌曲总时长
        infoDic[MPNowPlayingInfoPropertyElapsedPlaybackTime] = info.currentTime // 当前播放时间
        infoDic[MPNowPlayingInfoPropertyPlaybackRate] = 1.0 // 播放速度
        MPNowPlayingInfoCenter.default().nowPlayingInfo = infoDic
    }

    注意: MPNowPlayingInfoPropertyElapsedPlaybackTime 设置的并不是时时的,他是根据你设置的值进行计时的,如果想要在锁屏页面得到准确的时间,请及时刷新 MPNowPlayingInfoPropertyElapsedPlaybackTime 的值.当暂停时要暂停播放时间,只需将 MPNowPlayingInfoPropertyPlaybackRate 设置为 0.播放时设置回 1.

    1. 补充说明

    iOS 对后台管理十分严格,任何 app 都有大约3分钟或者10分钟的后台执行时间.3分钟或者10分钟后, app 就会被强制挂起.使用 AVAudioSession 申请后台权限时,可以保证播放本地音乐能在后台长久播放,当播放网络音乐时就会出现不能播放情况,针对这情况使用了 beginBackgroundTask 设置后台任务 ID,通过这种方式我们大约可以获得额外的 10 分钟来执行后台任务.为了能无限后台播放网络音乐添加计时器,当即将挂起时再次申请后台任务 ID.

    func applicationDidEnterBackground(_ application: UIApplication) {

       

        // 这样做,可以在按home键进入后台后 ,播放一段时间,几分钟吧。但是不能持续播放网络歌曲,若需要持续播放网络歌曲,还需要申请后台任务id

        bgTask = application.beginBackgroundTask(expirationHandler: nil)

        timer = Timer.scheduledTimer(timeInterval: 1, target: self, selector: #selector(timerAction), userInfo: nil, repeats: true)

    }

     

    @objc func timerAction() {

        timerCount = timerCount + 1

        if timerCount < 500 {

            return

        }

        timerCount = 0

        let newTask = UIApplication.shared.beginBackgroundTask(expirationHandler: nil)

        if bgTask != UIBackgroundTaskInvalid && newTask != UIBackgroundTaskInvalid {

            UIApplication.shared.endBackgroundTask(bgTask)

            bgTask = newTask

        }

       

    }

    1. 其他

    2. AVPlayer那些坑
    3. 项目参考地址
    4. 针对缓存实现 Demo

    iOS 音视频开发-MediaPlayer播放本地、远程音频

    https://www.jianshu.com/p/ec3533549aa6

    iOS音视频播放---改善用户体验

    https://blog.csdn.net/szk972092933/article/details/82770725

    iOS——给视频添加音频和字幕

    https://www.jianshu.com/p/9d16907b2c52

    iOS 一个功能很全的视频播放器

    https://www.jianshu.com/p/4c2a493fb4bf

    SJVideoPlayer

    基于AVPlayer.

    https://github.com/changsanjiang/SJVideoPlayer

    极速初始化, 不阻塞主线程. 这个应该是目前基于AVPlayer的播放器中, 功能最全的一个吧.

    图解使用请移步: https://www.jianshu.com/p/a60389f9acaf 模块化控制层请移步:  https://www.jianshu.com/p/6a968ec24d3f

     过滤视频中的黑帧(生成视频thumbnail时,跳过黑帧)

    https://www.jianshu.com/p/eaaee36195cb

  • 相关阅读:
    Android学习笔记37-使用Content Providers方式共享数据
    Android学习笔记36-使用SQLite方式存储数据
    Android学习笔记35-使用Shared Preferences方式存储数据
    Android学习笔记34-使用文件存储数据
    Android学习笔记33-Intent介绍及Intent在Activity中的使用方法
    Android学习笔记32-滑屏控件ViewPager的使用
    Android学习笔记31-使用惰性控件ViewStub实现布局动态加载
    Android学习笔记30-列表ListView控件的使用
    Android学习笔记29-切换卡TabHost控件的使用
    用户代理
  • 原文地址:https://www.cnblogs.com/sundaysgarden/p/12066037.html
Copyright © 2011-2022 走看看