一个JSF清单示例

这是使用JSF 2.0(JavaServer Faces)构建的示例列表应用程序。 该应用程序是待办事项列表。 该应用程序具有添加,编辑或删除列表中项目的功能。 待办事项具有名称和描述属性。

完成的应用程序的JSF页面具有:

  • 使用h:selectOneListbox html标记实现的待办事项列表。
  • 使用f:selectItems核心标签填充列表数据。
  • 待办事项名称和描述字段分别使用h:inputTexth:inputTextarea标记实现。
  • 新增,编辑,保存,删除和取消功能通过h:commandButton标记实现。
  • 使用h:outputText标记实现的状态消息。

应用中使用的类:

  • 待办事项:代表待办事项,并具有名称和描述属性。
  • TodosBean:这是一个托管bean; 它具有运行应用程序的代码,包括组件的侦听器和访问器方法。
  • TodoConverter:这是一个自定义转换器,将字符串todo名称转换为Todo对象,反之亦然。

下图显示了完成的应用程序的用户界面:

待办事项清单3

此示例应用程序分三个步骤进行说明。 第一步说明基本列表的实现。 该应用程序的功能在接下来的步骤中得到了增强。 这些步骤是:

  • 步骤1:待办事项列表显示项目,并在选择时显示选定的待办事项属性。
  • 第2步:列出待办事项,并具有将项目添加到列表的功能。
  • 第3步:列出待办事项和添加,编辑和删除列表项的功能。

步骤1:待办事项列表显示项目,并在选择时显示选定的待办事项属性。

待办事项清单1

以下是此应用程序的代码组件:

  • Todo.java类表示待办事项。
  • index.xhtml是带有列表框的JSF页面,以及一条状态消息,用于显示列表中的选定项目。
  • TodosBean.java托管Bean具有获取列表数据,运行列表的值更改侦听器以及显示状态消息的功能。

Todo.java:

此类表示待办事项。 它具有两个属性-名称和描述。 注意Object类的重写toString()方法。

Package com.javaquizplayer.example;
public class Todo {private String name;private String desc;public Todo() {}public Todo(String name, String desc) {this.name = name;this.desc = desc;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getDesc() {return desc;}public void setDesc(String desc) {this.desc = desc;}@Overridepublic String toString() {return this.name;}
}

index.xhtml:

此JSF页面显示待办事项列表。 列表可以滚动,可以选择一个项目。 所选项目名称显示在状态消息中。

列表框是使用h:selectOneListbox html标记实现的。 列表框的当前选定项目值由以下属性指定: value="#{bean.todo}" 。 选择项通过f:selectItems核心标签指定:

<f:selectItems value="#{bean.data}"/>

该标记位于h:selectOneListbox tag 。 列表框的项目是从TodosBeangetData()方法填充的,该方法返回List集合。 列表框显示标签–待办事项的名称值,即toString()方法中Todo对象的String值。

列表框的值更改侦听器使用以下属性指定:

valueChangeListener=""#{bean.valueChanged}"

选择列表项后,将提交表单并执行此侦听器代码。 在此示例中,当选择一个列表项时,待办事项的名称在状态消息中显示为“ todo_item_name selected”。

每次在列表框中选择一个项目时,都会提交该表单。 这是通过列表框的属性指定的: onchange="submit()"

状态消息与输出组件一起显示:

<h:outputText id="msg" value="#{bean.message}" />

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"xmlns:f="http://java.sun.com/jsf/core"xmlns:h="http://java.sun.com/jsf/html">
<h:head><title>A JSF List Example</title>
</h:head>
<h:body><h:form><h3>TODOs List</h3><h:selectOneListbox id="list" size="10"value="#{bean.todo}"valueChangeListener="#{bean.valueChanged}"onchange="submit()"><f:selectItems value="#{bean.data}"/>	</h:selectOneListbox><h:outputText id="msg" value="#{bean.message}" /></h:form>
</h:body>
</html>

TodosBean:

此托管bean类具有以下功能:

  • 创建初始待办事项数据并填充待办事项列表。
  • 值更改侦听器,以获取当前选定的列表项。
  • 在列表中捕获当前选定的项目值。
  • 设置状态消息。
package com.javaquizplayer.example;
import javax.faces.bean.SessionScoped;
import javax.faces.bean.ManagedBean;
import javax.faces.event.ValueChangeEvent;
import java.io.Serializable;
import java.util.List;
import java.util.ArrayList;
@ManagedBean(name="bean")
@SessionScoped
public class TodosBean implements Serializable {private List<Todo> data; // todo list dataprivate String todo; // the currently selected item valueprivate String msg; // status messagepublic TodosBean() {loadData();// select the first item in the listTodo t = data.get(0);setTodo(t.getName());setMessage(t.getName() + " selected.");}private void loadData() {data = new ArrayList<>();Todo t = new Todo("item 1", "item 1 description");data.add(t);t = new Todo("item 2", "item 2 description");data.add(t);t = new Todo("item 3", "item 3 description");data.add(t);t = new Todo("item 4", "item 4 description");data.add(t);}public List<Todo> getData() {return data;}public String getTodo() {return todo;}public void setTodo(String t) {todo = t;}// value change listener for list item selectionpublic void valueChanged(ValueChangeEvent e) {String t = (String) e.getNewValue();setMessage(t + " selected.");}public void setMessage(String s) {msg = s;}public String getMessage() {return msg;}
}

步骤2:列出待办事项,并具有将项目添加到清单的功能。

待办事项清单2

在此步骤中,应用程序具有添加新的待办事项的功能。 单击新建按钮,在名称和描述文本字段中输入待办事项数据并保存。 通过单击取消按钮或通过选择列表中的另一个项目来取消新的待办事项数据输入。

代码组成与上一步相同,但是增加了新功能。 新的自定义转换器类将添加到应用程序。

  • Todo.java类表示待办事项,未更改。
  • index.xhtml是带有列表框的JSF页面,以及显示列表中所选项目的状态消息。 此外,还有一些小部件可以输入新的待办事项并保存。
  • TodosBean.java托管Bean具有获取列表数据,运行列表值更改侦听器并显示消息的代码。 此外,还有用于新动作,保存动作和取消动作的动作侦听器。
  • 转换器TodoConverter.java将数据从todo字符串值转换为Todo对象,反之亦然。

Todo.java:

此类保持不变。

index.xhtml:

以下是更改:

列表框当前选择的项目值指定为: value="#{bean.todo}" 。 在上一步1中,项目值解析为todo的名称字符串。 现在,该值解析为Todo的一个实例。

以下是新添加的:

待办事项的名称和描述字段分别通过h:inputTexth:inputTextarea标记实现。 请注意,只有在编辑待办事项数据时(即,新的待办事项功能),这些字段才可编辑: readonly="#{not bean.editable}" 。 当列表处于选择模式时,这些字段是只读的。

转换器附加到列表中,以使用f:converter核心标记将选定的项目名称转换为Todo对象,反之亦然:

<f:converter converterId="todoConvertor"/>

请注意,为转换器设置了一个属性:

<f:attribute name="beanattr" value="#{bean}"/>

; 用于访问转换器类中的Todo数据。

使用h:commandButton标签为新的,保存和取消操作添加了三个命令按钮。 每个按钮都有其各自的动作侦听器。 例如:

<h:commandButton value="New" actionListener="#{bean.newListener}"/>

最后,使用f:ajax核心标记将列表框的Submit更改为Ajax调用: onchange="submit()"替换为

<f:ajax execute="@this" render="msg name desc" />

为什么会有这种变化?

使用提交选项时,如果通过选择另一个列表项取消了“新建”操作,则不会使用所选项目填充text和textarea字段。 值将保持编辑状态。 这是因为编辑后的文本值也随表格一起提交。 这些值将不会更改为所选项目的值。 但是,对于Ajax,不提交表单,仅更新文本值( f:ajax标记的render属性指定要更新的字段:状态消息,待办事项名称和描述)。

注意:在以下代码中,突出显示了上一步1中新添加和删除的代码行。

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"xmlns:f="http://java.sun.com/jsf/core"xmlns:h="http://java.sun.com/jsf/html">
<h:head><title>A JSF List Example</title>
</h:head>
<h:body><h:form><h3>TODOs List</h3><h:panelGrid><h:inputText id="name" value="#{bean.todoName}" size="30"readonly="#{not bean.editable}"/><h:inputTextarea id="desc" value="#{bean.todoDesc}"rows="2" cols="40"readonly="#{not bean.editable}"/><h:selectOneListbox id="list" size="10" value="#{bean.todo}"valueChangeListener="#{bean.valueChanged}"onchange="submit()"><f:ajax execute="@this" render="msg name desc"/><f:selectItems value="#{bean.data}"/><f:attribute name="beanattr" value="#{bean}"/><f:converter converterId="todoConvertor"/></h:selectOneListbox></h:panelGrid><h:commandButton value="New" actionListener="#{bean.newListener}"/><h:commandButton value="Save" actionListener="#{bean.saveListener}"/><h:commandButton value="Cancel" actionListener="#{bean.cancelListener}"/><h:outputText id="msg" value="#{bean.message}" /></h:form>
</h:body>
</html>

TodoConverter.java:

这是一个自定义转换器类,将Todo对象转换为字符串(todo名称),反之亦然。

package com.javaquizplayer.example;
import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import javax.faces.convert.Converter;
import javax.faces.convert.ConverterException;
import javax.faces.convert.FacesConverter;
import java.util.Map;
@FacesConverter(value="todoConvertor")
public class TodoConverter implements Converter {private static final long serialVersionUID = 1L;@Overridepublic Object getAsObject(FacesContext context, UIComponent component,String value)throws ConverterException {if (value == null) {return null;}Map<String, Object> attrs = component.getAttributes();		TodosBean bean = (TodosBean) attrs.get("beanattr");Todo todo = bean.getTodoForName(value);return todo;}@Overridepublic String getAsString(FacesContext context, UIComponent component,Object value)throws ConverterException {if (value == null) {return null;}Todo todo = (Todo) value;return todo.getName();}
}

TodosBean.java:

此托管bean类具有以下功能:

  • 填充待办事项列表。
  • 值更改侦听器,以获取当前选定的列表项。
  • 在列表中捕获当前选定的项目值。
  • 设置状态消息。

这些是更改:

  • 将列表中当前选择的项目值捕获为Todo对象(而不是先前使用的todo名称字符串)。
  • 更改值更改侦听器以获取Todo对象,而不是字符串值todo名称。

这些是新添加的:

  • 待办事项名称,描述及其可编辑性的访问器方法。
  • 动作侦听器,用于新动作,保存动作和取消动作。
package com.javaquizplayer.example;
import javax.faces.bean.SessionScoped;
import javax.faces.bean.ManagedBean;
import javax.faces.event.ValueChangeEvent;
import javax.faces.event.ActionEvent;
import java.io.Serializable;
import java.util.List;
import java.util.ArrayList;
@ManagedBean(name="bean")
@SessionScoped
public class TodosBean implements Serializable {private List<Todo> data;private Todo todo; // selected item valueprivate String msg;private String name; // text field valueprivate String desc;private String actionFlag = "NONE"; // specifies the current action (NEW, NONE)private boolean editable;public TodosBean() {loadData();if (data.size() == 0) {return;}Todo t = data.get(0);selectRow(t); // select the first item in the list}private void selectRow(Todo t) {setTodo(t);setTodoName(t.getName());setTodoDesc(t.getDesc());setMessage(t.getName() + " selected."););}private void loadData() {data = new ArrayList<>();}public List<Todo> getData() {return data;}public Todo getTodo() {return todo;}public void setTodo(Todo t) {todo = t;}public void valueChanged(ValueChangeEvent e) {if (! actionFlag.equals("NONE")) {setEditable(false);actionFlag = "NONE";}Todo t = (Todo) e.getNewValue();setMessage(t.getName() + " selected."););setTodoName(t.getName());setTodoDesc(t.getDesc());}public void setMessage(String msg) {this.msg = msg;}public String getMessage() {return msg;}public String getTodoName() {return name;}public void setTodoName(String n) {name = n;}public String getTodoDesc() {return desc;}public void setTodoDesc(String d) {desc = d;}// returns the Todo object for a given todo name// method used in converterpublic Todo getTodoForName(String name) {for (Todo t : data) {if (name.equals(t.getName())) {return t;}}return null;}public void setEditable(boolean b) {editable = b;}public boolean getEditable() {return editable;}public void newListener(ActionEvent e) {setEditable(true);setMessage("Enter new todo. Name must be uniquie and at least 5 chars.");setTodoName("NEW Todo");setTodoDesc("");actionFlag = "NEW";}public void saveListener(ActionEvent e) {if (! actionFlag.equals("NEW")) {return;}String name = getTodoName();String desc = getTodoDesc();if (name.length() < 5) {setMessage("Name must be at least 5 chars long.");return;}if (duplicateName(name)) {setMessage("Name must be unique.");return;}Todo t = new Todo(name, desc);data.add(t);setMessage(name + " saved.");setTodo(t); // select the saved itemsetEditable(false);actionFlag = "NONE";}private boolean duplicateName(String name) {for (Todo t : data) {if (t.getName().equals(name)) {return true;}}return false;}public void cancelListener(ActionEvent e) {if (actionFlag.equals("NONE")) {return;}actionFlag = "NONE";setMessage(actionFlag + " action cancelled");if (data.size() == 0) {setTodoName("");setTodoDesc("");setEditable(false);return;}// populate text fields with selected itemsetTodoName(todo.getName());setTodoDesc(todo.getDesc());setEditable(false);}
}

第3步:列出待办事项和添加,编辑和删除列表项的功能。

待办事项清单3

这是完成的应用程序。 它具有选择,添加,编辑和删除列表项的功能。

在此步骤中,添加了两个新功能–编辑,删除列表项。 要进行编辑,请选择一个待办事项列表项,然后单击“编辑”按钮。 这样可以更改名称和描述值,并保存。 可以通过单击取消按钮或选择另一个列表项来取消编辑。 删除功能允许删除选定的待办事项列表项。

这些代码组件与前面的步骤2相同,但是通过新功能进行了增强。

  • Todo.java类表示待办事项。 这没有改变。
  • 转换器TodoConverter.java将数据从todo字符串值转换为Todo对象,反之亦然。 这没有改变。
  • index.xhtml是具有列表框,待办事项属性和状态消息的JSF页面,该状态消息显示列表中的选定项目。 此外,还有一些小部件可用于添加,编辑所选项目或将其删除。
  • TodosBean.java托管bean具有用于获取列表数据,运行列表的值更改侦听器并显示消息的代码。 此外,还有用于新动作,编辑动作,删除动作,保存动作和取消动作的动作监听器。

Todo.java:

此类保持不变。

TodoConverter.java:

此类保持不变。

index.xhtml:

以下是新添加的:

添加了两个命令按钮以进行编辑和删除操作。 每个按钮都有其各自的动作侦听器。

TodoBean.java:

此托管bean类具有以下功能:

  • 填充待办事项列表。
  • 值更改侦听器,以获取当前选定的列表项。
  • 在列表中捕获当前选定的项目值。
  • 设置状态消息。
  • 待办事项名称,描述及其可编辑性的访问器方法。
  • 动作侦听器,用于新动作,保存动作和取消动作。 增强了这些功能以适应编辑和删除功能。

这些是新添加的:

  • 编辑和删除操作的操作侦听器。

代码下载:

这些是下载完整的应用程序的WAR文件和源代码的链接。

  • 源代码
  • WAR文件

注释和参考:

该应用程序是使用Apache MyFaces 2.0开发的(MyFaces 2.0实现了JavaServer Faces 2.0)。 该应用程序已在Tomcat 6 Web服务器和GlassFish 3应用程序服务器(GlassFish 3实现Java EE 6)上进行了测试。

有用的链接:

  • Apache MyFaces 2.0
  • GlassFish 3文档
  • Java EE 6 API

翻译自: https://www.javacodegeeks.com/2015/02/jsf-list-example.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/360680.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ubuntu下的jdk进行升级_如何在ubuntu上更新jdk版本

{"moduleinfo":{"card_count":[{"count_phone":1,"count":1}],"search_count":[{"count_phone":4,"count":4}]},"card":[{"des":"阿里云数据库专家保驾护航&#xff0c;为用户…

hdu 2444(二分图的判断以及求最大匹配)

题目链接&#xff1a;http://acm.hdu.edu.cn/showproblem.php?pid2444思路&#xff1a;首先要判断能否构成二分图&#xff0c;用bfs对当前点u染色&#xff0c;对u的邻接点v的颜色进行判断&#xff0c;如果为染色&#xff0c;则染色后入队列&#xff0c;否则&#xff0c;判断co…

Delta架构:统一Lambda架构并利用Hadoop / REST中的Storm

最近&#xff0c;一群人要求我详细介绍我为我们的书《分布式实时计算的风暴蓝图》撰写的Druid / Storm集成。 德鲁伊很棒。 风暴很大。 两者一起解决了实时维查询/聚合问题。 实际上&#xff0c;人们正在将其视为主流&#xff0c;将其称为RAD Stack &#xff0c;并添加了“ La…

activiti mysql 版本_Mysql8.0.17版本不能自动创建activiti表的坑

maven项目如下&#xff1a;配置好数据库&#xff0c;和activiti的配置之后&#xff0c;开始执行流程部署package com.yuanqiao.first_activiti.deployment;import java.util.List;import org.activiti.engine.ProcessEngine;import org.activiti.engine.ProcessEngines;import …

Linux Kernel 多个本地信息泄露漏洞

漏洞名称&#xff1a;Linux Kernel 多个本地信息泄露漏洞CNNVD编号&#xff1a;CNNVD-201306-028发布时间&#xff1a;2013-06-04更新时间&#xff1a;2013-06-04危害等级&#xff1a; 漏洞类型&#xff1a; 威胁类型&#xff1a;本地CVE编号&#xff1a; 漏洞来源&#xff1a…

JDBC教程– ULTIMATE指南(PDF下载)

在本文中&#xff0c;我们提供了全面的JDBC教程&#xff08;Java数据库连接性&#xff09;&#xff0c;这是Oracle提供的一种API&#xff0c;允许程序员处理Java应用程序中的不同数据库&#xff1a;它允许开发人员建立与数据库的连接&#xff0c;定义特定的客户端如何访问给定的…

Spinner 学习笔记

Adapter 适配器//方法一&#xff1a;直接使用字符串数组//String[] sSexList new String[]{"男", "女"};//方法二&#xff1a;使用资源文件String[] sSexList getResources().getStringArray(R.array.Sexarray);//实例化一个集合适配器ArrayAdapter<S…

Java 8发布一年后,IDE和编译器尚未完全就绪

一年前&#xff0c;即2014年3月18日 &#xff0c;发布了Java SE 8&#xff0c;并伴随着通过lambda表达式和streams API实现了功能性编程的幸福。 这对于我们所有的Java生态系统都是个好消息&#xff0c;许多人已经升级到Java8。Stack Overflow已经提出了大约2500个有关Java 8的…

mysql增量_mysql实时增量备份

采用binlog日志的好处掌控所有更改操作,必要时可用于恢复数据数据库主从复制的必要条件[linyouyilocalhost~]# vim /etc/my.cnf[mysqld].. ..log-binmysql-bin //启用二进制日志&#xff0c;并指定前缀.. ..[linyouyidbsvr1 ~]# service mysqld restart确认binlog日志文件新启用…

使用JDK 8流在包装对象的集合和包装对象的集合之间转换

我发现使用基于Java的应用程序时&#xff0c; 装饰器和适配器有时会很有用。 这些“包装器”在各种情况下都能很好地工作&#xff0c;并且相当容易理解和实现&#xff0c;但是当需要包装对象的层次结构而不是单个对象时&#xff0c;事情可能会变得有些棘手。 在这篇博客文章中&…

mysql注入提取邮件_【sql注入教程】mysql注入直接getshell

Mysql注入直接getshell的条件相对来说比较苛刻点把1:要知道网站绝对路径&#xff0c;可以通过报错&#xff0c;phpinfo界面&#xff0c;404界面等一些方式知道2:gpc没有开启&#xff0c;开启了单引号被转义了&#xff0c;语句就不能正常执行了3:要有file权限&#xff0c;默认情…

使用WildFly 8.2.0.Final,Primefaces 5.1和MySQL 5的JDBC领域和基于表单的身份验证

我会不时查看我博客上最受欢迎的内容&#xff0c;并尽力满足您的最佳需求。 因此&#xff0c;阅读我的博客是其他读者推动内容的一种方式。 另一种方法是通过评论或电子邮件与我联系。 今天&#xff0c;我将使用Primefaces修改我的JDBC Realm示例并将其更新到最新的WildFly服务…

【转】代码里的命名规则:错误的和正确的对比

原文出处&#xff1a; goyello 译文出处&#xff1a; 外刊IT评论 编程初学者总是把大量的时间用在学习编程语言&#xff0c;语法&#xff0c;技巧和编程工具的使用上。他们认为&#xff0c;如果掌握了这些技术技巧&#xff0c;他们就能成为不错的程序员。然而&#xff0c;计算…

mysql怎么给表设置查询语句_MySQL查询语句简单操作示例

本文实例讲述了MySQL查询语句简单操作。分享给大家供大家参考&#xff0c;具体如下&#xff1a;查询创建数据库、数据表-- 创建数据库create database python_test_1 charsetutf8;-- 使用数据库use python_test_1;-- students表create table students(id int unsigned primary …

您的Apache Camel应用程序现在包括现成的文档

几个月前&#xff0c;我在博客中发布了有关即将发布的2.15版本的信息&#xff0c;该功能包括获取有关在端点上配置的每个属性的详细信息的功能-Apache Camel&#xff0c;请向我解释这些端点选项的含义 。 我们沿着这条道路继续前进&#xff0c;今天&#xff0c;我们将其从端点…

iOS全局变量与属性的内存管理

在iOS开发中&#xff0c;为了节约时间&#xff0c;程序员经常会用全局变量代替属性。但是这样做&#xff0c;尤其是新手开发中&#xff0c;经常会引起内存泄露的报错&#xff0c;其实作为苹果自己也没有给出一个完美安全的内存管理代码例子。但是在iOS开发到如今&#xff0c;有…

mysql xml格式化输出_在MySQL中使用XML数据—数据格式化

在MySQL中使用XML数据可以将数据进行格式化以更好的管理数据。有IgorRomanenko编写的MySqlDump客户端最初是一个备份程序&#xff0c;它可以备份数据库&#xff0c;或将一个数据库转移到另一个数据库(MySQL或其它数据库)&#xff0c;它的这个备份过程其实是一个创建表&#xff…

休眠锁定模式– PESSIMISTIC_READ和PESSIMISTIC_WRITE如何工作

介绍 Java Persistence API带有完善的并发控制机制&#xff0c;支持隐式和显式锁定。 隐式锁定机制很简单&#xff0c;它依赖于&#xff1a; 乐观锁定&#xff1a;实体状态更改可以触发版本增加 行级锁定&#xff1a;基于当前运行的事务隔离级别 &#xff0c;INSERT / UPDATE…

廖雪峰mysql安装教程 pdf_使用MySQL - 廖雪峰 Python 3 教程

MySQL是Web世界中使用最广泛的数据库服务器。SQLite的特点是轻量级、可嵌入&#xff0c;但不能承受高并发访问&#xff0c;适合桌面和移动应用。而MySQL是为服务器端设计的数据库&#xff0c;能承受高并发访问&#xff0c;同时占用的内存也远远大于SQLite。此外&#xff0c;MyS…

将WildFly绑定到其他IP地址或多宿主上的所有地址

用WildFly的话来说&#xff0c; 接口是一个逻辑名称&#xff0c;用于套接字可以绑定到的网络接口/ IP地址/主机名。 有两个接口&#xff1a;“公共”和“管理”。 “公共”接口绑定用于所有与应用程序相关的网络通信&#xff08;例如&#xff0c;Web&#xff0c;Messaging等&a…