By 高焕堂   

 

学习Android最核心的Context主板模式

 

1.  认识AndroidContext主板模式

   我们从典型的主板模式(MB pattern)出发,来了解Android的Context主板模式设计。于是,架构师心中有个Context接口,并且设计一个抽象父类来实现(Implement)该接口,如下图: 

              

      这就是典型的软件主板了。如下图所示:

 

          

     由于接口就是一种退化型的<类>。所以上图里的接口,可以定义于一个类之中,成为一个<接口类>。于是此<接口类>及其<实现类>,就组合成为一个软件主板了。如下图: 

 

          

    此时,<E>的operation()函数可以去调用IE_Impl类里的operation()函数,执行其实现代码了,它(即<E>)的operation()函数就不必重复撰写operation()函数的实现代码了。这项途径,在软件设计上称为”委托”(Delegation)。也可以将IE_Impl类独立出来,如下图所示:

           

    本来是<E>类必须撰写operation()函数的实现代码,为了避免重复撰写实现代码,<E>类就就转而”委托”IE_Impl类,调用它的operation()实现代码。这种架构设计的效益是,<E>类不必再重复撰写operation()函数的实现代码,可节省开发、维护实现代码的负担。当<E>类的个数愈多时,这种架构设计的效益就愈大。如下图所示:

 

      图里的黑色菱形符号(◆)表示<E>类都会创建一个IE_Impl类的对象,然后<E>类的operation()函数会调用IE_Impl类的operation()的实现代码。也就是,<E>类将它自己的operation()函数的实现工作委托给IE_Impl类去做了。

    

2. Android的Context通用性接口

      在Android框架里,俗称”慈禧太后”的至高权力Context接口,就依循上一小节所叙述的架构,其的详细内涵如下图: 

   

     这个Context通用性接口,是Android框架里最具有权力地位的接口。例如,只要你能取得这个Context接口,就能调用极具权力象征的startActivity()函数。

  

    从上图可以看到,除了startActivity()函数之外,还有startService()函数、bindService()函数等,都是Android平台上最常用的函数。在Android里,上图的实现代码片段如下: 

public class ContextWrapper extends Context {

     Context mBase;

     public ContextWrapper(Context base) {

         mBase = base;

     }

    //………

    @Override public void startActivity(Intent intent) {

           mBase.startActivity(intent);

      }

     @Override public ComponentName startService(Intent service) {

           return mBase.startService(service);

      }

     //………

} 

    其中,mBase是指针,用来指向ContextImpl对象的Context接口。也就是说,ContextWrapper对象都会含有一个ContextImpl对象的指针。这让ContextWrapper能透过mBase而调用ContextImpl类里的实现代码。例如, 

    public void startActivity(Intent intent) {

           mBase.startActivity(intent);

      } 

就是”委托”(Delegation)机制的实际运行了。 

 

class ContextImpl extends Context {

//...........

@Override public void startActivity(Intent intent) {

        warnIfCallingFromSystemProcess();

        startActivity(intent, null);

    }

@Override

public void startActivity(Intent intent, Bundle options) {

         // 实现代码

    }

//………

@Override

public ComponentName startService(Intent service) {

        warnIfCallingFromSystemProcess();

        return startServiceAsUser(service, mUser);

    }

@Override

public ComponentName startServiceAsUser(Intent service, UserHandle user) {

       // 实现代码

    }    

    // ……..

} 

 

3.  举例说明Context通用性接口的应用

      在Android框架里,这俗称为”慈禧太后”至高权力的Context接口,其结构,也可以表示如下:

    

    无论是Android手机或是PC等其它设备,皆能透过HTTP来呼叫你手机上的Servlet(如执行于i-Jetty内),然后该Servlet进而呼叫同一支手机内的Android应用程序或服务。 

 

    这i-Jetty本身是以Android应用程序形式嵌入(运行)于Android平台里,它可以透过Android框架的API与其它应用程序沟通。因而,在i-Jetty里执行的Servlet程序也能透过该Android API而与其它应用程序沟通、分享数据。例如,在家庭的Android TV里,加入一个i-Jetty插件,来容纳HTML网页和Servlet程序。数千公里外的家庭成员,透过手机Browser解析HTML,与家里的TV沟通,形成大小屏互动、多机整合的架构设计了。 

 

       I-Jetty是一支Android应用程序(*.apk),其扮演Container角色,让许多支Servlet程序可以在Android手机里执行。然而,这些Servlet程序经常需要与同一只手机里的其它Android应用程序沟通,例如启动晨钟或闹铃等等。此时,只要能取得Android框架的Context接口,就能呼叫Android的服务了。这个Context界面位居幕后,仅从应用程序代码的表面,常常看不到,然而它却是Android平台框架的最主要服务窗口。Context接口提供数十个大家很常用的服务,例如startService()、sendBroadcast()函数等。

    由于Activity也是继承Context抽象父类别,意味着Activity也是实作(Implements)这个Context界面。所以我们在Android应用程序里(如myActivity子类别)随时可呼叫Context接口里的函数,如下述的程序范例:

 

// Android程序范例

//.........

public class ac01 extends Activity implements OnClickListener {

       private Button btn, btn3;

       @Override protected void onCreate(Bundle icicle) {

             super.onCreate(icicle);

             //…….

             setContentView(layout);

        }

      public void onClick(View v) {

            if(v == btn)  

                   startService(new Intent(this, myService.class));

            else if(v == btn3)    

                   finish();

  }}

其中的指令: startService(new Intent(this, myService.class));

相当于指令: this.startService(new Intent(this, myService.class));

意味着,其呼叫到Context接口的startService()函数。其幕后机制是,myActivity类别呼叫到ContextWrapper类别的Context接口(的startService()函数),然后转而呼叫其mBase的(例如ApplicationContext的对象)Context接口的startService()函数。除了startService()函数之外,Context接口还含有许多其它函数,例如getApplicationContext()函数等。

 

手机内的I-Jetty与Android沟通

     同理,我们撰写i-Jetty里的myServlet程序时,也能透过Context接口来呼叫startService()函数来启动Android应用程序里的myService子类别,也能呼叫Context接口startActivity()函数来启动Android应用程序里的myActivity子类别,如下图:

 

  (Context接口扮演Android云端整合的重要接口)

      在myServlet子类别里可利用指令:

                  config.getServletContext().getAttribute("org.mortbay.ijetty.context");

 指令来取得其当下的Context接口,如下述i-Jetty的范例程序代码: 

 

/* --- i-Jetty的Servlet程序范例 ----*/

/**** 摘自i-Jetty套件的范例 ****/

import java.io.IOException;

import javax.servlet.ServletConfig;

import javax.servlet.ServletException;

import javax.servlet.ServletOutputStream;

import javax.servlet.http.HttpServlet;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

 

public class HelloWorld extends HttpServlet {

     String proofOfLife = null;

     public void init(ServletConfig config) throws ServletException {

             super.init(config);

             Object o = 

                  config.getServletContext().getAttribute("org.mortbay.ijetty.contentResolver");

             android.content.ContentResolver resolver 

                  = (android.content.ContentResolver)o;

             android.content.Context androidContext 

                 = (android.content.Context) config.getServletContext().getAttribute("org.mortbay.ijetty.context");

             proofOfLife = androidContext.getApplicationInfo().packageName;

  }

  public void doPost(HttpServletRequest request, HttpServletResponse response) 

                          throws ServletException, IOException

        {  doGet(request, response);   }

  public void doGet(HttpServletRequest request, HttpServletResponse response)

                         throws ServletException, IOException {

        response.setContentType("text/html");

        ServletOutputStream out = response.getOutputStream();

        out.println("<html>");

        out.println("<h1>Hello From Servlet Land!</h1>");

        out.println("Brought to you by: "+proofOfLife);

        out.println("</html>");

        out.flush();

    }

}

 

I-Jetty与Android的跨手机沟通

   上图里是从同一支手机(本机)里的myActivity来呼叫本机的Servlet接口,这是可行的,但是意义比较小。更大的意义是:由别的手机里的myActivity来传送HTTP呼叫到本机的Servlet接口,呼叫到本机的myServlet子类别;然后由myServlet呼叫Android的Context接口,而完成两支手机里两个myActivity子类别之间的相互沟通。如下图:

 

 (两支手机之通讯)

       于是,Context和Servlet两大接口,成为Android**云的喜鹊桥的两个主要桥墩。 

[Go Back]