百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

挖穿Android第四十一天(挖穿android第四十一天视频)

liuian 2025-04-30 18:01 22 浏览

1 播放视频

[1]surfaceView 可以来展示视频播放的内容

[2]sufaceView控件是一个重量级控件 初始化需要一点时间,可以直接在子线程更新ui ,内部维护2个线程,

A ---- 负责加载数据 B----加载数据

B ---- 负责显示 A----负责显示 男女搭配干活不累

[3]实现代码

private MediaPlayer mediaPlayer;

private int currentPosition; //当前视频播放的位置

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

 //[0]找到控件 用来展示我们播放视频内容
 final SurfaceView sfv = (SurfaceView) findViewById(R.id.sfv);
 SurfaceHolder holder = sfv.getHolder();
 //[0.1]SurfaceHolder内部维护了一个生命周期 
 holder.addCallback(new Callback() {
 //当SurfaceView表面销毁
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
 System.out.println("surfaceDestroyed");
 //当SurfaceView表秒销毁的时候 停在播放视频 并记录当前播放的位置 下次在播放的时候继续上次的位置继续播
 if (mediaPlayer!=null && mediaPlayer.isPlaying()) {
 //获取当前视频播放的位置 
 currentPosition = mediaPlayer.getCurrentPosition();
 //停止播放 
 mediaPlayer.stop();
 }
 }
 //SurfaceView表面创建 说明SurfaceView一定初始化好了 播放视频
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
 try {
 //[1]创建MediaPlayer实例 
 mediaPlayer = new MediaPlayer();
 //[2]播放sd卡里面的xpg.mp3 path:路径可以是本地路径也可以是网络路径
 mediaPlayer.setDataSource("http://192.168.77.88:8080/cc.mp4");
 //[3]准备播放
 mediaPlayer.prepareAsync();
 //[3.1]把播放的内容进行展示 SurfaceHolder就是用来维护视频展示
 mediaPlayer.setDisplay(holder);
 //当要播放的歌曲的数据缓冲完了 在播放 
 mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
 @Override
 public void onPrepared(MediaPlayer mp) {
 //[4]开始播放 
 mediaPlayer.start();
 //[5]播放上次位置 
 mediaPlayer.seekTo(currentPosition);
 }
 });
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 @Override
 public void surfaceChanged(SurfaceHolder holder, int format, int width,
 int height) {
 }
 });
}

2 VideoView 视频的view 只能播放视频

原理:就是surfaceView和mediaplayer的一个组合

//[1]找到控件 用来播放视频

VideoView vv = (VideoView) findViewById(R.id.vv);

//[2]设置播放视频的路径

vv.setVideoPath("http://192.168.77.88:8080/cc.mp4");

//[3]播放视频

vv.start();

mediaplayer只支持3gp 和mp4格式.

3 vitamio开源项目

这个称之为 类库 这个Android项目不可以运行

实现步骤

1 引入vitamio框架 以library、

2 在布局中定义VideoView
 <io.vov.vitamio.widget.VideoView 
 android:id="@+id/vv"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 />
3 mainactivity代码
 插件vitamio框架检查是否可用
 if (!LibsChecker.checkVitamioLibs(this)) {
 return;
 }
 final VideoView vv = (VideoView) findViewById(R.id.vv);
 vv.setVideoPath("http://192.168.1.2:8080/haha.avi");
 vv.setOnPreparedListener(new OnPreparedListener() {
 @Override
 public void onPrepared(MediaPlayer mp) {
 vv.start();
 }
 });
 //设置video的控制器
 vv.setMediaController(new MediaController(this));
 4 一定要在清单文件初始化InitActivity
 <activity android:name="io.vov.vitamio.activity.InitActivity"></activity>

底层解码开源项目ffmpeg.

4 照相和录像

照相代码

//[1]创建意图开启系统的照相机应用

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

//[2]设置保存 照片的路径

File file = new File(Environment.getExternalStorageDirectory().getPath(),"test1.png");

intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file));

//[3]开启系统的应用的界面

startActivityForResult(intent, 1);

录像代码

//[1]创建意图开启系统的照相机应用

Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);

//[2]设置保存 照片的路径

File file = new File(Environment.getExternalStorageDirectory().getPath(),"test2.3gp");

intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file));

//[3]开启系统的应用的界面

startActivityForResult(intent, 1);

5 Fragment

★Fragment是Activity的一部分

★Fragment必须嵌入到Activity里面

★ Fragment技术是在Android3.0引人的

总结

1)在布局里面声明 name属性指定你自己声明的fragment

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

android:orientation="horizontal"

android:layout_width="match_parent"

android:layout_height="match_parent">

<fragment android:name="com.itcast.fragment.DemoFragment"

android:id="@+id/list"

android:layout_weight="1"

android:layout_width="0dp"

android:layout_height="match_parent" />

<fragment android:name="com.itcast.fragment.Demo2Fragment"

android:id="@+id/viewer"

android:layout_weight="1"

android:layout_width="0dp"

android:layout_height="match_parent" />

2)自己定义fragment 注意的地方就是必须要重写onCreateview方法在这个方法里面加载fragment要显示的页面

public class Demo2Fragment extends Fragment {

//在这个方法里面加载布局

@Override

public View onCreateView(LayoutInflater inflater, ViewGroup container,

Bundle savedInstanceState) {

//使用打气筒把一个布局转换成view对象

View view = inflater.inflate(R.layout.fragment_demo2, null);

 return view;
}

}

6 动态添加fragment

//[1]获取手机的分辨率

WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);

int width = wm.getDefaultDisplay().getWidth();

int height = wm.getDefaultDisplay().getHeight();

//[2]获取Fragment的管理者

FragmentManager fragmentManager = getFragmentManager();

//开启一个事务

FragmentTransaction beginTransaction = fragmentManager.beginTransaction();

 //[3]判断手机是竖屏还是 横屏 
 if (height > width) {
 //说明是竖屏 我加载一个fragment android.R.id.content:代表id都是系统定义好的 理解成是当前手机的窗口
 beginTransaction.replace(android.R.id.content, new Demo1Fragment());
 }else {
 //说明是横屏 我加载另外一个fragment
 beginTransaction.replace(android.R.id.content, new Demo2Fragment());
 }
 //[4]最后一步一定记得提交事务 
 beginTransaction.commit();

7 Fragment兼容低版本

★ 兼容低版本就是使用v4包中的Fragment

★声明mianActivity要继承FragmentActivity

★ 获取Fragment的管理者的方式不一样了

FragmentManager supportFragmentManager = getSupportFragmentManager();

FragmentTransaction beginTransaction = supportFragmentManager.beginTransaction();

8 Fragment的生命周期

★实际开发中用到 最多的就是onCreateView 在这个方法里面加载 fragment要显示页面

★onDestroy 方法销毁 在这个方法里面做一些扫尾 或者释放内存的工作

9 fragment之间的通信

★fragment之间有一个公共的桥梁就是Activity

★在fragment里面如何弹土司 上下文使用getActivity();

10菜单

★1添加菜单

getMenuInflater().inflate(R.menu.main, menu);

★2动态添加

//[2]添加菜单第二种方式

menu.add(0, 1, 0, "前进");

menu.add(0, 2, 0, "后退");

menu.add(0, 3, 0, "首页");

★3自定义菜单 比如当点击菜单 弹出一个对话框

@Override

public boolean onMenuOpened(int featureId, Menu menu) {

//弹出自己想弹出的内容

AlertDialog.Builder builder = new Builder(this);

builder.setTitle("警告");

builder.setMessage("世界上最遥远的距离是没有网络");

builder.setPositiveButton("确定", new OnClickListener() {

 @Override
 public void onClick(DialogInterface dialog, int which) {
 }
 });
 builder.setNegativeButton("取消", new OnClickListener() {
 @Override
 public void onClick(DialogInterface dialog, int which) {
 }
 });
 //最后一步一定要记得 show
 builder.show();
 return false;
}

11 应用反编译

[1]apktools 可以获取应用的资源文件(布局资源 图片资源 清单文件)

[2]dex2jar 可以获取源代码

[3]jdgui.exe 查看源代码

[4]Android逆向助手

12 使用xml的方式定义补间动画

回顾补间动画的原理:产生动画效果没有改变控件的真实坐标,只是产生了一个动画效果而已,障眼法

步骤

1)在res下创建一个目录 anim(目录名)

2)在anim目录下创建对应的补间动画

3)使用AnimUtils.loadAnimation();加载动画

Animation sa = AnimationUtils.loadAnimation(this, R.anim.scale);

13 通知栏

[1]Toast

[2]对话框

[3]通知栏

//创建通知 链式调用

Notification noti = new Notification.Builder(this)

.setContentTitle("我是大标题")

.setContentText("我是大标题的内容")

.setSmallIcon(R.drawable.ic_launcher)

.setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.ic_launcher))

.build();

//发送通知

nm.notify(10, noti);

[4]取消通知

// 点击按钮 取消通知

public void click2(View v) {

nm.cancel(10);

}

相关推荐

eino v0.4.5版本深度解析:接口类型处理优化与错误机制全面升级

近日,eino框架发布了v0.4.5版本,该版本在错误处理、类型安全、流处理机制以及代理配置注释等方面进行了多项优化与修复。本次更新共包含6个提交,涉及10个文件的修改,由2位贡献者共同完成。本文将详...

SpringBoot异常处理_springboot异常注解

在SpringBoot中,异常处理是构建健壮、可维护Web应用的关键部分。良好的异常处理机制可以统一返回格式、提升用户体验、便于调试和监控。以下是SpringBoot中处理异常的完整指...

Jenkins运维之路(Jenkins流水线改造Day02-1-容器项目)

这回对线上容器服务器的流水线进行了一定的改造来满足目前线上的需求,还是会将所有的自动化脚本都放置到代码库中统一管理,我感觉一章不一定写的完,所以先给标题加了个-1,话不多说开干1.本次流水线的流程设计...

告别宕机!零基础搭建服务器监控告警系统!小白也能学会!

前言本文将带你从零开始,一步步搭建一个完整的服务器指标监控与邮件告警系统,使用的技术栈均为业界主流、稳定可靠的开源工具:Prometheus:云原生时代的监控王者,擅长指标采集与告警规则定义Node_...

httprunner实战接口测试笔记,拿走不谢

每天进步一点点,关注我们哦,每天分享测试技术文章本文章出自【码同学软件测试】码同学公众号:自动化软件测试码同学抖音号:小码哥聊软件测试01开始安装跟创建项目pipinstallhttprunne...

基于JMeter的性能压测平台实现_jmeter压测方案

这篇文章已经是两年前写的,短短两年时间,JMeter开源应用技术的发展已经是翻天覆地,最初由github开源项目zyanycall/stressTestPlatform形成的这款测试工具也开始慢...

12K+ Star!新一代的开源持续测试工具!

大家好,我是Java陈序员。在企业软件研发的持续交付流程中,测试环节往往是影响效率的关键瓶颈,用例管理混乱、接口调试复杂、团队协作不畅、与DevOps流程脱节等问题都能影响软件交付。今天,给大家...

Spring Boot3 中分库分表之后如何合并查询

在当今互联网应用飞速发展的时代,数据量呈爆发式增长。对于互联网软件开发人员而言,如何高效管理和查询海量数据成为了一项关键挑战。分库分表技术应运而生,它能有效缓解单库单表数据量过大带来的性能瓶颈。而在...

离线在docker镜像方式部署ragflow0.17.2

经常项目上会出现不能连外网的情况,要怎么使用ragflow镜像部署呢,这里提供详细的步骤。1、下载基础镜像根据docker-compose-base.yml及docker-compose.yml中的i...

看,教你手写一个最简单的SpringBoot Starter

何为Starter?想必大家都使用过SpringBoot,在SpringBoot项目中,使用最多的无非就是各种各样的Starter了。那何为Starter呢?你可以理解为一个可拔插式...

《群星stellaris》军事基地跳出怎么办?解决方法一览

《群星stellaris》军事基地跳出情况有些小伙伴出现过这种情况,究竟该怎么解决呢?玩家“gmjdadk”分享的自己的解决方法,看看能不能解决。我用英文原版、德语、法语和俄语四个版本对比了一下,结果...

数据开发工具dbt手拉手教程-03.定义数据源模型

本章节介绍在dbt项目中,如何定义数据源模型。定义并引入数据源通过Extract和Load方式加载到仓库中的数据,可以使用dbt中的sources组件进行定义和描述。通过在dbt中将这些数据集(表)声...

docker compose 常用命令手册_docker-compose init

以下是DockerCompose常用命令手册,按生命周期管理、服务运维、构建配置、扩缩容、调试工具分类,附带参数解析、示例和关键说明,覆盖多容器编排核心场景:一、生命周期管理(核心命令...

RagFlow与DeepSeek R1本地知识库搭建详细步骤及代码实现

一、环境准备硬件要求独立显卡(建议NVIDIAGPU,8GB显存以上)内存16GB以上,推荐32GB(处理大规模文档时更高效)SSD硬盘(加速文档解析与检索)软件安装bash#必装组件Docker...

Docker Compose 配置更新指南_docker-compose配置

高效管理容器配置变更的最佳实践方法重启范围保留数据卷适用场景docker-composeup-d变更的服务常规配置更新--force-recreate指定/所有服务强制重建down→up流程...