淘淘商城第二天

1 篇文章 0 订阅
1 篇文章 0 订阅

淘淘商城
第二天

  1. 课程计划
    第二天:商品列表功能实现
    1、Ssm框架整合。
    2、服务中间件dubbo
    3、整合测试
    4、商品列表查询功能实现。

  2. Ssm框架整合
    2.1. 数据库
    数据库使用mysql数据库,要求5.5以上版本。
    1、在mysql数据库中创建数据库taotao
    2、将创建数据库的脚本导入到taotao中。

2.2. Mybatis逆向工程
使用mybatis官方提供的mybatis-generator生成pojo、mapper接口及映射文件。
并且将pojo放到toatao-manager-pojo工程中。
将mapper接口及映射文件放到taotao-manager-dao工程中。

2.3. 整合思路
1、Dao层:
mybatis整合spring,通过spring管理SqlSessionFactory、mapper代理对象。需要mybatis和spring的整合包。
2、Service层:
所有的service实现类都放到spring容器中管理。由spring创建数据库连接池,并有spring管理实务。

3、表现层:
Springmvc框架,由springmvc管理controller。

2.4. Dao整合
2.4.1. 创建SqlMapConfig.xml配置文件

<?xml version="1.0" encoding="UTF-8" ?>

2.4.2. Spring整合mybatis
创建applicationContext-dao.xml


<context:property-placeholder location=“classpath:properties/*.properties” />




















db.properties
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/taotao?characterEncoding=utf-8
jdbc.username=root
jdbc.password=root

备注:
Druid是目前最好的数据库连接池,在功能、性能、扩展性方面,都超过其他数据库连接池,包括DBCP、C3P0、BoneCP、Proxool、JBoss DataSource。
Druid已经在阿里巴巴部署了超过600个应用,经过多年多生产环境大规模部署的严苛考验。

2.5. Service整合
2.5.1. 管理Service

<?xml version="1.0" encoding="UTF-8"?>

<context:component-scan base-package="com.taotao.service"></context:component-scan>

2.5.2. 事务管理
创建applicationContext-trans.xml







<tx:advice id=“txAdvice” transaction-manager=“transactionManager”>
tx:attributes

<tx:method name=“save*” propagation=“REQUIRED” />
<tx:method name=“insert*” propagation=“REQUIRED” />
<tx:method name=“add*” propagation=“REQUIRED” />
<tx:method name=“create*” propagation=“REQUIRED” />
<tx:method name=“delete*” propagation=“REQUIRED” />
<tx:method name=“update*” propagation=“REQUIRED” />
<tx:method name=“find*” propagation=“SUPPORTS” read-only=“true” />
<tx:method name=“select*” propagation=“SUPPORTS” read-only=“true” />
<tx:method name=“get*” propagation=“SUPPORTS” read-only=“true” />
</tx:attributes>
</tx:advice>

aop:config
<aop:advisor advice-ref=“txAdvice”
pointcut=“execution(* com.taotao.service..(…))” />
</aop:config>

2.5.3. Web.xml

<?xml version="1.0" encoding="UTF-8"?>


taotao-manager


contextConfigLocation
classpath:spring/applicationContext*.xml


org.springframework.web.context.ContextLoaderListener

2.6. 表现层整合
2.6.1. Springmvc.xml

<?xml version="1.0" encoding="UTF-8"?>

<context:component-scan base-package="com.taotao.controller" />
<mvc:annotation-driven />
<bean
	class="org.springframework.web.servlet.view.InternalResourceViewResolver">
	<property name="prefix" value="/WEB-INF/jsp/" />
	<property name="suffix" value=".jsp" />
</bean>	

2.6.2. web.xml

<?xml version="1.0" encoding="UTF-8"?>


taotao-manager-web

login.html



CharacterEncodingFilter
org.springframework.web.filter.CharacterEncodingFilter

encoding
utf-8



CharacterEncodingFilter
/*

<!-- springmvc的前端控制器 -->
<servlet>
	<servlet-name>taotao-manager</servlet-name>
	<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
	<!-- contextConfigLocation不是必须的, 如果不配置contextConfigLocation, springmvc的配置文件默认在:WEB-INF/servlet的name+"-servlet.xml" -->
	<init-param>
		<param-name>contextConfigLocation</param-name>
		<param-value>classpath:spring/springmvc.xml</param-value>
	</init-param>
	<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
	<servlet-name>taotao-manager</servlet-name>
	<url-pattern>/</url-pattern>
</servlet-mapping>

2.7. 系统间通信
2.7.1. 分析
由于淘淘商城是基于soa的架构,表现层和服务层是不同的工程。所以要实现商品列表查询需要两个系统之间进行通信。
如何实现远程通信?
1、Webservice:效率不高基于soap协议。项目中不推荐使用。
2、使用restful形式的服务:http+json。很多项目中应用。如果服务太多,服务之间调用关系混乱,需要治疗服务。
3、使用dubbo。使用rpc协议进行远程调用,直接使用socket通信。传输效率高,并且可以统计出系统之间的调用关系、调用次数。
2.7.2. 什么是dubbo
随着互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已无法应对,分布式服务架构以及流动计算架构势在必行,亟需一个治理系统确保架构有条不紊的演进。

• 单一应用架构
• 当网站流量很小时,只需一个应用,将所有功能都部署在一起,以减少部署节点和成本。
• 此时,用于简化增删改查工作量的 数据访问框架(ORM) 是关键。
• 垂直应用架构
• 当访问量逐渐增大,单一应用增加机器带来的加速度越来越小,将应用拆成互不相干的几个应用,以提升效率。
• 此时,用于加速前端页面开发的 Web框架(MVC) 是关键。
• 分布式服务架构
• 当垂直应用越来越多,应用之间交互不可避免,将核心业务抽取出来,作为独立的服务,逐渐形成稳定的服务中心,使前端应用能更快速的响应多变的市场需求。
• 此时,用于提高业务复用及整合的 分布式服务框架(RPC) 是关键。
• 流动计算架构
• 当服务越来越多,容量的评估,小服务资源的浪费等问题逐渐显现,此时需增加一个调度中心基于访问压力实时管理集群容量,提高集群利用率。
• 此时,用于提高机器利用率的 资源调度和治理中心(SOA) 是关键。

Dubbo就是资源调度和治理中心的管理工具。

2.7.3. Dubbo的架构

节点角色说明:
Provider: 暴露服务的服务提供方。
Consumer: 调用远程服务的服务消费方。
Registry: 服务注册与发现的注册中心。
Monitor: 统计服务的调用次调和调用时间的监控中心。
Container: 服务运行容器。
调用关系说明:
0. 服务容器负责启动,加载,运行服务提供者。

  1. 服务提供者在启动时,向注册中心注册自己提供的服务。
  2. 服务消费者在启动时,向注册中心订阅自己所需的服务。
  3. 注册中心返回服务提供者地址列表给消费者,如果有变更,注册中心将基于长连接推送变更数据给消费者。
  4. 服务消费者,从提供者地址列表中,基于软负载均衡算法,选一台提供者进行调用,如果调用失败,再选另一台调用。
  5. 服务消费者和提供者,在内存中累计调用次数和调用时间,定时每分钟发送一次统计数据到监控中心。
    2.7.4. 使用方法
    Dubbo采用全Spring配置方式,透明化接入应用,对应用没有任何API侵入,只需用Spring加载Dubbo的配置即可,Dubbo基于Spring的Schema扩展进行加载。

单一工程中spring的配置



远程服务:
在本地服务的基础上,只需做简单配置,即可完成远程化:

将上面的local.xml配置拆分成两份,将服务定义部分放在服务提供方remote-provider.xml,将服务引用部分放在服务消费方remote-consumer.xml。
并在提供方增加暴露服务配置dubbo:service,在消费方增加引用服务配置dubbo:reference。
发布服务:

调用服务:

<dubbo:reference id=“xxxService” interface=“com.xxx.XxxService” />

2.7.5. 注册中心
注册中心负责服务地址的注册与查找,相当于目录服务,服务提供者和消费者只在启动时与注册中心交互,注册中心不转发请求,压力较小。使用dubbo-2.3.3以上版本,建议使用zookeeper注册中心。
Zookeeper是Apacahe Hadoop的子项目,是一个树型的目录服务,支持变更推送,适合作为Dubbo服务的注册中心,工业强度较高,可用于生产环境,并推荐使用

Zookeeper的安装:
第一步:安装jdk
第二步:解压缩zookeeper压缩包
第三步:将conf文件夹下zoo_sample.cfg复制一份,改名为zoo.cfg
第四部:创建一个目录叫做data 路径为/usr/local/zookeeper/zookeeper-3.4.6/data
第五步:修改配置dataDir属性,指定一个真实目录 /usr/local/zookeeper/zookeeper-3.4.6/data
第六步:
启动zookeeper:bin/zkServer.sh start
关闭zookeeper:bin/zkServer.sh stop
查看zookeeper状态:bin/zkServer.sh status
注意要关闭linux的防火墙。
/sbin/iptables -I INPUT -p tcp --dport 2181 -j ACCEPT
/etc/rc.d/init.d/iptables save
3. 整合测试
3.1. 需求
根据商品id查询商品信息,并将商品信息使用json数据返回。
3.2. 分析
请求的url:/item/{itemId}
参数:商品id,从请求的url中获得
返回值:TbItem对象,逆向工程生成的pojo(响应json数据)
3.3. Dao层
根据商品id查询商品信息,单表查询可以使用逆向工程生成的代码。
3.4. Service层
1、在taotao-manager-interface工程中创建一个ItemService接口
2、在taotao-manager-Service工程中创建一个itemSeviceImpl的实现类。
@Service
public class ItemServiceImpl implements ItemService {

@Autowired
private TbItemMapper itemMapper;

@Override
public TbItem getItemById(long itemId) {
	TbItem tbItem = itemMapper.selectByPrimaryKey(itemId);
	return tbItem;
}

}
3.5. 发布服务
1、在taotao-manager-Service工程中添加dubbo依赖的jar包。

	<dependency>
		<groupId>com.alibaba</groupId>
		<artifactId>dubbo</artifactId>
		<exclusions>
			<exclusion>
				<groupId>org.springframework</groupId>
				<artifactId>spring</artifactId>
			</exclusion>
			<exclusion>
				<groupId>org.jboss.netty</groupId>
				<artifactId>netty</artifactId>
			</exclusion>
		</exclusions>
	</dependency>
	<dependency>
		<groupId>org.apache.zookeeper</groupId>
		<artifactId>zookeeper</artifactId>
	</dependency>
	<dependency>
		<groupId>com.github.sgroschupf</groupId>
		<artifactId>zkclient</artifactId>
	</dependency>

2、在spring的配置文件中添加dubbo的约束,然后使用dubbo:service发布服务。

<?xml version="1.0" encoding="UTF-8"?>

<context:component-scan base-package="com.taotao.service"></context:component-scan>

<!-- 使用dubbo发布服务 -->
<!-- 提供方应用信息,用于计算依赖关系 -->
<dubbo:application name="taotao-manager" />
<dubbo:registry protocol="zookeeper"
	address="192.168.25.154:2181,192.168.25.154:2182,192.168.25.154:2183" />
<!-- 用dubbo协议在20880端口暴露服务 -->
<dubbo:protocol name="dubbo" port="20880" />
<!-- 声明需要暴露的服务接口 -->
<dubbo:service interface="com.taotao.service.ItemService" ref="itemServiceImpl" />
3.6. 引用服务 1、在taotao-manager-web工程中添加dubbo依赖的jar包 com.alibaba dubbo org.springframework spring org.jboss.netty netty org.apache.zookeeper zookeeper com.github.sgroschupf zkclient 2、在springmvc的配置文件中添加服务的引用 <?xml version="1.0" encoding="UTF-8"?>
<context:component-scan base-package="com.taotao.controller" />
<mvc:annotation-driven />
<bean
	class="org.springframework.web.servlet.view.InternalResourceViewResolver">
	<property name="prefix" value="/WEB-INF/jsp/" />
	<property name="suffix" value=".jsp" />
</bean>

<!-- 引用dubbo服务 -->
<dubbo:application name="taotao-manager-web"/>
<dubbo:registry protocol="zookeeper" address="192.168.25.154:2181,192.168.25.154:2182,192.168.25.154:2183"/>	
<dubbo:reference interface="com.taotao.service.ItemService" id="itemService" />
3.7. Controller @Controller public class ItemController {
@Autowired
private ItemService itemService;

@RequestMapping("/item/{itemId}")
@ResponseBody
public TbItem getItemById(@PathVariable Long itemId) {
	//根据商品id查询商品信息
	TbItem tbItem = itemService.getItemById(itemId);
	return tbItem;
}

}
3.8. 解决mapper映射文件不发布问题
在taotao-manager-dao工程的pom文件中添加如下内容:

<build>
	<resources>
        <resource>
            <directory>src/main/java</directory>
            <includes>
                <include>**/*.properties</include>
                <include>**/*.xml</include>
            </includes>
            <filtering>false</filtering>
        </resource>
    </resources>
</build>
  1. Dubbo监控中心

需要安装tomcat,然后部署监控中心即可。

1、部署监控中心:
[root@localhost ~]# cp dubbo-admin-2.5.4.war apache-tomcat-7.0.47/webapps/dubbo-admin.war

2、启动tomcat

3、访问http://192.168.25.167:8080/dubbo-admin/
用户名:root
密码:root

如果监控中心和注册中心在同一台服务器上,可以不需要任何配置。
如果不在同一台服务器,需要修改配置文件:
/root/apache-tomcat-7.0.47/webapps/dubbo-admin/WEB-INF/dubbo.properties

  1. 商品列表查询
    5.1. 展示后台首页
    5.1.1. 功能分析
    请求的url:/
    参数:无
    返回值:逻辑视图String
    5.1.2. Controller
    @Controller
    public class PageController {

    @RequestMapping("/")
    public String showIndex() {
    return “index”;
    }

    @RequestMapping("/{page}")
    public String showPage(@PathVariable String page) {
    return page;
    }
    }

5.2. 功能分析
5.2.1. 整合静态页面
静态页面位置:02.第二天(三大框架整合,后台系统搭建)\01.参考资料\后台管理系统静态页面

使用方法:
把静态页面添加到taotao-manager-web工程中的WEB-INF下:

由于在web.xml中定义的url拦截形式为“/”表示拦截所有的url请求,包括静态资源例如css、js等。所以需要在springmvc.xml中添加资源映射标签:
<mvc:resources location="/WEB-INF/js/" mapping="/js/"/>
<mvc:resources location="/WEB-INF/css/" mapping="/css/
"/>
5.2.2. 商品列表页面

对应的jsp为:
item-list.jsp

请求的url:
/item/list
请求的参数:
page=1&rows=30
响应的json数据格式:
Easyui中datagrid控件要求的数据格式为:
{total:”2”,rows:[{“id”:”1”,”name”:”张三”},{“id”:”2”,”name”:”李四”}]}

5.2.3. 响应的json数据格式EasyUIResult
public class EasyUIDataGridResult {

private Integer total;

private List<?> rows;

public EasyUIResult(Integer total, List<?> rows) {
    this.total = total;
    this.rows = rows;
}

public EasyUIResult(Long total, List<?> rows) {
    this.total = total.intValue();
    this.rows = rows;
}

public Integer getTotal() {
    return total;
}

public void setTotal(Integer total) {
    this.total = total;
}

public List<?> getRows() {
    return rows;
}

public void setRows(List<?> rows) {
    this.rows = rows;
}

}

5.2.4. 分页处理
逆向工程生成的代码是不支持分页处理的,如果想进行分页需要自己编写mapper,这样就失去逆向工程的意义了。为了提高开发效率可以使用mybatis的分页插件PageHelper。

5.3. 分页插件PageHelper
5.3.1. Mybatis分页插件 - PageHelper说明
如果你也在用Mybatis,建议尝试该分页插件,这个一定是最方便使用的分页插件。
该插件目前支持Oracle,Mysql,MariaDB,SQLite,Hsqldb,PostgreSQL六种数据库分页。
5.3.2. 使用方法
第一步:把PageHelper依赖的jar包添加到工程中。官方提供的代码对逆向工程支持的不好,使用参考资料中的pagehelper-fix。

第二步:在Mybatis配置xml中配置拦截器插件:







第二步:在代码中使用
1、设置分页信息:
//获取第1页,10条内容,默认查询总数count
PageHelper.startPage(1, 10);

//紧跟着的第一个select方法会被分页

List list = countryMapper.selectIf(1);
2、取分页信息
//分页后,实际返回的结果list类型是Page,如果想取出分页信息,需要强制转换为Page,
Page listCountry = (Page)list;
listCountry.getTotal();
3、取分页信息的第二种方法
//获取第1页,10条内容,默认查询总数count
PageHelper.startPage(1, 10);
List list = countryMapper.selectAll();
//用PageInfo对结果进行包装
PageInfo page = new PageInfo(list);
//测试PageInfo全部属性
//PageInfo包含了非常全面的分页属性
assertEquals(1, page.getPageNum());
assertEquals(10, page.getPageSize());
assertEquals(1, page.getStartRow());
assertEquals(10, page.getEndRow());
assertEquals(183, page.getTotal());
assertEquals(19, page.getPages());
assertEquals(1, page.getFirstPage());
assertEquals(8, page.getLastPage());
assertEquals(true, page.isFirstPage());
assertEquals(false, page.isLastPage());
assertEquals(false, page.isHasPreviousPage());
assertEquals(true, page.isHasNextPage());

5.3.3. 分页测试
@Test
public void testPageHelper() throws Exception {
//初始化spring容器
ApplicationContext applicationContext = new ClassPathXmlApplicationContext(“classpath:spring/applicationContext-*.xml”);
//获得Mapper的代理对象
TbItemMapper itemMapper = applicationContext.getBean(TbItemMapper.class);
//设置分页信息
PageHelper.startPage(1, 30);
//执行查询
TbItemExample example = new TbItemExample();
List list = itemMapper.selectByExample(example);
//取分页信息
PageInfo pageInfo = new PageInfo<>(list);
System.out.println(pageInfo.getTotal());
System.out.println(pageInfo.getPages());
System.out.println(pageInfo.getPageNum());
System.out.println(pageInfo.getPageSize());
}

5.4. Service层
参数:int page ,int rows
业务逻辑:查询所有商品列表,要进行分页处理。
返回值:EasyUIDataGridResult
@Override
public EasyUIDataGridResult getItemList(int page, int rows) {

	//设置分页信息
	PageHelper.startPage(page, rows);
	//执行查询
	TbItemExample example = new TbItemExample();
	List<TbItem> list = itemMapper.selectByExample(example);
	//取分页信息
	PageInfo<TbItem> pageInfo = new PageInfo<>(list);
	
	//创建返回结果对象
	EasyUIDataGridResult result = new EasyUIDataGridResult();
	result.setTotal(pageInfo.getTotal());
	result.setRows(list);
	
	return result;
}

5.4.1. 发布服务

5.5. 表现层
引用服务:

1、初始化表格请求的url:/item/list
2、Datagrid默认请求参数:
1、page:当前的页码,从1开始。
2、rows:每页显示的记录数。
3、响应的数据:json数据。EasyUIDataGridResult
@RequestMapping("/item/list")
@ResponseBody
public EasyUIDataGridResult getItemList(Integer page, Integer rows) {
EasyUIDataGridResult result = itemService.getItemList(page, rows);
return result;
}

可以设置服务超时时间:
服务调用超时时间默认1秒,

Debug设置源代码:

5.6. 安装maven工程跳过测试
clean install -DskipTests

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值