挖穿Android第四十一天(挖穿android第四十一天视频)
liuian 2025-04-30 18:01 10 浏览
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);
}
相关推荐
- 深入解析 MySQL 8.0 JSON 相关函数:解锁数据存储的无限可能
-
引言在现代应用程序中,数据的存储和处理变得愈发复杂多样。MySQL8.0引入了丰富的JSON相关函数,为我们提供了更灵活的数据存储和检索方式。本文将深入探讨MySQL8.0中的JSON...
- MySQL的Json类型个人用法详解(mysql json类型对应java什么类型)
-
前言虽然MySQL很早就添加了Json类型,但是在业务开发过程中还是很少设计带这种类型的表。少不代表没有,当真正要对Json类型进行特定查询,修改,插入和优化等操作时,却感觉一下子想不起那些函数怎么使...
- MySQL的json查询之json_array(mysql json_search)
-
json_array顾名思义就是创建一个数组,实际的用法,我目前没有想到很好的使用场景。使用官方的例子说明一下吧。例一selectjson_array(1,2,3,4);json_array虽然单独...
- 头条创作挑战赛#一、LSTM 原理 长短期记忆网络
-
#头条创作挑战赛#一、LSTM原理长短期记忆网络(LongShort-TermMemory,LSTM)是一种特殊类型的循环神经网络(RNN),旨在解决传统RNN在处理长序列数据时面临的梯度...
- TensorBoard最全使用教程:看这篇就够了
-
机器学习通常涉及在训练期间可视化和度量模型的性能。有许多工具可用于此任务。在本文中,我们将重点介绍TensorFlow的开源工具套件,称为TensorBoard,虽然他是TensorFlow...
- 图神经网络版本的Kolmogorov Arnold(KAN)代码实现和效果对比
-
本文约4600字,建议阅读10分钟本文介绍了图神经网络版本的对比。KolmogorovArnoldNetworks(KAN)最近作为MLP的替代而流行起来,KANs使用Kolmogorov-Ar...
- kornia,一个实用的 Python 库!(python kkb_tools)
-
大家好,今天为大家分享一个实用的Python库-kornia。Github地址:https://github.com/kornia/kornia/Kornia是一个基于PyTorch的开源计算...
- 图像分割掩码标注转YOLO多边形标注
-
Ultralytics团队付出了巨大的努力,使创建自定义YOLO模型变得非常容易。但是,处理大型数据集仍然很痛苦。训练yolo分割模型需要数据集具有其特定格式,这可能与你从大型数据集中获得的...
- [python] 向量检索库Faiss使用指北
-
Faiss是一个由facebook开发以用于高效相似性搜索和密集向量聚类的库。它能够在任意大小的向量集中进行搜索。它还包含用于评估和参数调整的支持代码。Faiss是用C++编写的,带有Python的完...
- 如何把未量化的 70B 大模型加载到笔记本电脑上运行?
-
并行运行70B大模型我们已经看到,量化已经成为在低端GPU(比如Colab、Kaggle等)上加载大型语言模型(LLMs)的最常见方法了,但这会降低准确性并增加幻觉现象。那如果你和你的朋友们...
- ncnn+PPYOLOv2首次结合!全网最详细代码解读来了
-
编辑:好困LRS【新智元导读】今天给大家安利一个宝藏仓库miemiedetection,该仓库集合了PPYOLO、PPYOLOv2、PPYOLOE三个算法pytorch实现三合一,其中的PPYOL...
- 人工智能——图像识别(人工智能图像识别流程)
-
概述图像识别(ImageRecognition)是计算机视觉的核心任务之一,旨在通过算法让计算机理解图像内容,包括分类(识别物体类别)、检测(定位并识别多个物体)、分割(像素级识别)等,常见的应用场...
- PyTorch 深度学习实战(15):Twin Delayed DDPG (TD3) 算法
-
在上一篇文章中,我们介绍了DeepDeterministicPolicyGradient(DDPG)算法,并使用它解决了Pendulum问题。本文将深入探讨TwinDelayed...
- 大模型中常用的注意力机制GQA详解以及Pytorch代码实现
-
分组查询注意力(GroupedQueryAttention)是一种在大型语言模型中的多查询注意力(MQA)和多头注意力(MHA)之间进行插值的方法,它的目标是在保持MQA速度的同时...
- pytorch如何快速创建具有特殊意思的tensor张量?
-
专栏推荐正文我们通过值可以看到torch.empty并没有进行初始化创建tensor并进行随机初始化操作,常用rand/rand_like,randint正态分布(0,1)指定正态分布的均值还有方差i...
- 一周热门
-
-
Python实现人事自动打卡,再也不会被批评
-
Psutil + Flask + Pyecharts + Bootstrap 开发动态可视化系统监控
-
一个解决支持HTML/CSS/JS网页转PDF(高质量)的终极解决方案
-
再见Swagger UI 国人开源了一款超好用的 API 文档生成框架,真香
-
【验证码逆向专栏】vaptcha 手势验证码逆向分析
-
网页转成pdf文件的经验分享 网页转成pdf文件的经验分享怎么弄
-
C++ std::vector 简介
-
python使用fitz模块提取pdf中的图片
-
《人人译客》如何规划你的移动电商网站(2)
-
Jupyterhub安装教程 jupyter怎么安装包
-
- 最近发表
-
- 深入解析 MySQL 8.0 JSON 相关函数:解锁数据存储的无限可能
- MySQL的Json类型个人用法详解(mysql json类型对应java什么类型)
- MySQL的json查询之json_array(mysql json_search)
- 头条创作挑战赛#一、LSTM 原理 长短期记忆网络
- TensorBoard最全使用教程:看这篇就够了
- 图神经网络版本的Kolmogorov Arnold(KAN)代码实现和效果对比
- kornia,一个实用的 Python 库!(python kkb_tools)
- 图像分割掩码标注转YOLO多边形标注
- [python] 向量检索库Faiss使用指北
- 如何把未量化的 70B 大模型加载到笔记本电脑上运行?
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- table.render (33)
- uniapp textarea (33)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- react-admin (33)
- vscode切换git分支 (35)
- vscode美化代码 (33)
- python bytes转16进制 (35)