• Android基础新手教程——4.1.1 Activity初学乍练


    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设置一个属性:

    android:persistableMode="persistAcrossReboots"

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

    public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState)
    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的语句,来进行推断,又或者设一个button一个TextView点击button后,改动TextView
    文本,然后横竖屏切换,会奇妙的发现TextView文本变回之前的内容了!
    横竖屏切换时Act走下述生命周期:
    onPause-> onStop-> onDestory-> onCreate->onStart->onResume
    关于横竖屏切换可能遇到下述问题:


    1.先说下怎样禁止屏幕横竖屏自己主动切换吧,非常easy,在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的话。

    直接按模拟器上的切换button就可以,原生模拟器可按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:2
    intent.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:2
    intent.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联系人ID
    Intent intent = new Intent();
    intent.setAction(Intent.ACTION_VIEW);
    intent.setData(personUri);
    startActivity(intent);

    本节小结:

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

  • 相关阅读:
    没有被实例化的类 中的 非static成员函数竟然也可以被调用。。。前提是该成员函数没有用到成员变量
    c++注意
    关于类大小的小试验
    C语言|博客作业02
    在C#中进行时间和时间戳的转换
    正则表达式中匹配中括号 [ ]
    在C#中将对象序列化成Json格式
    在MSSQL中的简单数据类型递归
    HTML中padding和margin的区别和用法
    C#中的对称加密
  • 原文地址:https://www.cnblogs.com/liguangsunls/p/7234626.html
Copyright © 2020-2023  润新知