Android理解:显式和隐式Intent

<p>Intent分两种:显式(Explicit intent)和隐式(Implicit intent)。</p><p>一、显式(设置Component)</p><p>显式,即直接指定需要打开的activity对应的类。 以下多种方式都是一样的,实际上都是设置Component直接指定Activity类的显式Intent,由MainActivity跳转到SecondActivity:</p><p>1、构造方法传入Component,最常用的方式</p><p>Intent intent = new Intent(this, SecondActivity.class); </p><p>startActivity(intent);</p><p>2、setComponent方法</p><p>ComponentName componentName = new ComponentName(this, SecondActivity.class); </p><p>// 或者ComponentName componentName = new ComponentName(this, &quot;com.example.app016.SecondActivity&quot;); </p><p>// 或者ComponentName componentName = new ComponentName(this.getPackageName(), &quot;com.example.app016.SecondActivity&quot;); </p><p></p><p>Intent intent = new Intent(); </p><p>intent.setComponent(componentName); </p><p>startActivity(intent);</p><p>3、setClass/setClassName方法</p><p>Intent intent = new Intent(); </p><p></p><p>intent.setClass(this, SecondActivity.class); </p><p>// 或者intent.setClassName(this, &quot;com.example.app016.SecondActivity&quot;); </p><p>// 或者intent.setClassName(this.getPackageName(), &quot;com.example.app016.SecondActivity&quot;); </p><p> </p><p>startActivity(intent); </p><p>显式Intent通过Component可以直接设置需要调用的Activity类,可以唯一确定一个Activity,意图特别明确,所以是显式的。设置这个类的方式可以是Class对象(如SecondActivity.class),</p><p>也可以是包名加类名的字符串(如&quot;com.example.app016.SecondActivity&quot;)。这个很好理解,在应用程序内部跳转界面常用这种方式。</p><p></p><p>二、隐式</p><p>隐式,即不是像显式的那样直接指定需要调用的Activity,隐式不明确指定启动哪个Activity,而是设置Action、Data、Category,让系统来筛选出合适的Activity。</p><p>筛选是根据所有的&lt;intent-filter&gt;来筛选。</p><p>下面以Action为例:</p><p>AndroidManifest.xml文件中,首先被调用的Activity要有一个带有&lt;intent-filter&gt;并且包含&lt;action&gt;的Activity,设定它能处理的Intent,并且category设为&quot;android.intent.category.DEFAULT&quot;。</p><p>action的name是一个字符串,可以自定义,例如我在这里设成&quot;abcdefg&quot;:</p><p>&lt;activity </p><p> android:name=&quot;com.example.app016.SecondActivity&quot;&gt; </p><p> &lt;intent-filter&gt; </p><p> &lt;action android:name=&quot;abcdefg&quot;/&gt; </p><p> &lt;category android:name=&quot;android.intent.category.DEFAULT&quot;/&gt; </p><p> &lt;/intent-filter&gt; </p><p>&lt;/activity&gt; </p><p>然后,在MainActivity,才可以通过这个action name找到上面的Activity。</p><p>下面两种方式分别通过setAction和构造方法方法设置Action,两种方式效果相同。</p><p>1、setAction方法</p><p>Intent intent = new Intent(); </p><p>intent.setAction(&quot;abcdefg&quot;); </p><p>startActivity(intent); </p><p>2、构造方法直接设置Action</p><p>Intent intent = new Intent(&quot;abcdefg&quot;); </p><p>startActivity(intent);</p><p>通过设置Action字符串,表明自己的意图,即我想干嘛,需要由系统解析,找到能够处理这个Intent的Activity并启动。</p><p>比如我想打电话,则可以设置Action为&quot;android.intent.action.DIAL&quot;字符串,表示打电话的意图,系统会找到能处理这个意图的Activity,例如调出拨号面板。</p><p></p><p>有几点需要注意:</p><p>1、 这个Activity其他应用程序也可以调用,只要使用这个Action字符串。这样应用程序之间交互就很容易了,例如手机QQ可以调用QQ空间,可以调用腾讯微博等。</p><p>因为如此,为了防止应用程序之间互相影响,一般命名方式是包名+Action名,例如这里命名&quot;abcdefg&quot;就很不合理了,就应该改成&quot;com.example.app016.MyTest&quot;。</p><p>2、 当然,你可以在自己的程序中调用其他程序的Action。 例如可以在自己的应用程序中调用拨号面板:</p><p>Intent intent = new Intent(Intent.ACTION_DIAL); </p><p>// 或者Intent intent = new Intent(&quot;android.intent.action.DIAL&quot;); </p><p>// Intent.ACTION_DIAL是内置常量,值为&quot;android.intent.action.DIAL&quot; </p><p>startActivity(intent); </p><p><img src="/up_pic/201812/140427111255.png" title="140427111255.png" alt="1.png"/></p><p>3、一个Activity可以处理多种Action 只要你的应用程序够牛逼,一个Activity可以看网页,打电话,发短信,发邮件。。。当然可以。 Intent的Action只要是其中之一,就可以打开这个Activity。</p><p></p><p>activity </p><p> android:name=&quot;com.example.app016.SecondActivity&quot;&gt; </p><p> &lt;intent-filter&gt; </p><p> &lt;!-- 可以处理下面三种Intent --&gt; </p><p> &lt;action android:name=&quot;com.example.app016.SEND_EMAIL&quot;/&gt; </p><p> &lt;action android:name=&quot;com.example.app016.SEND_MESSAGE&quot;/&gt; </p><p> &lt;action android:name=&quot;com.example.app016.DAIL&quot;/&gt; </p><p> &lt;category android:name=&quot;android.intent.category.DEFAULT&quot; /&gt; </p><p> &lt;/intent-filter&gt; </p><p>&lt;/activity&gt; </p><p>对于一个Action字符串,系统有可能会找到一个Activity能处理这个Action,也有可能找到多个Activity,也可能一个都找不到。</p><p>1、找到一个Activity</p><p>很简单,直接打开这个Activity。这个不需要解释。</p><p>2、找到多个Acyivity</p><p>系统会提示从多个activity中选择一个打开。</p><p>例如我们自己开发一个拨号面板应用程序,可以设置activity的&lt;intent-filter&gt;中Action name为&quot;android.intent.action.DIAL&quot;,这样别的程序调用拨号器时,</p><p>用户可以从Android自带的拨号器和我们自己开发的拨号器中选择。</p><p>&lt;activity </p><p> android:name=&quot;com.example.app016.SecondActivity&quot;&gt; </p><p> &lt;intent-filter&gt; </p><p> &lt;action android:name=&quot;android.intent.action.DIAL&quot;/&gt; </p><p> &lt;category android:name=&quot;android.intent.category.DEFAULT&quot; /&gt; </p><p> &lt;/intent-filter&gt; </p><p>&lt;/activity&gt; </p><p><img src="/up_pic/201812/140427454867.jpg" title="140427454867.jpg" alt="20130908184144625.jpg"/></p><p>这也就是当Android手机装上UC浏览器后,打开网页时会弹出选择Android自带浏览器还是UC浏览器,可能都会遇到过。</p><p>3、一个Activity都没找到</p><p>一个都没找到的话,程序就会出错,会抛出ActivityNotFoundException。比如随便写一个action字符串:</p><p>Intent intent = new Intent(&quot;asasasas&quot;); </p><p>startActivity(intent); </p><p><img src="/up_pic/201812/140428015066.jpg" title="140428015066.jpg" alt="aaa.jpg"/></p><p>所以应该注意try catch异常。</p><p>Intent intent = new Intent(&quot;asasasas&quot;); </p><p>try </p><p>{ </p><p> startActivity(intent); </p><p>} </p><p>catch(ActivityNotFoundException e) </p><p>{ </p><p> Toast.makeText(this, &quot;找不到对应的Activity&quot;, Toast.LENGTH_SHORT).show(); </p><p>} </p><p>或者也可以使用Intent的resolveActivity方法判断这个Intent是否能找到合适的Activity,如果没有,则不再startActivity,或者可以直接禁用用户操作的控件。</p><p>Intent intent = new Intent(Intent.ACTION_DIAL); </p><p>if(intent.resolveActivity(getPackageManager()) == null) </p><p>{ </p><p> // 设置控件不可用 </p><p>} </p><p>注意resolveActivity方法返回值就是显式Intent上面讲到的ComponentName对象,一般情况下也就是系统找到的那个Activity。</p><p>但是如果有多个Activity可供选择的话,则返回的Component是com.android.internal.app.ResolverActivity,也就是用户选择Activity的那个界面对应的Activity,这里不再深究。</p><p>Intent intent = new Intent(Intent.ACTION_DIAL); </p><p>ComponentName componentName = intent.resolveActivity(getPackageManager()); </p><p>if(componentName != null) </p><p>{ </p><p> String className = componentName.getClassName(); </p><p> Toast.makeText(this, className, Toast.LENGTH_SHORT).show(); </p><p>} </p>
RangeTime:0.004975s
RangeMem:207.27 KB
返回顶部 留言