android 四大组件作用及相关知识点补充

今天被面试官被BB了,知道andorid四大组件把,知道为啥是四大组件,为啥不是3大组件,2大组件,然后我说,这,,不是就跟你有2个手一样嘛,

回去后仔细想想,他面试官是不是要问我这4个的作用呢

首先学android的都知道四大组件是activity,service,BroadCastReceive,ContentProvide;

Activity

具体可结合Activity中常用知识点总结去看,这里只是做个简单的概括

是一种展示型组件,主要是向用户展示一个界面,接受用户输入的信息和用户进行交互,对用户来说,Activity就是Android应用的全部,因为其他三大组件对用户来说是不可感知的。

通过intent启动,其中intent启动分显式启动和隐式启动

显式启动实例:

 Intent intent = new Intent(MainActivity.this,secondActivity.class);
 startActivity(intent1);

显式intent非常简单,直接创建一个intent对象,第一个参数代表启动活动的上下文,第二个代表想要启动的目标,之后调用startactivity即可。

还有种就是ComponentName的启动,具体看ComponentName使用,例如:

 ComponentName chatActivity =new ComponentName("com.npf.chat", "com.npf.chat.ui.ChatActivity");
 
    Intent intent =new Intent();
 
    intent.setComponent(chatActivity);
 
    startActivity(intent);

还有种是setClass/setClassName方法的

Intent intent = new Intent();  
  
intent.setClass(this, SecondActivity.class);  
// 或者intent.setClassName(this, "com.example.SecondActivity");  
// 或者intent.setClassName(this.getPackageName(), "com.example.SecondActivity");  
          
startActivity(intent);  

隐式intent

隐式,即不是像显式的那样直接指定需要调用的Activity,隐式不明确指定启动哪个Activity,而是设置Action、Data、Category,让系统来筛选出合适的Activity。

筛选是根据所有的<intent-filter>来筛选。

例如:

<activity  
    android:name="com.example.SecondActivity">  
    <intent-filter>  
        <action android:name="test"/>  
        <category android:name="android.intent.category.DEFAULT"/>  
    </intent-filter>  
</activity>  

其中action和category更多具体参数值参考Intent中的Action、Category,这里采用自定义的

这样我们就可以通过action来找到这个Activity了

Intent intent = new Intent();  
intent.setAction("test");  
startActivity(intent);  

//或者

Intent intent = new Intent("test");  
startActivity(intent); 

还有个骚玩法

   <activity android:name=".SecondActivity">
            <intent-filter>
                <action android:name="android.intent.action.DIAL" />
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </activity>

然后其他应用通过action调用这个SecondActivity

  Intent intent=new Intent();
        intent.setAction("android.intent.action.DIAL");
        startActivity(intent);

就会这样子

牛逼吧,还可以这么玩,哎,骚的很

有几点需要注意:

1、 这个Activity其他应用程序也可以调用,只要使用这个Action字符串。这样应用程序之间交互就很容易了,例如手机QQ可以调用QQ空间,可以调用腾讯微博等。

因为如此,为了防止应用程序之间互相影响,一般命名方式是包名+Action名,例如这里命名"test"就很不合理了,就应该改成"com.example.Test"。

2、如果Intent请求或<intent-filter>中没有说明具体的Action类型,那么会出现下面两种情况。如果<intent-filter>中没有包含任何Action类型,那么无论什么Intent请求都无法和这条<intent-filter>匹配。 反之,如果Intent请求中没有设定Action类型,那么只要<intent-filter>中包含有Action类型,这个Intent请求就将顺利地通过<intent-filter>的行为测试。

如果该activity想要通过隐式intent方式激活,那么不能没有任何category设置,至少包含一个android.intent.category.DEFAULT

<activity android:name=".SecondActivity"
    android:exported="false">

设置了这个之后发现,其他应用就无法启动当前action对应的这个界面了,默认export是开启的,如果设置了对应的action,和 <category android:name="android.intent.category.DEFAULT" />这个属性的话,默认是可以多进程访问的,想不被其他应用使用的话,就得设置为false

总结来说就是:

  • Activity的作用
    Activity是一种展示型组件,主要是向用户展示一个界面,并且可以接收用户的输入信息从而和用户进行交互。对用户来说,Activity就是Android应用的全部,因为其他三大组件对用户来说是不可感知的。
    Activity的启动由Intent触发,其中Intent分为显式启动和隐式启动。
    Activity组件的主要作用是展示一个界面并和用户交互,它扮演的是一种前台界面的角色。
  • Activity能否开启多进程
    Activity可以通过在AndroidManifest中通过添加android:process的属性来开启多进程。若该Activity需要被外界调用,则需要设置android:exported=true这个属性。android:exported默认是false的,除非该Activity设置了Intent-Filter。设置了android:exported=true

Service

  • Service的作用
    Service是一种计算型组件,用于在后台执行一系列计算任务。由于Service组件工作在后台,因此用户无法直接感知到它的存在。Service组件和Activity组件不同,Activity组件只有一种运行模式,即Activity处于启动状态,但是Service组件却有两种状态:启动状态和绑定状态。
    Service组件处于启动状态时,它的内部可以执行一些后台计算,并且不需要和外界有直接的交互。Service处于绑定状态,Service内部同样也可以执行后台计算,但是处于这种状态的Service可以很方便地和外界进行通信。
    Service组件的主要作用是在后台执行计算任务,执行任务的结果可以和外界进行通信。
  • Service能否开启多进程
    Service可以通过在AndroidManifest中通过添加android:process的属性来开启多进程。若该Service需要被外界调用,则需要设置android:exported=true这个属性。android:exported默认是false的,除非该Service设置了Intent-Filter。设置了android:exported=true

BroadcastReceiver

  • BroadcastReceiver的作用
    BroadcastReceiver是一种消息型组件,用于在不同组件乃至不同应用之间传递消息。BroadcastReceiver同样无法被用户所感知,因为它工作在系统内部。
    BroadcastReceiver也叫做广播,广播的注册方式有两种:静态注册和动态注册。静态注册指在AndroidManifest中注册广播,这种广播在应用安装时被系统解析,此种形式的广播不需要应用启动就可以接收到相应的广播。动态广播需要通过Context.registerReceiver()来实现,并且在不需要的时候通过Context.unRegisterReceiver()解除广播,此种形态的广播必须要应用启动才能注册并接收广播。
    BroadcastReceiver组件的主要作用是消息的传递,该消息的传递可以在应用内,也可以在应用之间,它的角色是一个消息的传递者。
  • BroadcastReceiver能否开启多进程
    BroadcastReceiver可以通过在AndroidManifest中通过添加android:process的属性来开启多进程。若该BroadcastReceiver需要被外界调用,则需要设置android:exported=true这个属性。android:exported默认是false的,除非该BroadcastReceiver设置了Intent-Filter。设置了android:exported=true,同时也可以设置android:permission来限制外部应用调用。
  • 多应用调用BroadcastReceiver的例子
    发送端代码:
public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Intent intent = new Intent("com.test.abc");
        intent.putExtra("msg", "hello world");
        sendBroadcast(intent, "com.test.mypermission");
    }
}

发送端代码挺简单的,就使用sendBroadcast发送一个Intent对象,其中的Intent对象的action名为“com.test.abc”,并且携带了key为“msg”,value为“hello world”的信息。注意sendBroadcast方法的第二个参数为"com.test.mypermission",其实第二个参数代表的是permission,也就是该广播具有permission权限,当接收端的Receiver具有permission权限时,才能接收到该广播。下面我们看一下接收端的代码:

接收端代码:

//自定义的一个广播接收器
public class MyBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        String msg = intent.getStringExtra("msg");
        Log.d("ABC", "msg from remote:" + msg);
    }
}

// AndroidManifest:
<permission
    android:name="com.test.mypermission"/>  <!--声明自定义权限-->
<uses-permission android:name="com.test.mypermission"/> <!--引用自己声明的权限-->

<receiver
    android:name=".MyBroadcastReceiver"
    android:exported="true">
    <intent-filter>
        <action android:name="com.test.abc"/>
    </intent-filter>
</receiver>

接收端的代码逻辑分为三部分:
1、首先自定义一个广播接收器,在onReceiver方法中接收发送端发过来的请求信息。
2、然后我们需要在AndroidManifest中注册该广播接收器,由于发送端和接收端属于两个不同的应用,所以需要声明android:exported=true,并且添加action标签,表明只接收action为"com.test.abc"的广播。
3、最后由于发送端发送广播时定义了一个permission权限,所以我们需要用<permission />标签声明该权限,并且用<uses-permisssion />标签引用该权限。

ContentProvider

  • ContentProvider的作用
    ContentProvider是一种数据共享型组件,用于向其他组件乃至其他应用共享数据。同样的,它也无法被用户所感知。
    对于ContentProvider组件来说,它的内部需要实现增删改查这四种操作。需要注意的是,ContentProvider内部的insert、delete、update和query方法需要处理好线程同步,因为这几个方法都是在Binder线程池中被调用的。ContentProvider组件不需要手动停止。
    ContentProvider组件的主要作用是作为一个平台,提供数据的共享,并且提供数据的增删改查功能。主要应用于应用之间的数据共享场景。
  • ContentProvider是否可以开启多进程
    ContentProvider可以通过在AndroidManifest中通过添加android:process的属性来开启多进程。若该ContentProvider需要被外界调用,则需要设置android:exported=true这个属性。当android sdk的minSdkVersion或者targetSdkVersion为16或者以下时,android:exported默认是true的。当android sdk的minSdkVersion或者targetSdkVersion为17或者以上时,android:exported默认是false。设置了android:exported=true,同时也可以设置android:permission来限制外部应用调用。

由此可以知道4大组件基本特性都是可以跨进程的,而且都是要在配置文件中配置的,然后activity这种不用说了,必须要的,其他3个看情况进行选择

参考文档:

https://www.jianshu.com/p/4d506f130342

https://blog.csdn.net/sinat_29355599/article/details/81002540

https://www.cnblogs.com/liaojie970/p/5827433.html

猜你喜欢

转载自blog.csdn.net/z936689039/article/details/108477855