Struts2学习总结(一)

一、Struts2的执行流程

当用户访问某一个Action的时候,先经过核心过滤器,在核心过滤器中执行一组拦截器(这组拦截器实现部分功能),执行目标Action,根据Action的返回值,进行页面跳转。

二、Action的编写方式

1:Action是一个POJO的类

public class ActionDemo1 {
	public String execute(){
		System.out.println("ActionDemo1执行了...........");
		return null;
	}
}

2:实现一个Action的接口

public class ActionDemo2 implements Action {
	@Override
	public String execute() throws Exception {
		System.out.println("ActionDemo2执行了...........");
		return null;
	}
}

3:Action的编写方式三:Action类继承ActionSupport类

(推荐使用继承ActionSupport方式,ActionSupport中提供了数据校验、国际化等一系列操作的方法。)

public class ActionDemo3 extends ActionSupport{
	@Override
	public String execute() throws Exception{
		System.out.println("ActionDemo3执行了......");
		return NONE;	
	}
}

XML配置方式

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
	"-//Apache Software Foundation//DTD Struts Configuration 2.3//EN"
	"http://struts.apache.org/dtds/struts-2.3.dtd">
<struts>
	<!-- Struts2为了管理Action的配置,通过包进行管理。 -->
	<!-- 配置Struts2的包 ================ -->
	<package name="demo" extends="struts-default" namespace="/">
		<!-- 配置Action================ -->
		<action name="actionDemo1" class="demo/ActionDemo1"></action>
		<action name="actionDemo2" class="demo/ActionDemo2"></action>
		<action name="actionDemo3" class="demo/ActionDemo3"></action>
	</package>
</struts>

三、Action的访问方式

1、通过method方式访问Action

public class UserAction extends ActionSupport {
	public String Find(){
		System.out.println("查询用户....................");
		return NONE;
	}
	public String Update(){
		System.out.println("更新用户....................");
		return NONE;
	}
	public String Delete(){
		System.out.println("删除用户....................");
		return NONE;
	}
	public String Save(){
		System.out.println("保存用户....................");
		return NONE;
	}
}

<struts>
<package name="demo1" extends="struts-default" namespace="/">
		<!--通过method方式访问Action  -->
		<action name="userFind" class="demo1.UserAction" method="Find"></action>
		<action name="userUpdate" class="demo1.UserAction" method="Update"></action>
		<action name="userDelete" class="demo1.UserAction" method="Delete"></action>
		<action name="userSave" class="demo1.UserAction" method="Save"></action>
</package>
</struts>

**在JSP中的编写**
	<h3>通过method方式</h3>
	<a href="${ pageContext.request.contextPath }/userFind.action">查询用户</a>
	<br />
	<a href="${ pageContext.request.contextPath }/userUpdate.action">更新用户</a>
	<br />
	<a href="${ pageContext.request.contextPath }/userDelete.action">删除用户</a>
	<br />
	<a href="${ pageContext.request.contextPath }/userSave.action">保存用户</a>
	<br />

2、通过通配符的方法访问Action

public class ProductAction extends ActionSupport {
	public String find() {
		System.out.println("查询商品.................");
		return NONE;
	}
	public String update() {
		System.out.println("更新商品.................");
		return NONE;
	}
	public String delete() {
		System.out.println("删除商品.................");
		return NONE;
	}
	public String save() {
		System.out.println("保存商品.................");
		return NONE;
	}
}

<struts>
<package name="demo1" extends="struts-default" namespace="/">
<!--通过通配符的方式访问Action  -->
		<action name="product_*"  class="demo1.ProductAction" method="{1}"></action>
</package>
</struts>

**在JSP中的编写**
	<h3>通过通配符的方式</h3>
	<a href="${ pageContext.request.contextPath }/product_find.action">查询商品</a>
	<br />
	<a href="${ pageContext.request.contextPath }/product_update.action">更新商品</a>
	<br />
	<a href="${ pageContext.request.contextPath }/product_delete.action">删除商品</a>
	<br />
	<a href="${ pageContext.request.contextPath }/product_save.action">保存商品</a>
	<br />

3、通过动态方法访问的方式

public class CustomerAction extends ActionSupport {
	public String find() {
		System.out.println("查询客户................");
		return NONE;
	}
	public String update() {
		System.out.println("更新客户................");
		return NONE;
	}
	public String delete() {
		System.out.println("更新客户................");
		return NONE;
	}
	public String save() {
		System.out.println("保存客户................");
		return NONE;
	}
}

<struts>
	<!--配置动态方法的常量:开启动态方法访问  -->
	<constant name="struts.enable.DynamicMethodInvocation" value="true"></constant>
		<!--通过动态方法的方式访问Action  -->
	<action name="customer" class="demo1.CustomerAction"></action>
</package>
</struts>

**在JSP中的编写**
	<h3>通过动态方法访问的方式</h3>
	<a href="${ pageContext.request.contextPath }/customer!find.action">查询客户</a>
	<br />
	<a href="${ pageContext.request.contextPath }/customer!update.action">更新客户</a>
	<br />
	<a href="${ pageContext.request.contextPath }/customer!delete.action">删除客户</a>
	<br />
	<a href="${ pageContext.request.contextPath }/customer!save.action">保存客户</a>
	<br />

总结:第二种方式最常用,第三种方式记得开启动态方法访问,配置其常量。

四、Struts2对Servlet的API的访问

1、完全解耦合的方式

public class RequestDemo1 extends ActionSupport {
	@Override
	public String execute() throws Exception {
		// 一、接收参数:
		// 利用Struts2中的对象ActionContext对象
		ActionContext context = ActionContext.getContext();
		// 调用ActionContext中的方法
		// 类似于Map<String,String[]>request.getParameterMap();
		Map<String,Object> map=context.getParameters();
		for(String key:map.keySet()){
			String[] values=(String[]) map.get(key);
			System.out.println(key+" :	 "+Arrays.toString(values));
		}
		//二、向域对象中存入数据
		context.put("reqName", "reqValue");//相当于request.setAttribute();
		context.getSession().put("sessName", "sessValue");//相当于session.setAttribute();
		context.getApplication().put("appName", "appValue");//相当于application.setAttribute();
		return SUCCESS;
	}
}

<struts>
	<package name="demo1" extends="struts-default" namespace="/">
		<action name="requestDemo1" class="demo1.RequestDemo1">
			<!--局部结果页面 -->
			<result name="success">/demo1/demo2.jsp</result>
		</action>
	</package>
</struts>

**在JSP中的编写**
demo1.jsp
<h3>方式一:完全解耦合的方式</h3>
<form action="${ pageContext.request.contextPath }/requestDemo1.action" method="post">
	姓名:<input type="text" name="name"/><br/>
	密码:<input type="password" name="password"><br/>
	<input type="submit" value="提交">
</form>

demo2.jsp
<h1>显示数据</h1>
${reqName }
${sessName }
${appName }

2、使用Servlet的API的原生方式

public class RequestDemo2 extends ActionSupport{
	@Override
	public String execute()throws Exception{
		//一、接收数据
		//直接获得request对象,通过ServletActionContext
		HttpServletRequest request=ServletActionContext.getRequest();
		Map<String,String[]> map=request.getParameterMap();
		for(String key:map.keySet()){
			String[] values=map.get(key);
			System.out.println(key+":"+Arrays.toString(values));
		}
		//二、向域对象中保存数据
		//向request中保存数据
		request.setAttribute("reqName", "reqValue");
		//向session中保存数据
		request.getSession().setAttribute("sessName", "sessValue");
		//向application中保存数据
		ServletActionContext.getServletContext().setAttribute("appName", "appValue");
		return SUCCESS;
	}
}

<struts>
	<package name="demo1" extends="struts-default" namespace="/">
		<action name="requestDemo3" class="demo1.RequestDemo3"></action>
	</package>
</struts>

**在JSP中的编写**
demo1.jsp
<h3>方式二:使用原生的方式访问</h3>
<form action="${ pageContext.request.contextPath }/requestDemo2.action" method="post">
	姓名:<input type="text" name="name"/><br/>
	密码:<input type="password" name="password"><br/>
	<input type="submit" value="提交">
</form>

demo2.jsp
<h1>显示数据</h1>
${reqName }
${sessName }
${appName }

3、接口注入的方式

public class RequestDemo3 extends ActionSupport implements ServletRequestAware,ServletContextAware {
	private HttpServletRequest request;
	private ServletContext context;
	@Override
	public String execute() throws Exception {
		// 一、接收参数
		// 通过接口注入的方式获得request对象
		Map<String, String[]> map = request.getParameterMap();
		for (String key : map.keySet()) {
			String[] values = map.get(key);
			System.out.println(key + " 	   " + Arrays.toString(values));
		}	
		// 二、向域对象中保存数据
		// 向request中保存数据
		request.setAttribute("reqName", "reqValue");
		// 向session中保存数据
		request.getSession().setAttribute("sessName", "sessValue");
		// 向application中保存数据
		context.setAttribute("appName", "appValue");
		return SUCCESS;
	}
	@Override
	public void setServletRequest(HttpServletRequest request) {
		this.request = request;
	}
	@Override
	public void setServletContext(ServletContext context) {
		// TODO Auto-generated method stub
		this.context=context;
	}
}

<struts>
	<package name="demo1" extends="struts-default" namespace="/">
		<!--全局结果页面 -->
		<global-results>
			<result name="success">/demo1/demo2.jsp</result>
		</global-results>
		<action name="requestDemo3" class="demo1.RequestDemo3"></action>
	</package>
</struts>

**在JSP中的编写**
demo1.jsp
<h3>方式三:接口注入的方式</h3>
<form action="${ pageContext.request.contextPath }/requestDemo3.action" method="post">
	姓名:<input type="text" name="name"/><br/>
	密码:<input type="password" name="password"><br/>
	<input type="submit" value="提交">
</form>

demo2.jsp
<h1>显示数据</h1>
${reqName }
${sessName }
${appName }

总结:第一种方式只能获得代表request、session、application的数据的Map集合,不能操作这些对象的本身的方法;第二种方式可以操作域对象的数据,同时也可以获得对象的方法;第三种方式中的Servlet是单例的,多个程序访问同一个Servlet只会创建一个Servlet的实例;Action是多例的,一次请求创建一个Action的实例(不会出现线程安全的问题)。

五、结果页面的配置

1、全局结果页面:全局结果页面指的是,在包中配置一次,其他的在这个包中的所有的action只要返回了这个值,都可以跳转到这个页面。
针对这个包下的所有的action的配置都有效。
2、局部结果页面:局部结果页面指的是,只能在当前的action中的配置有效。
针对当前的action的配置有效。

六、result标签的配置

result标签用于配置页面的跳转,在result标签上有两个属性:

name属性:逻辑视图的名称。默认值:success

type属性:页面跳转的类型。

					dispatcher:默认值,请求转发(Action转发Jsp)
					redirect:重定向(Action重定向Jsp)
					chain:转发(Action转发Action)
					redirectAction:重定向(Action重定向Action)
					stream:Struts2中提供文件下载的功能

注:dispatcher和redirect比较重要。

七、Struts2的数据封装

1、属性驱动,数据封装方式一:提供属性的set方法的方式(不常用)

public class UserAction1 extends ActionSupport{	
	private String username;
	private String password;
	private Integer age;
	private Date birthday;
	private Double salary;

	public void setUsername(String username) {
		this.username = username;
	}

	public void setPassword(String password) {
		this.password = password;
	}

	public void setAge(Integer age) {
		this.age = age;
	}

	public void setBirthday(Date birthday) {
		this.birthday = birthday;
	}

	public void setSalary(Double salary) {
		this.salary = salary;
	}
	
	public String execute() throws Exception{
		System.out.println(username);
		System.out.println(password);
		System.out.println(age);
		System.out.println(birthday);
		System.out.println(salary);
		return NONE;
	}
}

<struts>
	<package name="demo2" extends="struts-default" namespace="/">		
		<action name="userAction1" class="demo2.UserAction1"></action>
	</package>
</struts>

JSP的编写:
<h3>方式一:属性驱动-提供属性的set方法的方式</h3>
<s:fielderror/>
	<form action="${ pageContext.request.contextPath }/userAction1.action"
		method="post">
		<table border="1px">
			<tr align="center">
				<td>用户名:</td>
				<td><input type="text" name="username"></td>
			</tr>
			<tr align="center">
				<td>密码:</td>
				<td><input type="password" name="password"></td>
			</tr>
			<tr align="center">
				<td>年龄:</td>
				<td><input type="text" name="age"></td>
			</tr>
			<tr align="center">
				<td>生日:</td>
				<td><input type="text" name="birthday"></td>
			</tr>
			<tr align="center">
				<td>工资:</td>
				<td><input type="text" name="salary"></td>
			</tr>
			<tr align="left">
				<td colspan="2"><input type="submit" value="提交" /></td>
			</tr>
		</table>
	</form>

2、属性驱动,数据封装方式二:在页面中提供表达式方式

public class UserAction2 extends ActionSupport {
	// 提供User对象
	private User user;
	// 提供user的set和get方法:一定要提供get方法
	// 因为拦截器完成数据封装,需要创建User对象将其赋给“private User user;”中的user,
	// 对属性赋值的时候会判断有没有user的实例,如何获取user,通过get方法,没有get方法,就获取不到user
	// 每一次对属性进行赋值的时候,如果没有user就会new一个,User有5个属性,如果没有get方法,new了五次对象
	public User getUser() {
		return user;
	}
	public void setUser(User user) {
		this.user = user;
	}
	
   @Override
	public String execute() throws Exception {
		System.out.println(user);
		return NONE;
	}
}

<struts>

	<package name="demo2" extends="struts-default" namespace="/">
		<action name="userAction2" class="demo2.UserAction2"></action>
	</package>

</struts>

在JSP中的编写:
<h3>方式二:属性驱动-在页面中提供表达式方式</h3>

	<form action="${ pageContext.request.contextPath }/userAction2.action"
		method="post">
		<table border="1px">
			<tr align="center">
				<td>用户名:</td>
				<td><input type="text" name="user.username"></td>
			</tr>
			<tr align="center">
				<td>密码:</td>
				<td><input type="password" name="user.password"></td>
			</tr>
			<tr align="center">
				<td>年龄:</td>
				<td><input type="text" name="user.age"></td>
			</tr>
			<tr align="center">
				<td>生日:</td>
				<td><input type="text" name="user.birthday"></td>
			</tr>
			<tr align="center">
				<td>工资:</td>
				<td><input type="text" name="user.salary"></td>
			</tr>
			<tr align="left">
				<td colspan="2"><input type="submit" value="提交" /></td>
			</tr>
		</table>
	</form>

3、模型驱动-模型驱动方式

public class UserAction3 extends ActionSupport implements ModelDriven<User> {

	// 模型驱动使用的对象:前期必须手动提供对象的
	private User user = new User();//手动实例化User

	@Override
	public User getModel() {
		// 模型驱动需要使用的方法:
		return user;
	}
	@Override
	public String execute() throws Exception {
		System.out.println(user);
		return NONE;
		
	}
}

<struts>

	<package name="demo2" extends="struts-default" namespace="/">
		<action name="userAction3" class="demo2.UserAction3"></action>
	</package>

</struts>

在JSP中编写:
<h3>方式三:模型驱动-模型驱动方式</h3>

	<form action="${ pageContext.request.contextPath }/userAction3.action"
		method="post">
		<table border="1px">
			<tr align="center">
				<td>用户名:</td>
				<td><input type="text" name="username"></td>
			</tr>
			<tr align="center">
				<td>密码:</td>
				<td><input type="password" name="password"></td>
			</tr>
			<tr align="center">
				<td>年龄:</td>
				<td><input type="text" name="age"></td>
			</tr>
			<tr align="center">
				<td>生日:</td>
				<td><input type="text" name="birthday"></td>
			</tr>
			<tr align="center">
				<td>工资:</td>
				<td><input type="text" name="salary"></td>
			</tr>
			<tr align="left">
				<td colspan="2"><input type="submit" value="提交" /></td>
			</tr>
		</table>
	</form>

总结:模型驱动方式是最常用的方式:

缺点:只能同时向一个对象中封装数据,而使用第二种方式可以向多个对象中同时封装数据。

八、关于INPUT逻辑视图

INPUT的逻辑视图的配置:
Action接口中提供了五个逻辑视图的名称:
SUCCESS
ERROR
LOGIN
INPUT —input在某些拦截器中会使用
NONE
拦截器会检查错误区域中是否有错误信息:
如果没有:直接到目标Action
如果有:跳转到INPUT逻辑视图

九、Struts2的复杂类型的数据封装

1、封装数据到List集合中

public class ProductAction1 extends ActionSupport{
	
	private List<Product> products;
	//提供集合的get、set方法
	
	public List<Product> getProducts() {
		return products;
	}
	
	public void setProducts(List<Product> products) {
		this.products = products;
	}
	
	@Override
	public String execute(){
		for(Product product:products){
			System.out.println(product);
		}
		return NONE;
	}
	
}

<struts>
	<package name="demo3" extends="struts-default" namespace="/">
		<action name="productAction1" class="demo3.ProductAction1"></action>
	</package>
</struts>

在Jsp中的编写:
<h3>封装到List集合中:批量插入商品</h3>
<form action="${ pageContext.request.contextPath }/productAction1.action" method="post">
	商品名称:<input type="text" name="products[0].name"><br/>
	商品价格:<input type="text" name="products[0].price"><br/>
	商品名称:<input type="text" name="products[1].name"><br/>
	商品价格:<input type="text" name="products[1].price"><br/>
	商品名称:<input type="text" name="products[2].name"><br/>
	商品价格:<input type="text" name="products[2].price"><br/>
	<input type="submit" value="提交">
</form>

2、封装数据到Map集合中

public class ProductAction2 extends ActionSupport {

	private Map<String, Product> map;

	public Map<String, Product> getMap() {
		return map;
	}

	public void setMap(Map<String, Product> map) {
		this.map = map;
	}

	@Override
	public String execute() {
		for(String key:map.keySet()){
			Product product=map.get(key);
			System.out.println(key+"    "+product);
		}
		return NONE;
	}

}

<struts>

	<package name="demo3" extends="struts-default" namespace="/">
		<action name="productAction2" class="demo3.ProductAction2"></action>
	</package>

</struts>

在Jsp中的编写:
<h3>封装到Map集合中:批量插入商品</h3>

<form action="${ pageContext.request.contextPath }/productAction2.action" method="post">
	商品名称:<input type="text" name="map['one'].name"><br/>
	商品价格:<input type="text" name="map['one'].price"><br/>
	商品名称:<input type="text" name="map['two'].name"><br/>
	商品价格:<input type="text" name="map['two'].price"><br/>
	商品名称:<input type="text" name="map['three'].name"><br/>
	商品价格:<input type="text" name="map['three'].price"><br/>
	<input type="submit" value="提交">
</form>

注:下一节总结值栈(重点)。

学习总结:多总结,多思考,学海无涯。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值