前面实现RMS系统时,我们让其直接访问底层数据库。后面我们在idlewow-game模块实现游戏逻辑时,将不再直接访问底层数据,而是通过hessian服务暴露接口给表现层。

  本章,我们先把hessian服务搭好,并做一个简单的测试,这里以用户注册接口为例。

  先简单介绍下,实现hessian接口,只需要在facade模块暴露接口,然后在core模块实现接口,最后在hessain模块配置好接口路由,将其启动即可。

实现步骤

idlewow-facade

  新建包com.idlewow.user.model,在此包下添加模型类:

package com.idlewow.user.model;

import com.idlewow.common.model.BaseModel;
import lombok.Data; import java.io.Serializable; @Data
public class UserAccount extends BaseModel implements Serializable {
private String username;
private String password;
private String mail;
private String phone;
private String realName;
private String idNo;
private Integer status;
private String remark;
private String registerIp;
}

UserAccount.java

  新建包com.idlewow.user.service,在此包下添加接口类:

package com.idlewow.user.service;

import com.idlewow.common.model.CommonResult;

public interface UserService {
CommonResult register(String username, String password, String ip); CommonResult login(String username, String password);
}

UserService.java

idlewow-core

    新建包com.idlewow.user.mapper,添加mapper文件:

package com.idlewow.user.mapper;

import com.idlewow.user.model.UserAccount;

public interface UserAccountMapper {
int register(UserAccount userAccount); UserAccount login(UserAccount userAccount); UserAccount findByUsername(String username);
}

UserAccountMapper.java

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.idlewow.user.mapper.UserAccountMapper">
<resultMap id="BaseResultMap" type="com.idlewow.user.model.UserAccount">
<result column="id" property="id"/>
<result column="username" property="username"/>
<result column="password" property="password"/>
<result column="mail" property="mail"/>
<result column="phone" property="phone"/>
<result column="real_name" property="realName"/>
<result column="id_no" property="idNo"/>
<result column="status" property="status"/>
<result column="remark" property="remark"/>
<result column="register_ip" property="registerIp"/>
<result column="create_user" property="createUser"/>
<result column="update_user" property="updateUser"/>
<result column="create_time" property="createTime"/>
<result column="update_time" property="updateTime"/>
<result column="is_delete" property="isDelete"/>
<result column="version" property="version"/>
</resultMap>
<!-- 注册 -->
<insert id="register">
insert into user_account (username, password, register_ip, create_user)
values (#{username}, #{password}, #{registerIp}, #{createUser})
</insert>
<!-- 登陆 -->
<select id="login" resultMap="BaseResultMap">
select *
from user_account
where username = #{username} and password = #{password} and is_delete = 0
</select>
<!-- id查询 -->
<select id="find" resultMap="BaseResultMap">
select *
from user_account
where id = #{id} and is_delete = 0
</select>
<!-- 根据用户名查找用户 -->
<select id="findByUsername" resultMap="BaseResultMap">
select *
from user_account
where username = #{username} and is_delete = 0
</select>
<!-- 列表查询总数 -->
<select id="count" resultType="int">
select count(1)
from map_mob
<where>
is_delete = 0
<if test="mapId != null">
and map_id = #{mapId}
</if>
<if test="faction != null">
and faction = #{faction}
</if>
<if test="mobClass != null">
and mob_class = #{mobClass}
</if>
<if test="mobType != null">
and mob_type = #{mobType}
</if>
<if test="levelStart != null">
and level &gt;= #{levelStart}
</if>
<if test="levelEnd != null">
and level &lt;= #{levelEnd}
</if>
<if test="name != null and name != ''">
and name like concat('%', #{name}, '%')
</if>
</where>
</select>
<!-- 列表查询 -->
<select id="list" resultMap="BaseResultMap">
select *
from map_mob
<where>
is_delete = 0
<if test="mapId != null">
and map_id = #{mapId}
</if>
<if test="faction != null">
and faction = #{faction}
</if>
<if test="mobClass != null">
and mob_class = #{mobClass}
</if>
<if test="mobType != null">
and mob_type = #{mobType}
</if>
<if test="levelStart != null">
and level &gt;= #{levelStart}
</if>
<if test="levelEnd != null">
and level &lt;= #{levelEnd}
</if>
<if test="name != null and name != ''">
and name like concat('%', #{name}, '%')
</if>
</where>
<if test="pageParam != null">
limit ${(pageParam.pageIndex - 1) * pageParam.pageSize}, ${pageParam.pageSize}
</if>
</select>
</mapper>

UserAccountMapper.xml

  新建com.idlewow.user.manager包,添加manager类:

package com.idlewow.user.manager;

import com.idlewow.user.mapper.UserAccountMapper;
import com.idlewow.user.model.UserAccount;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component; @Component
public class UserAccountManager {
@Autowired
UserAccountMapper userAccountMapper; public UserAccount findByUsername(String username) {
return userAccountMapper.findByUsername(username);
} public void register(String username, String password, String ip) {
UserAccount userAccount = new UserAccount();
userAccount.setUsername(username);
userAccount.setPassword(password);
userAccount.setRegisterIp(ip);
userAccount.setCreateUser("idlewow");
int effected = userAccountMapper.register(userAccount);
if (effected == 0) {
throw new RuntimeException("sql effected 0 rows");
}
} public UserAccount login(String username, String password) {
UserAccount userAccount = new UserAccount();
userAccount.setUsername(username);
userAccount.setPassword(password);
return userAccountMapper.login(userAccount);
}
}

UserAccountManager.java

  新建com.idlewow.user.service.impl包,添加接口的实现类:

package com.idlewow.user.service.impl;

import com.idlewow.common.model.CommonResult;
import com.idlewow.user.manager.UserAccountManager;
import com.idlewow.user.model.UserAccount;
import com.idlewow.user.service.UserService;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service; @Service("userService")
public class UserServiceImpl implements UserService {
private final Logger logger = LogManager.getLogger(this.getClass().getName());
@Autowired
UserAccountManager userAccountManager; public CommonResult register(String username, String password, String ip) {
try {
UserAccount userAccount = userAccountManager.findByUsername(username);
if (userAccount != null) {
return CommonResult.fail("此用户名已被注册!");
} userAccountManager.register(username, password, ip);
return CommonResult.success();
} catch (Exception ex) {
logger.error("用户注册失败:" + ex.getMessage(), ex);
return CommonResult.fail("用户注册失败");
}
} @Override
public CommonResult login(String username, String password) {
try {
UserAccount userAccount = userAccountManager.login(username, password);
if (userAccount == null) {
return CommonResult.fail("用户名或密码错误!");
} if (userAccount.getStatus() == 1) {
return CommonResult.fail("账号已冻结!");
} return CommonResult.success("", userAccount);
} catch (Exception ex) {
logger.error("用户登录失败:" + ex.getMessage(), ex);
return CommonResult.fail("用户登录失败");
}
}
}

UserServiceImpl.java

  注意,这里ServiceImple类上面有个注解 @Service("userService")。后面我们再添加这种对外的服务类时,都要加这个注解。

idlewow-hessian

  hessian用于服务(器)间通信,实际上也是由一个DispatherServlet接收请求,并转发到各个Service中处理,和springmvc差不多,只不过返回的是二进制数据,而不是视图。我们在pom下添加下列依赖,可以发现依赖的包和mvc差不多。另外,作为启动项目,在plugins节点下,我们配置了启动插件tomcat7以及启动端口20000。

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>idlewow</artifactId>
<groupId>idlewow</groupId>
<version>1.0-SNAPSHOT</version>
</parent>
<modelVersion>4.0.0</modelVersion> <artifactId>idlewow-hessian</artifactId>
<packaging>war</packaging> <dependencies>
<dependency>
<groupId>idlewow</groupId>
<artifactId>idlewow-facade</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>idlewow</groupId>
<artifactId>idlewow-core</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>com.caucho</groupId>
<artifactId>hessian</artifactId>
<version>4.0.60</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context-support</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.1.6.RELEASE</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>4.0.1</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.3</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.1</version>
</dependency>
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-dbcp2</artifactId>
<version>2.6.0</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.tomcat.maven</groupId>
<artifactId>tomcat7-maven-plugin</artifactId>
<version>2.2</version>
<configuration>
<uriEncoding>UTF-8</uriEncoding>
<port>20000</port>
<path>/</path>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-war-plugin</artifactId>
<version>2.6</version>
</plugin>
</plugins>
</build>
</project>

pom.xml

  然后,我们需要在web.xml中配置hessian的servlet,以及添加一个字符编码的filter等等,如下:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
version="3.0">
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/applicationContext.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<filter>
<filter-name>CharacterEncodingFilter</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>utf-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>CharacterEncodingFilter</filter-name>
<url-pattern>/remoting/*</url-pattern>
</filter-mapping>
<servlet>
<servlet-name>remoting</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/hessian-servlet.xml</param-value>
</init-param>
<load-on-startup>2</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>remoting</servlet-name>
<url-pattern>/remoting/*</url-pattern>
</servlet-mapping>
</web-app>

web.xml

  在/resource/spring目录下,新建hessian服务的配置文件hessian-servlet.xml。这个xml主要配置对外暴露的hessian服务。现在我们只配置了UserService,后面每次添加对外的服务接口时,都需要在这里添加配置。

<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
"> <mvc:annotation-driven/> <bean name="/UserService" class="org.springframework.remoting.caucho.HessianServiceExporter">
<property name="service" ref="userService"/>
<property name="serviceInterface" value="com.idlewow.user.service.UserService"/>
</bean>
</beans>

hessian-servlet.xml

  除了这3个配置外,还需要配置applicationContext.xml, jdbc.propetries, dataSource.xml, log4j2.xml,和RMS系统大体一致,这里就不再重复了。具体可在源码中查看。

  全部搞定后,只要把hessian项目启动,即可调用hessian接口了。启动步骤和rms一样,maven命令也是tomcat7:run,工作目录切换到hessian目录下即可。

运行效果

  在game模块中调用hessian时,也是通过在xml中配置注入的方式调用。这里我们还没开始写game模块,为了测试,先简单写一个入口类(即带main函数的类)调用。类似于C#中写控制台程序调用。

  在/src/test/java包下新建一个类HessianTest.java如下:

import com.caucho.hessian.client.HessianProxyFactory;
import com.idlewow.common.model.CommonResult;
import com.idlewow.user.service.UserService;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger; public class HessianTest {
protected static final Logger logger = LogManager.getLogger(HessianTest.class); public static void main(String[] args) {
String url = "http://localhost:20000/remoting/UserService";
HessianProxyFactory factory = new HessianProxyFactory();
try {
UserService userService = (UserService) factory.create(UserService.class, url);
CommonResult commonResult = userService.register("testuser", "123456", "127.0.0.1");
logger.info(commonResult);
} catch (Exception e) {
logger.error(e.getMessage(), e);
e.printStackTrace();
}
}
}

HessianTest.java

  在这个类中,我们指定hessian服务地址,并利用代理工厂创建一个服务代理。然后调用用户注册方法。测试的时候,先把hessian项目启动。然后执行这个main方法即可。

运行效果如下图,可以看到,接口调用成功,并把执行结果在日志中打印了出来。

小结

  本节把hessian服务搭建运行起来,并实现了用户注册登录的接口。后面game模块凡是访问底层数据,均需调用hessian服务。

  本章源码下载:https://idlestudio.ctfile.com/fs/14960372-387256708

  本文原文地址:https://www.cnblogs.com/lyosaki88/p/idlewow_10.html

  项目交流群:329989095

从零开始实现放置游戏(十)——实现战斗挂机(1)hessian服务端搭建的更多相关文章

  1. Kbengine游戏引擎-【2】kbengine引擎服务端目录结构分析

    |- kbengine (KBE_ROOT 根目录) |- demo (游戏例子工程的根目录) |- res (工程的所有资源) |- spaces (通常存放游戏场景相关的资源,例如Navmesh) ...

  2. cocos2d-lua ARPG手机游戏《烈焰遮天》(客户端+服务端+数据库)发布说明

    服务器发布流程及其规范1,环境准备        a, mvn命令行:从\\10.21.210.161\share\tools\apache-maven-3.1.1-bin.tar.gz取出安装包,  ...

  3. 谈一款MOBA游戏《码神联盟》的服务端架构设计与实现

    一.前言 <码神联盟>是一款为技术人做的开源情怀游戏,每一种编程语言都是一位英雄.客户端和服务端均使用C#开发,客户端使用Unity3D引擎,数据库使用MySQL.这个MOBA类游戏是笔者 ...

  4. 谈一款MOBA类游戏《码神联盟》的服务端架构设计与实现(更新优化思路)

    注:本文仅用于在博客园学习分享,还在随着项目不断更新和完善中,多有不足,暂谢绝各平台或个人的转载和推广,感谢支持. 一.前言 <码神联盟>是一款为技术人做的开源情怀游戏,每一种编程语言都是 ...

  5. 看懂 游戏《Minecraft》的崩溃报告 服务端/客户端

    如何看懂Minecraft报错的关键信息. 让你如何看懂Minecraft报错 前言 一些俏皮话 寻找崩溃日志 打开崩溃日志 重要的事说三遍 下载文本编辑器 开始分析 深度分析 得出结论 修复报错 解 ...

  6. 从零开始实现放置游戏(十三)——实现战斗挂机(4)添加websocket组件

    前两张,我们已经实现了登陆界面和游戏的主界面.不过游戏主界面的数据都是在前端写死的文本,本章我们给game模块添加websocket组件,实现前后端通信,这样,前端的数据就可以从后端动态获取到了. 一 ...

  7. 从零开始实现放置游戏(六)——实现挂机战斗(4)导入Excel数值配置

    前面我们已经实现了在后台管理系统中,对配置数据的增删查改.但每次添加只能添加一条数据,实际生产中,大量数据通过手工一条一条添加不太现实.本章我们就实现通过Excel导入配置数据的功能.这里我们还是以地 ...

  8. 从零开始实现放置游戏(七)——实现挂机战斗(5)RMS系统后台参数校验

    前面几章实现了在RMS系统中进行数据的增删查改以及通过Excel批量导入.但仍有遗留的问题,比如在新增或编辑时,怪物的生命值.护甲等数据我们可以输入负值,这种数据是不合理且没有意义的.本章我们就实现服 ...

  9. 从零开始实现放置游戏(六)——实现后台管理系统(4)Excel批量导入

    前面我们已经实现了在后台管理系统中,对配置数据的增删查改.但每次添加只能添加一条数据,实际生产中,大量数据通过手工一条一条添加不太现实.本章我们就实现通过Excel导入配置数据的功能.这里我们还是以地 ...

随机推荐

  1. jquery评分星星

    <!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...

  2. 前端开发常用PhotoShop快捷键整理(更新中)

    图片来源 UI提供的psd图 印屏幕:PrScrn SysRq(键盘按键) 浏览器(插件)获取 常用的快捷键: 新建 Ctrl + N 取消选框 Ctrl + D 反选 Ctrl + shift + ...

  3. RSA加密解密及RSA签名和验证

    原文:RSA加密解密及RSA签名和验证 1.RSA加密解密: (1)获取密钥,这里是产生密钥,实际应用中可以从各种存储介质上读取密钥 (2)加密 (3)解密2.RSA签名和验证 (1)获取密钥,这里是 ...

  4. 如何在项目中添加Log4net_web.config

    <log4net> <appender name="ConsoleAppender" type="log4net.Appender.ConsoleApp ...

  5. winform picturebox设置布局样式

    这里分背景图和直接显示的图片的布局 背景图的布局为BackgroundImageLayout设置为strecth即为自动拉伸 图片的布局为SizeMode 设置为StretchImage即自动拉伸

  6. 使用MSYS2环境中编译Qt5.5.0的补丁

    Qt的configure脚本对MinGW静态编译支持不太完善,总有这样那样的问题.如果你不嫌麻烦,而且可以接受高版本的Qt的话,可以考虑使用我做的补丁在MSYS2环境中编译.Qt5.4.2的补丁 Qt ...

  7. 自动获取淘宝API数据访问的SessionKey

    最近在忙与淘宝做对接的工作,总体感觉淘宝的api文档做的还不错,不仅有沙箱测试环境,而且对于每一个api都可以通过api测试工具生成想要的代码,你完全可以先在测试工具中测试之后再进行代码的编写,这样就 ...

  8. Gps坐标有效性判定

    百科:纬度 是指某点与地球球心的连线和地球赤道面所成的线面角,其数值在0至90度之间.位于赤道以北的点的纬度叫北纬,记为N:位于赤道以南的点的纬度称南纬,记为S. var regex = new Re ...

  9. Google C++测试框架系列入门篇:第二章 开始一个新项目

    上一篇:Google C++测试框架系列入门篇:第一章 介绍:为什么使用GTest? 原始链接:Setting up a New Test Project 词汇表 版本号:v_0.1 开始一个新项目 ...

  10. python实现常用查找算法

    http://www.cnblogs.com/feixuelove1009/p/6148357.html