写点什么

Activity 初学乍练

作者:向阳逐梦
  • 2023-03-28
    四川
  • 本文字数:5899 字

    阅读完需:约 19 分钟

Activity初学乍练

本节开始讲解 Android 的四大组件之一的 Activity(活动),先来看下官方对于 Activity 的介绍:PS:官网文档:Activity

Activity 是一个应用程序的组件,他在屏幕上提供了一个区域,允许用户在上面做一些交互性的操作,比如打电话,照相,发送邮件,或者显示一个地图!Activity 可以理解成一个绘制用户界面的窗口,而这个窗口可以填满整个屏幕,也可能比屏幕小或者浮动在其他窗口的上方!

从上面这段话,我们可以得到以下信息:

1. Activity 用于显示用户界面,用户通过 Activity 交互完成相关操作 2. 一个 App 允许有多个 Activity

好了,大概的引言就介绍到这里,想深入了解可以继续看 API,开始本节内容~

1.Activity 的概念与 Activity 的生命周期图

注意事项:

1. onPause()和onStop()被调用的前提是: 打开了一个新的Activity!而前者是旧Activity还可见的状态;后者是旧Activity已经不可见!2. 另外,亲测:AlertDialog和PopWindow是不会触发上述两个回调方法的~
复制代码

2.Activity/ActionBarActivity/AppCompatActivity 的区别

在开始讲解创建 Activity 之前要说下这三个的一个区别:Activity 就不用说啦,后面这两个都是为了低版本兼容而提出的提出来的,他们都在 v7 包下,ActionBarActivity 已被废弃,从名字就知道,ActionBar~,而在 5.0 后,被 Google 弃用了,现在用 ToolBar...而我们现在在 Android Studio 创建一个 Activity 默认继承的会是:AppCompatActivity!当然你也可以只写 Activity,不过 AppCompatActivity 给我们提供了一些新的东西而已!两个选一个,Just you like~

3.Activity 的创建流程

好了,上面也说过,可以继承 Activity 和 AppCompatActivity,只不过后者提供了一些新的东西而已!另外,切记,Android 中的四大组件,只要你定义了,无论你用没用,都要在 AndroidManifest.xml 对这个组件进行声明,不然运行时程序会直接退出,报 ClassNotFindException...

4.onCreate()一个参数和两个参数的区别

相信用 as 的朋友在重写 Act 的 onCreate()方法时会发现,这玩意有两个参数:

可是正常的才只有一个参数啊:

恩呢,这就是 5.0 给我们提供的新的方法,要用它,先要在配置文件中为我们的 Activity 设置一个属性:

android:persistableMode="persistAcrossReboots"
复制代码

然后我们的 Activity 就拥有了持久化的能力了,一般我们会搭配另外两个方法来使用

public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState)public void onRestoreInstanceState(Bundle savedInstanceState, PersistableBundle persistentState)
复制代码

相信有些朋友对这两个方法名不陌生吧,前一个方法会在下述情形中被调用:

  • 点击 home 键回到主页或长按后选择运行其他程序

    按下电源键关闭屏幕

    启动新的 Activity

    横竖屏切换时,肯定会执行,因为横竖屏切换的时候会先销毁 Act,然后再重新创建 重要原则:当系统"未经你许可"时销毁了你的 activity,则 onSaveInstanceState 会被系统调用, 这是系统的责任,因为它必须要提供一个机会让你保存你的数据(你可以保存也可以不保存)。

而后一个方法,和 onCreate 同样可以从取出前者保存的数据:一般是在 onStart()和 onResume()之间执行!之所以有两个可以获取到保存数据的方法,是为了避免 Act 跳转而没有关闭,然后不走 onCreate()方法,而你又想取出保存数据~

说回来:说回这个 Activity 拥有了持久化的能力,增加的这个 PersistableBundle 参数令这些方法拥有了系统关机后重启的数据恢复能力!!而且不影响我们其他的序列化操作,卧槽,具体怎么实现的,暂时还不了解,可能是另外弄了个文件保存吧~!后面知道原理的话会告知下大家!另外,API 版本需要>=21,就是要 5.0 以上的版本才有效~

4.启动一个 Activity 的几种方式

在 Android 中我们可以通过下面两种方式来启动一个新的 Activity,注意这里是怎么启动,而非启动模式!!分为显示启动和隐式启动!

1. 显式启动:通过包名来启动,写法如下:

①最常见的:

startActivity(new Intent(当前Act.this,要启动的Act.class));
复制代码

②通过 Intent 的 ComponentName:

ComponentName cn = new ComponentName("当前Act的全限定类名","启动Act的全限定类名") ;Intent intent = new Intent() ;intent.setComponent(cn) ;startActivity(intent) ;
复制代码

初始化 Intent 时指定包名:

Intent intent = new Intent("android.intent.action.MAIN");intent.setClassName("当前Act的全限定类名","启动Act的全限定类名");startActivity(intent);
复制代码

2.隐式启动:通过 Intent-filter 的 Action,Category 或 data 来实现这个是通过 Intent 的 intent-filter**来实现的,这个 Intent 那章会详细讲解!这里知道个大概就可以了!



3. 另外还有一个直接通过包名启动 apk 的:

Intent intent = getPackageManager().getLaunchIntentForPackage("apk第一个启动的Activity的全限定类名") ;if(intent != null) startActivity(intent) ;
复制代码

5.横竖屏切换与状态保存的问题

前面也也说到了App横竖屏切换的时候会销毁当前的Activity然后重新创建一个,你可以自行在生命周期 的每个方法里都添加打印Log的语句,来进行判断,又或者设一个按钮一个TextView点击按钮后,修改TextView 文本,然后横竖屏切换,会神奇的发现TextView文本变回之前的内容了! 横竖屏切换时Act走下述生命周期:onPause-> onStop-> onDestory-> onCreate->onStart->onResume关于横竖屏切换可能遇到下述问题:
复制代码

1.先说下如何禁止屏幕横竖屏自动切换吧,很简单,在 AndroidManifest.xml 中为 Act 添加一个属性:android:screenOrientation,有下述可选值:


  • unspecified:默认值 由系统来判断显示方向.判定的策略是和设备相关的,所以不同的设备会有不同的显示方向。

  • landscape:横屏显示(宽比高要长)

  • portrait:竖屏显示(高比宽要长)

  • user:用户当前首选的方向

  • behind:和该 Activity 下面的那个 Activity 的方向一致(在 Activity 堆栈中的)

  • sensor:有物理的感应器来决定。如果用户旋转设备这屏幕会横竖屏切换。

  • nosensor:忽略物理感应器,这样就不会随着用户旋转设备而更改了("unspecified"设置除外)。

2.横竖屏时想加载不同的布局

1)准备两套不同的布局,Android 会自己根据横竖屏加载不同布局:创建两个布局文件夹:layout-land 横屏,layout-port 竖屏然后把这两套布局文件丢这两文件夹里,文件名一样,Android 就会自行判断,然后加载相应布局了!

2 )自己在代码中进行判断,自己想加载什么就加载什么:

我们一般是在 onCreate()方法中加载布局文件的,我们可以在这里对横竖屏的状态做下判断,关键代码如下:

if (this.getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE){       setContentView(R.layout.横屏);}  
else if (this.getResources().getConfiguration().orientation ==Configuration.ORIENTATION_PORTRAIT) { setContentView(R.layout.竖屏);}
复制代码

3. 如何让模拟器横竖屏切换

如果你的模拟器是 GM 的话。直接按模拟器上的切换按钮即可,原生模拟器可按 ctrl + f11/f12 切换!

4. 状态保存问题:

这个上面也说过了,通过一个 Bundle savedInstanceState 参数即可完成!三个核心方法:

onCreate(Bundle savedInstanceState);onSaveInstanceState(Bundle outState);onRestoreInstanceState(Bundle savedInstanceState);
复制代码

你只重写 onSaveInstanceState()方法,往这个 bundle 中写入数据,比如:

outState.putInt("num",1);
复制代码

这样,然后你在 onCreate 或者 onRestoreInstanceState 中就可以拿出里面存储的数据,不过拿之前要判断下是否为 null 哦!

savedInstanceState.getInt("num");
复制代码

6.系统给我们提供的常见的 Activity

好的,最后给大家附上一些系统给我们提供的一些常见的 Activtiy 吧!

//1.拨打电话// 给移动客服10086拨打电话Uri uri = Uri.parse("tel:10086");Intent intent = new Intent(Intent.ACTION_DIAL, uri);startActivity(intent);
//2.发送短信// 给10086发送内容为“Hello”的短信Uri uri = Uri.parse("smsto:10086");Intent intent = new Intent(Intent.ACTION_SENDTO, uri);intent.putExtra("sms_body", "Hello");startActivity(intent);
//3.发送彩信(相当于发送带附件的短信)Intent intent = new Intent(Intent.ACTION_SEND);intent.putExtra("sms_body", "Hello");Uri uri = Uri.parse("content://media/external/images/media/23");intent.putExtra(Intent.EXTRA_STREAM, uri);intent.setType("image/png");startActivity(intent);
//4.打开浏览器:// 打开Google主页Uri uri = Uri.parse("http://www.baidu.com");Intent intent = new Intent(Intent.ACTION_VIEW, uri);startActivity(intent);
//5.发送电子邮件:(阉割了Google服务的没戏!!!!)// 给someone@domain.com发邮件Uri uri = Uri.parse("mailto:someone@domain.com");Intent intent = new Intent(Intent.ACTION_SENDTO, uri);startActivity(intent);// 给someone@domain.com发邮件发送内容为“Hello”的邮件Intent intent = new Intent(Intent.ACTION_SEND);intent.putExtra(Intent.EXTRA_EMAIL, "someone@domain.com");intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");intent.putExtra(Intent.EXTRA_TEXT, "Hello");intent.setType("text/plain");startActivity(intent);// 给多人发邮件Intent intent=new Intent(Intent.ACTION_SEND);String[] tos = {"1@abc.com", "2@abc.com"}; // 收件人String[] ccs = {"3@abc.com", "4@abc.com"}; // 抄送String[] bccs = {"5@abc.com", "6@abc.com"}; // 密送intent.putExtra(Intent.EXTRA_EMAIL, tos);intent.putExtra(Intent.EXTRA_CC, ccs);intent.putExtra(Intent.EXTRA_BCC, bccs);intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");intent.putExtra(Intent.EXTRA_TEXT, "Hello");intent.setType("message/rfc822");startActivity(intent);
//6.显示地图:// 打开Google地图中国北京位置(北纬39.9,东经116.3)Uri uri = Uri.parse("geo:39.9,116.3");Intent intent = new Intent(Intent.ACTION_VIEW, uri);startActivity(intent);
//7.路径规划// 路径规划:从北京某地(北纬39.9,东经116.3)到上海某地(北纬31.2,东经121.4)Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=39.9 116.3&daddr=31.2 121.4");Intent intent = new Intent(Intent.ACTION_VIEW, uri);startActivity(intent);
//8.多媒体播放:Intent intent = new Intent(Intent.ACTION_VIEW);Uri uri = Uri.parse("file:///sdcard/foo.mp3");intent.setDataAndType(uri, "audio/mp3");startActivity(intent);
//获取SD卡下所有音频文件,然后播放第一首=-= Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");Intent intent = new Intent(Intent.ACTION_VIEW, uri);startActivity(intent);
//9.打开摄像头拍照:// 打开拍照程序Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); startActivityForResult(intent, 0);// 取出照片数据Bundle extras = intent.getExtras(); Bitmap bitmap = (Bitmap) extras.get("data");
//另一种://调用系统相机应用程序,并存储拍下来的照片Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); time = Calendar.getInstance().getTimeInMillis();intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(Environment.getExternalStorageDirectory().getAbsolutePath()+"/tucue", time + ".jpg")));startActivityForResult(intent, ACTIVITY_GET_CAMERA_IMAGE);
//10.获取并剪切图片// 获取并剪切图片Intent intent = new Intent(Intent.ACTION_GET_CONTENT);intent.setType("image/*");intent.putExtra("crop", "true"); // 开启剪切intent.putExtra("aspectX", 1); // 剪切的宽高比为1:2intent.putExtra("aspectY", 2);intent.putExtra("outputX", 20); // 保存图片的宽和高intent.putExtra("outputY", 40); intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp"))); // 保存路径intent.putExtra("outputFormat", "JPEG");// 返回格式startActivityForResult(intent, 0);// 剪切特定图片Intent intent = new Intent("com.android.camera.action.CROP"); intent.setClassName("com.android.camera", "com.android.camera.CropImage"); intent.setData(Uri.fromFile(new File("/mnt/sdcard/temp"))); intent.putExtra("outputX", 1); // 剪切的宽高比为1:2intent.putExtra("outputY", 2);intent.putExtra("aspectX", 20); // 保存图片的宽和高intent.putExtra("aspectY", 40);intent.putExtra("scale", true);intent.putExtra("noFaceDetection", true); intent.putExtra("output", Uri.parse("file:///mnt/sdcard/temp")); startActivityForResult(intent, 0);
//11.打开Google Market // 打开Google Market直接进入该程序的详细页面Uri uri = Uri.parse("market://details?id=" + "com.demo.app");Intent intent = new Intent(Intent.ACTION_VIEW, uri);startActivity(intent);
//12.进入手机设置界面:// 进入无线网络设置界面(其它可以举一反三) Intent intent = new Intent(android.provider.Settings.ACTION_WIRELESS_SETTINGS); startActivityForResult(intent, 0);
//13.安装apk:Uri installUri = Uri.fromParts("package", "xxx", null); returnIt = new Intent(Intent.ACTION_PACKAGE_ADDED, installUri);
//14.卸载apk:Uri uri = Uri.fromParts("package", strPackageName, null); Intent it = new Intent(Intent.ACTION_DELETE, uri); startActivity(it);
//15.发送附件:Intent it = new Intent(Intent.ACTION_SEND); it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text"); it.putExtra(Intent.EXTRA_STREAM, "file:///sdcard/eoe.mp3"); sendIntent.setType("audio/mp3"); startActivity(Intent.createChooser(it, "Choose Email Client"));
//16.进入联系人页面:Intent intent = new Intent();intent.setAction(Intent.ACTION_VIEW);intent.setData(People.CONTENT_URI);startActivity(intent);
//17.查看指定联系人:Uri personUri = ContentUris.withAppendedId(People.CONTENT_URI, info.id);//info.id联系人IDIntent intent = new Intent();intent.setAction(Intent.ACTION_VIEW);intent.setData(personUri);startActivity(intent);
复制代码


发布于: 刚刚阅读数: 4
用户头像

向阳逐梦

关注

人生享受编程,编程造就人生! 2022-06-01 加入

某公司芯片测试工程师,嵌入式开发工程师,InfoQ签约作者,阿里云星级博主,华为云·云享专家。座右铭:向着太阳,追逐梦想!

评论

发布
暂无评论
Activity初学乍练_android_向阳逐梦_InfoQ写作社区