J2ME 设计收集

 MIDP设计模式之集结贴[JavaME] <script language="javascript" type="text/javascript">document.title="MIDP设计模式之集结贴[JavaME] - "+document.title</script>

1:

架构性宣言: MIDP 2.0 中的工厂设计模式

如何使用 MIDP Generic Connection Framework

http://www-128.ibm.com/developerworks/cn/java/wi-arch22/

级别: 初级

Mikko Kontio, 产品经理, Softera

2006 年 3 月 13 日

这个月将学习有关 MIDP 2.0 的更多知识,同 Mikko 一起观察 Mobile Information Device Profile (MIDP) 的通用连接器框架 —— 工厂设计模式。

手机开发人员通常使用 Generic Connection Framework 在 Mobile Information Device Profile (MIDP) 中创建和维护顺利的连接。好的架构师都知道该框架背后真正的动力是不可缺少的工厂设计模式。工厂设计模式是面向对象编程所必需的,它构成了应用程序开发人员使用的大多数强大框架的基础 —— 包括 MIDP。在本月的架构性声明 专栏中,我将介绍工厂模式的三种变体,然后展示如何在 MIDP 2.0 通用连接框架中工厂支持连接处理。

工厂模式

工厂设计模式是面向对象编程中最常用的设计模式之一。它又被称为创建性模式,因为它被用来创建其他类。在应用程序预见不到自己要创建的对象类型时,就会使用工厂解决方案。在这些情况下,可以使用工厂模式作为创建对象的基础,不需要确切地了解将要创建哪些对象。

工厂 实际上是一组模式的名称,这组模式的目的是创建类。每个变体都指定一个不同的创建方法。这些模式变体是:

  • 静态工厂(Static Factory)
  • 工厂方法(Factory Method)
  • 抽象工厂(Abstract Factory)

首先,我先一般性地查看一下工厂模式解决常见应用程序设计问题的方式。然后将演示工厂模式如何开始对 MIDP 2.0 中大量连接类型的连接进行处理。

 

其余部分请到IBM站点上看吧。

 

2:

《MVC设计模式在客户端的应用》英文文档下载地址:
http://www.j2medev.com/Soft/ShowSoftDown.asp?UrlID=1&SoftID=346

3:

《应用MVC模式解决J2ME导航问题》
http://www.j2medev.com/Article/ShowArticle.asp?ArticleID=28

摘要:开发MIDlet的程序员也许经常会被界面的导航问题所困扰,尤其界面比较多的时候,通常有七、八个界面就会很让人头疼了。本文讲述如何应用MVC设计模式解决这类的问题。

    MVC 设计模式已经非常的成熟并在WEB Application的开发中广泛使用,apache的开源项目struts就是典型的例子。MVC的本质就是是逻辑和显示分开,通过控制器进行协调。通常我们会感到控制器比较的肥大,这个是个有争议的问题。MIDP的用户界面开发是比较简单的,只有那么20几个类。但是由于导航一般只能通过 Command来实现,所以界面增多的情况下,如果没有有效的组织那么程序写起来非常的乱,最致命的是这样的程序可读性差、扩展性差、可维护性差。

    应用MVC解决这个问题的关键是提供一个桥梁作用的控制器,它通常要有一个MIDlet作为参数。

 

4:

层叠式菜单模式的流程流转示例:


Figure 1. A city guide application using Cascading Menu

code:

  public void commandAction(Command command, Displayable displayable)
  {
    if ( command.equals( List.SELECT_COMMAND ) && displayable instanceof MenuList )
    {
      MenuList list = (MenuList) displayable;
      int i = list.getSelectedIndex();
      MenuElement item = (MenuElement) children.elementAt( i );
      Object next = menuMap.get( item );

      if ( next instanceof MenuElement )
      {
        list.showMenu( (MenuElement) next );

      } else if ( next instanceof MDisplayable && next instanceof Displayable )
      {
        ((MDisplayable)next).onDisplay( item );
        list.getDisplay().setCurrent( (Displayable) next );

      } else if ( next instanceof Displayable )
      {
        list.getDisplay().setCurrent( (Displayable) next );
      }
    }
  }

更多代码参见

http://www.cnblogs.com/Files/zhengyun_ustc/PatternShow.rar

中的PatternShow/src/pattern/menu。

 

5:

向导式对话框模式 幻灯片模式流程流转参见:

流程流转图样看:


Figure 3. A sample wizard application

/**
   * this commandAction is only invoked by WDialog
   * This function control the flow of dialogs when user press Next or Back
   * button. This function also invoke onEnter() and onLeave() on WDialog. If a dialog
   * return WDialog.REJECT on either onEnter() or onLeave(), WizardEngine will not
   * switch screen.
   * @param command
   * @param displayable
   */
  public void commandAction(Command command, Displayable displayable)
  {
    WDialog cur_dialog = (WDialog)displayable;

    if ( command == WDialog.NEXT_COMMAND )
    {
      int i1 = dialogs.indexOf( displayable );
      if ( i1 < dialogs.size() - 1 )
      {
        WDialog next_dialog = (WDialog)dialogs.elementAt( i1 + 1 );
        if ( cur_dialog.onLeave( WDialog.FORWARD ) != WDialog.OK )
           return;
        if ( next_dialog.onEnter( WDialog.FORWARD ) != WDialog.OK )
           return;
        display.setCurrent( next_dialog );
      }

    } else if ( command == WDialog.BACK_COMMAND )
    {
      int i1 = dialogs.indexOf( displayable );
      if ( i1 > 0 )
      {
        WDialog prev_dialog = (WDialog)dialogs.elementAt( i1 - 1 );
        if ( cur_dialog.onLeave( WDialog.BACKWARD ) != WDialog.OK )
           return;
        if ( prev_dialog.onEnter( WDialog.BACKWARD ) != WDialog.OK )
           return;

        display.setCurrent( prev_dialog );
      }
    }
  }

更多代码参见

http://www.cnblogs.com/Files/zhengyun_ustc/PatternShow.rar

中的PatternShow/src/pattern/wizard。

 

6:

 


Figure 6. Example slide show sequence

Listing 6. Slide show implementation

    // Create a slide show engine with Display object as parameter.
    // 'this' pointer refers to current MIDlet.
    engine = new SlideEngine( Display.getDisplay(this ) );

    // Create 3 slides. All the slides are formed here.
    // You can use something else.
    Form f1 = new Form( "Slide 1");
    f1.append( "This is slide number 1" );

    Form f2 = new Form( "Slide 2");
    f2.append( "This is slide number 2" );

    Form f3 = new Form( "Slide 3");
    f3.append( "This is slide number 3" );

    // Add the slides to engine, each slide stays on screen
    // for 2 seconds.
    engine.addSlide( f1, 2000 );
    engine.addSlide( f2, 2000 );
    engine.addSlide( f3, 2000 );

    // Start the slide show
    engine.startShow();

更多代码参见

http://www.cnblogs.com/Files/zhengyun_ustc/PatternShow.rar

中的PatternShow/src/pattern/slide。

 

7:

分页List模式

流程流转:

其实就是一个List,但他有很多很多项,以至于需要用户手动翻页查看。

用户选择More菜单就是往后翻,选择Prev就是向前翻。

Listing 5. PagableList sample usage

   PagableList pagablelist = new PagableList( "Paging", List.IMPLICIT);
  
   // 'this' refer to the current MIDlet
   Display.getDisplay(this).setCurrent(pagablelist);
  
   for ( int i=0; i< 100; i++ )
   {
      pagablelist.append( "Item  #"+i, null );
   }

更多代码参见

http://www.cnblogs.com/Files/zhengyun_ustc/PatternShow.rar

中的PatternShow/src/pattern/pagination。

后面四个模式来自于《Big designs for small devices
Four J2ME design patterns simplify interactive content development
http://www.javaworld.com/javaworld/jw-12-2002/jw-1213-j2medesign.html》。



Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=644684

 

架构性宣言: MIDP 2.0 中的工厂设计模式

如何使用 MIDP 的 Generic Connection Framework

developerWorks
文档选项
将此页作为电子邮件发送

将此页作为电子邮件发送


最新推荐

Java 应用开发源动力 - 下载免费软件,快速启动开发


级别: 初级

Mikko Kontio, 产品经理, Softera

2006 年 3 月 13 日

这个月将学习有关 MIDP 2.0 的更多知识,同 Mikko 一起观察 Mobile Information Device Profile (MIDP) 的通用连接器框架 —— 工厂设计模式。

手机开发人员通常使用 Generic Connection Framework 在 Mobile Information Device Profile (MIDP) 中创建和维护顺利的连接。好的架构师都知道该框架背后真正的动力是不可缺少的工厂设计模式。工厂设计模式是面向对象编程所必需的,它构成了应用程序开发人员使用的大多数强大框架的基础 —— 包括 MIDP。在本月的架构性声明 专栏中,我将介绍工厂模式的三种变体,然后展示如何在 MIDP 2.0 通用连接框架中工厂支持连接处理。

工厂模式

工厂设计模式是面向对象编程中最常用的设计模式之一。它又被称为创建性模式,因为它被用来创建其他类。在应用程序预见不到自己要创建的对象类型时,就会使用工厂解决方案。在这些情况下,可以使用工厂模式作为创建对象的基础,不需要确切地了解将要创建哪些对象。

工厂 实际上是一组模式的名称,这组模式的目的是创建类。每个变体都指定一个不同的创建方法。这些模式变体是:

  • 静态工厂(Static Factory)
  • 工厂方法(Factory Method)
  • 抽象工厂(Abstract Factory)

首先,我先一般性地查看一下工厂模式解决常见应用程序设计问题的方式。然后将演示工厂模式如何开始对 MIDP 2.0 中大量连接类型的连接进行处理。





回页首


基于工厂的解决方案

不管使用哪种变体,所有的工厂模式都是通过将应用程序从将要初始化的类中隔离出来进行启动,实现方法是插入一个 factory 类来做实际的创建工作。图 1 是工厂模式的结构图。可以看到,Client 是需要创建新实例的应用程序或类,Product 是需要创建的类;Factory 是实际创建产品的类。


图 1. 工厂设计模式的结构
工厂设计模式的结构

图 2 是工厂模式使用方式的概述,其中的工厂可以创建两种产品。


图 2. 静态工厂模式的实现
静态工厂模式的实现

客户机使用工厂类中的不同方法来创建 ProductA 和 ProductB 的实例。该模式的优点在于,只要工厂接口可用,客户机就不需要考虑如何创建对象。例如,客户机只需要调用以下代码,就可以创建 ProductA 类的实例。


ProductA p = Factory.createA();

更高级的解决方案

虽然以上设置非常有用,但它确实存在一些限制。首先,客户机需要知道它要初始化哪些类。例如,如果需要添加第三个类 ProductC (或类似的情况),那么既需要修改客户机,也需要修改工厂类。工厂类还需要一些新方法,客户机也必须能够处理 ProductC。虽然修改一个 Factory 类很简单,但可能有许多客户机需要考虑。修改所有客户机并不是好的解决方案,而且有可能无法实现。

工厂模式的更高级实现要向产品类添加一个抽象层。然后,客户机可以只引用抽象产品,而不必引用 ProductA 或 ProductC。我在 图 3 中实现了该操作,从图中可以看出,除了两个具体产品(现在是 ConcreteProductA 和 ConcreteProductB)之外,现在有了一个抽象的 Product


图 3. 带有抽象的工厂设计模式
带有抽象的工厂设计模式

图 3 中更高级的实现让客户机使用抽象产品进行对象处理,让 Factory 类创建具体的产品类实例。图 3 中还需要注意的是,工厂现在是抽象的。这样,在需要新工厂时(例如,需要创建新类时),就可以容易地构建新工厂。

工厂模式简化了类的创建,而且为在现有应用程序中做修改提供了良好基础。但是,不要太过兴奋,碰到能用的地方就到处使用该模式。在某些情况下,工厂模式会给应用程序带来超乎需要的复杂性。作为架构师,您必须问自己:更简单、通用性差些的解决方案是否更好。工厂模式最适用的情况是:有许多客户机要使用应用程序的类,或者进行修改很困难或不可能时。如果要避免对客户机进行修改,则工厂模式会是一个便捷的解决方案。





回页首


MIDP 2.0 中的工厂模式

MIDP 2.0 通用连接框架用工厂设计模式的高级实现作为基础。顾名思义,通用连接框架的设计目的是处理所有连接种类。在 MIDP 1.0 中,该框架受到限制,只能处理 HTTP 连接,但是 MIDP 2.0 增加了 HTTPS、串口连接、套接字和更多连接。在这一节,我将把重点放在如何在 MIDP 2.0 通用连接框架中使用工厂设计模式。

通用连接框架

图 4 中,可以看到支持 HTTP 连接处理的那部分通用连接框架。


图 4. 部分通用连接框架
部分通用连接框架

请注意,因为通用连接框架实现了工厂模式,所以使用 Connection 接口可以处理所有连接。抽象是通过对接口进行扩展实现的。Connector 这个工厂类有一组用来创建连接的 open() 方法。这些 open() 方法接受字符串作为参数。这些参数的表达形式是:


{scheme}:[{target}][{params}]

{scheme} 是协议的名称,例如 HTTP,{target} 通常是某种形式的网络地址,{params} 由一系列 “;param=value” 形式的等式构成。





回页首


MIDP 中的连接

在 MIDP 中创建连接非常简单。要做的全部工作就是用正确的字符串调用 open() 方法,并拥有连接。例如,只用一行代码就可以打开 HTTP 连接:


Connection con = Connector.open("http://www.ibm.com");

当字符串参数是正常的 HTTP 地址时,Connector 就打开 HTTP 连接,可以把它当作 Connection 来处理,也可以像下面这样把它转变成 HttpConnection


HttpConnection httpcon = (HttpConnection)con;

获取文件轻而易举!

建立了 HTTP 连接之后,就可以使用它。在 清单 1 中,我用通用连接框架从 Web 服务器上下载了一个 XML 文件。


清单 1. 用 GCF 获取 XML 文件

HttpConnection con = null;
InputStream is	= null;
String xml = new String();
OutputStream out = null;
try {
	con = (HttpConnection)Connector.open(this.url);
	con.setRequestMethod(HttpConnection.GET);
	con.setRequestProperty( "Connection", "close" );
	
	// The call to openInputStream() opens the connection
	is	= con.openInputStream();

	// Read the XML file
	ByteArrayOutputStream bas = new ByteArrayOutputStream();
	int ch;
	while ((ch = is.read()) != -1) {
		bas.write(ch);
	}

	// The 
	xml = bas.toString();	

} catch (Exception e) {
	e.printStackTrace();
} finally {
	try {
		if (null!=out) out.close();
		if (null!= is) is.close();
		con.close();
	} catch (Exception ex) {
		ex.printStackTrace();
	}
}  


首先,我用特定 Web 地址调用 Connectoropen() 方法,然后设置请求方法和 Connection 属性。为了实际打开连接,我调用了 openInputStream()。打开了到内容(在这个示例中是 XML 文件)的流之后,我用 ByteArrayOutputStream 来读取流。





回页首


结束语

虽然多数手机应用程序开发人员都很清楚地知道如何使用 MIDP 2.0 中的通用连接框架,但是很少有人知道或关心实际推动该框架的丰富功能的模式。对于架构师来说,观察角度是相反的:MIDP 2.0 代表工厂设计模式的优秀(和高级)实现!

在本月的专栏中,通过展示如何从架构师的观点来查 MIDP 2.0,我将帮助您更好地理解它。关于工厂模式本身,您已经了解了一些知识:即它的工作方式和它最适合解决哪类问题。请继续关注下个月的专栏,我们将介绍在为下一个手机开发项目选择技术时可以使用的实践技巧。





回页首


参考资料

学习

讨论




回页首


关于作者

Mikko Kontio 是领先的芬兰软件公司 Softera 的产品经理。他拥有计算机科学的硕士学位,是多部书籍的作者和合作者,最新力作是 Professional Mobile Java with J2ME,由 IT Press 出版。您可以通过 Mikko 的邮件 mikko.kontio@softera.fi 与他联系。

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值