[关闭]
@coder-pig 2015-08-06T07:09:14.000000Z 字数 9114 阅读 1967

Android基础入门教程——4.1.1 Activity初学乍练

Android基础入门教程


本节引言:

本节开始讲解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的创建流程

PS:

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


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

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

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

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

  1. android:persistableMode="persistAcrossReboots"

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

  1. public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState)
  2. public void onRestoreInstanceState(Bundle savedInstanceState, PersistableBundle persistentState)

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

  1. 点击home键回到主页或长按后选择运行其他程序
  2. 按下电源键关闭屏幕
  3. 启动新的Activity
  4. 横竖屏切换时,肯定会执行,因为横竖屏切换的时候会先销毁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()方法中加载布局文件的,我们可以在这里对横竖屏的状态做下判断,关键代码如下:

  1. if (this.getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE){
  2. setContentView(R.layout.横屏);
  3. }
  4. else if (this.getResources().getConfiguration().orientation ==Configuration.ORIENTATION_PORTRAIT) {
  5. setContentView(R.layout.竖屏);
  6. }

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


4. 状态保存问题:
这个上面也说过了,通过一个Bundle savedInstanceState参数即可完成!
三个核心方法:


  1. onCreate(Bundle savedInstanceState);
  2. onSaveInstanceState(Bundle outState);
  3. onRestoreInstanceState(Bundle savedInstanceState);

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

outState.putInt("num",1);

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

savedInstanceState.getInt("num");

然后想干嘛就干嘛~


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

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

  1. //1.拨打电话
  2. // 给移动客服10086拨打电话
  3. Uri uri = Uri.parse("tel:10086");
  4. Intent intent = new Intent(Intent.ACTION_DIAL, uri);
  5. startActivity(intent);
  6. //2.发送短信
  7. // 给10086发送内容为“Hello”的短信
  8. Uri uri = Uri.parse("smsto:10086");
  9. Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
  10. intent.putExtra("sms_body", "Hello");
  11. startActivity(intent);
  12. //3.发送彩信(相当于发送带附件的短信)
  13. Intent intent = new Intent(Intent.ACTION_SEND);
  14. intent.putExtra("sms_body", "Hello");
  15. Uri uri = Uri.parse("content://media/external/images/media/23");
  16. intent.putExtra(Intent.EXTRA_STREAM, uri);
  17. intent.setType("image/png");
  18. startActivity(intent);
  19. //4.打开浏览器:
  20. // 打开Google主页
  21. Uri uri = Uri.parse("http://www.baidu.com");
  22. Intent intent  = new Intent(Intent.ACTION_VIEW, uri);
  23. startActivity(intent);
  24. //5.发送电子邮件:(阉割了Google服务的没戏!!!!)
  25. // 给someone@domain.com发邮件
  26. Uri uri = Uri.parse("mailto:someone@domain.com");
  27. Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
  28. startActivity(intent);
  29. // 给someone@domain.com发邮件发送内容为“Hello”的邮件
  30. Intent intent = new Intent(Intent.ACTION_SEND);
  31. intent.putExtra(Intent.EXTRA_EMAIL, "someone@domain.com");
  32. intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
  33. intent.putExtra(Intent.EXTRA_TEXT, "Hello");
  34. intent.setType("text/plain");
  35. startActivity(intent);
  36. // 给多人发邮件
  37. Intent intent=new Intent(Intent.ACTION_SEND);
  38. String[] tos = {"1@abc.com", "2@abc.com"}; // 收件人
  39. String[] ccs = {"3@abc.com", "4@abc.com"}; // 抄送
  40. String[] bccs = {"5@abc.com", "6@abc.com"}; // 密送
  41. intent.putExtra(Intent.EXTRA_EMAIL, tos);
  42. intent.putExtra(Intent.EXTRA_CC, ccs);
  43. intent.putExtra(Intent.EXTRA_BCC, bccs);
  44. intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
  45. intent.putExtra(Intent.EXTRA_TEXT, "Hello");
  46. intent.setType("message/rfc822");
  47. startActivity(intent);
  48. //6.显示地图:
  49. // 打开Google地图中国北京位置(北纬39.9,东经116.3)
  50. Uri uri = Uri.parse("geo:39.9,116.3");
  51. Intent intent = new Intent(Intent.ACTION_VIEW, uri);
  52. startActivity(intent);
  53. //7.路径规划
  54. // 路径规划:从北京某地(北纬39.9,东经116.3)到上海某地(北纬31.2,东经121.4)
  55. Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=39.9 116.3&daddr=31.2 121.4");
  56. Intent intent = new Intent(Intent.ACTION_VIEW, uri);
  57. startActivity(intent);
  58. //8.多媒体播放:
  59. Intent intent = new Intent(Intent.ACTION_VIEW);
  60. Uri uri = Uri.parse("file:///sdcard/foo.mp3");
  61. intent.setDataAndType(uri, "audio/mp3");
  62. startActivity(intent);
  63. //获取SD卡下所有音频文件,然后播放第一首=-= 
  64. Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");
  65. Intent intent = new Intent(Intent.ACTION_VIEW, uri);
  66. startActivity(intent);
  67. //9.打开摄像头拍照:
  68. // 打开拍照程序
  69. Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
  70. startActivityForResult(intent, 0);
  71. // 取出照片数据
  72. Bundle extras = intent.getExtras();
  73. Bitmap bitmap = (Bitmap) extras.get("data");
  74. //另一种:
  75. //调用系统相机应用程序,并存储拍下来的照片
  76. Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
  77. time = Calendar.getInstance().getTimeInMillis();
  78. intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(Environment
  79. .getExternalStorageDirectory().getAbsolutePath()+"/tucue", time + ".jpg")));
  80. startActivityForResult(intent, ACTIVITY_GET_CAMERA_IMAGE);
  81. //10.获取并剪切图片
  82. // 获取并剪切图片
  83. Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
  84. intent.setType("image/*");
  85. intent.putExtra("crop", "true"); // 开启剪切
  86. intent.putExtra("aspectX", 1); // 剪切的宽高比为1:2
  87. intent.putExtra("aspectY", 2);
  88. intent.putExtra("outputX", 20); // 保存图片的宽和高
  89. intent.putExtra("outputY", 40);
  90. intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp"))); // 保存路径
  91. intent.putExtra("outputFormat", "JPEG");// 返回格式
  92. startActivityForResult(intent, 0);
  93. // 剪切特定图片
  94. Intent intent = new Intent("com.android.camera.action.CROP");
  95. intent.setClassName("com.android.camera", "com.android.camera.CropImage");
  96. intent.setData(Uri.fromFile(new File("/mnt/sdcard/temp")));
  97. intent.putExtra("outputX", 1); // 剪切的宽高比为1:2
  98. intent.putExtra("outputY", 2);
  99. intent.putExtra("aspectX", 20); // 保存图片的宽和高
  100. intent.putExtra("aspectY", 40);
  101. intent.putExtra("scale", true);
  102. intent.putExtra("noFaceDetection", true);
  103. intent.putExtra("output", Uri.parse("file:///mnt/sdcard/temp"));
  104. startActivityForResult(intent, 0);
  105. //11.打开Google Market
  106. // 打开Google Market直接进入该程序的详细页面
  107. Uri uri = Uri.parse("market://details?id=" + "com.demo.app");
  108. Intent intent = new Intent(Intent.ACTION_VIEW, uri);
  109. startActivity(intent);
  110. //12.进入手机设置界面:
  111. // 进入无线网络设置界面(其它可以举一反三) 
  112. Intent intent = new Intent(android.provider.Settings.ACTION_WIRELESS_SETTINGS); 
  113. startActivityForResult(intent, 0);
  114. //13.安装apk:
  115. Uri installUri = Uri.fromParts("package", "xxx", null);
  116. returnIt = new Intent(Intent.ACTION_PACKAGE_ADDED, installUri);
  117. //14.卸载apk:
  118. Uri uri = Uri.fromParts("package", strPackageName, null);
  119. Intent it = new Intent(Intent.ACTION_DELETE, uri);
  120. startActivity(it);
  121. //15.发送附件:
  122. Intent it = new Intent(Intent.ACTION_SEND);
  123. it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");
  124. it.putExtra(Intent.EXTRA_STREAM, "file:///sdcard/eoe.mp3");
  125. sendIntent.setType("audio/mp3");
  126. startActivity(Intent.createChooser(it, "Choose Email Client"));
  127. //16.进入联系人页面:
  128. Intent intent = new Intent();
  129. intent.setAction(Intent.ACTION_VIEW);
  130. intent.setData(People.CONTENT_URI);
  131. startActivity(intent);
  132. //17.查看指定联系人:
  133. Uri personUri = ContentUris.withAppendedId(People.CONTENT_URI, info.id);//info.id联系人ID
  134. Intent intent = new Intent();
  135. intent.setAction(Intent.ACTION_VIEW);
  136. intent.setData(personUri);
  137. startActivity(intent);

本节小结:

好吧,写着写着就不像入门教程了,哈哈,不过学多点没事的,本节初窥门径就到这里吧~下节我们会继续来研究这个Activity,比如数据传递,启动模式等~敬请期待~

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注