Jeesite开垦

时间:2022-11-25 04:39:53

1. userIndex里面,$ctx是在哪里定义的?

就是request.context

2. 增加新的包后,扫描配置修改
1) spring-context.xml文件中,扫描非@controller注解,需要添加路径

     <context:component-scan base-package="com.thinkgem.jeesite,com.neusoft.bigdata">
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

2)spring-context.xml文件中,MyBatis的类entity/Dao类扫描

     <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="typeAliasesPackage" value="com.thinkgem.jeesite,com.neusoft.bigdata"/>
... ...
</bean>

3)spring-mvc.xml文件中扫描@controller所需要添加的路径

     <context:component-scan base-package="com.thinkgem.jeesite,com.neusoft.bigdata" use-default-filters="false">
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

3. Jeesite的权限控制

  为什么cms的菜单配置其实没什么权限都没有,但是后台代码却又有@RequiresPermissions("cms:site:view")?
  其实cms的菜单项没有权限是第一层没有权限,真正点开看到下级节点是view、edit、delete等操作定义,都是有明确权限定义的。

  Jeesite的权限控制思路是这样的,首先是给menu赋权限,然后把menu赋给角色,角色包含用户;这样用户就具备了menu权限;然后后台的controller通过@RequestPermission来设定调用该方法的权限;只要你的用户权限中(menu)有了该权限,也就可以调用方法;
  Jeesite的权限思路是:
  1)菜单来实现权限定义;菜单有可见菜单和隐藏菜单,隐藏菜单就是那种纯权限意义,比如XX功能的编辑,删除等。
  2)通过用户和菜单绑定实现授权;
  3)controller的@RequestPermission来指定请求权限;

  权限(认证以及授权逻辑)都是在com.thinkgem.jeesite.modules.sys.security.SystemAuthorizingRealm里面实现的;加载权限的逻辑首先把用户的被赋予的菜单(权限)便利出来放到用户的权限字典中去;在授权的过程中,将会判断操作所需要的权限(由@RequestPermission中定义)是否在用户的permissons字段中有定义;有则可以继续操作,没有则不能继续操作。

4.菜单的设计
  关于菜单的设定,一定是大于两级的(至少三级,否则无法显示左侧的子菜单),第一个级别是顶部菜单(和“我的面板”,“在线办公”等平级);第二个界别是左侧菜单的父级目录(至少有个子菜单),第三级别是左侧父级目录的下级目录/ 叶子节点。

5. jeesite消息提示

  发现提交后,“正在提交,请稍等”的字样一直在转圈,跳转到了List页面也没有消失,而且成功提示也没有。
  后来发现原来需要在List页面增加一句话:
<sys:message content="${message}" />
  这句话一方面展示了成功提示,另一方面可以把“正在提交”的字样给冲跑。

  但是注意,需要在提交处理的函数中增加一个RedirectAttributes,用于存放在redirect过程中存放的变量;这里用于存放成功提示。

6. spring的配置文件

  在实现分页的时候,想要使用缓存来保存查询结构再来进行分页(hive的分页只能这样来处理)。在使用Jeesite内部缓存(EchCache)的时候,在spring的文件中看到了

    <!-- 缓存配置 -->
<bean id="cacheManager" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean">
<property name="configLocation" value="classpath:${ehcache.configFile}" />
</bean>

  是的,${ehcache.configFile}是什么鬼,配置项,但是配置项在哪里?

  原来在这里:
 <context:property-placeholder ignore-unresolvable="true" location="classpath:jeesite.properties" />

  但是为什么点击下一页的时候还是定位pageNo为0?
  1)这是因为jeesite提供了一个分页拦截器PaginationInterceptor;这个分页拦截器将会填充count值;
  2)有了这个count值,还不够,在Page的setList函数中,将会调用一个函数initialize(),这个函数将会根据count来计算last等一系列值;这一系列值将会在渲染HTML页面的时候起到设定参数的作用。
  

  分页的水,还是很深的。

7. 分页

前台页面有一个脚本函数,叫做page(默认的名称,如果是多个分页,用到不同的逻辑,需要设置Page对象funcName属性);

 function page(n,s){
$("#pageNo").val(n);
$("#pageSize").val(s);
$("#pageForm").submit(); return false;
}

  这里面的n参数是关键,页面在渲染的时候,会把前一页的调用脚本参数设置为当前页的前一个索引;会把后一个页的脚本参数设置为当前页的后一个索引;
  提交(commit)之后,到了controller,controller将会从request对象中取出pageNo以及pageSize的值填充到后台构造的新的Page对象,然后到后台数据库中取出对应的数据集,这里的分页的sql是自动生成的sql中已经生成好的,你只需要调用一个findList函数即可;
  Page的构造函数中有个HttpServletRequest对象比较好理解,用于获取参数,但是还有一个HttpServletResponse对象,这个response对象用于在cookie中存放对象,这样就可以在前台的cookie中存放这些内容;从代码逻辑来看,是当request中无法获取这些信息的时候,将会从cookie中获取。
  因为page需要response对象,所以处理分页的controller的参数中是需要包含response的。
  public String execSql(SqlExeInfo info, Model model, HttpServletRequest request, HttpServletResponse response) { ... ... } 
  然后page对象将会生成渲染html,前面讲到的前一页,后一页的绑定脚本以及参数的设置都是在page中来做的;另外,通过访问数据库获取的分页数据集也是保留在Page对象中。
  controller的返回也是把填充好的page对象放到model的attribute中。
  页面渲染的时候会把分页部分html生成好;展示出来。然后点击、输入分页信息将会跳转到对应的controller进行处理... ...

8.代码生成需要指定至少一个更新列

  今天要自动生成SqlHistory,但是生成总是失败;后来跟踪了一下代码;才发现原来是因为Jeesite是要求你至少要有一列是可以“更新”的;因为在生成dao.xml的时候,他会去读取更新的字段,如果没有那么就会返回一个异常;这个地方也是它的一个BUG吧。
  不过代码生成这部分使用的是freemarker来搞的;有时间可以研究一下。

9.Could not resolve type alias 'SqlHistory'

Caused by: org.apache.ibatis.builder.BuilderException: Error resolving class. Cause: org.apache.ibatis.type.TypeException: Could not resolve type alias 'SqlHistory'.  Cause: java.lang.ClassNotFoundException: Cannot find class: SqlHistory
  Jeesite需要在spring-context.xml中节点“sqlSessionFactory”以及“mapperScannerConfigurer”节点,增加项目的节点扫描(默认只有jeesite的)

10.未解配置

   <filter>
<filter-name>httpMethodFilter</filter-name>
<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>httpMethodFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

RESTful的filter要放在最后面,如果放在encodingFilter的前面将会导致乱码问题。这是为什么?

10.Jeesite字段变更,SQL文件重新生成处理

  如果表的字段发生了变化,这个时候注意需要把业务表配置以及生成方案都删掉,这是因为表的配置一旦配置死了字段名称和entity的对应关系,也就没有了修改的余地了(表字段是固定死的,不再会和物理表进行同步)。

11. 缓存

  发现总是爆Ehcache序列化异常,因为HiveConnection;但是诡异的是我根本就没有把hive session放置到Ehcache中啊。
  没事,你没放,自然有人放;其实HttpSession被Jessite框架管理的是会被不断序列化的,因为HiveConnection没有实现ISerializalbe接口,所以序列化失败。
  我在读源码的时候,发现还有一个MemerySessionDao挺好,没有Ehcache那么重,还序列化到硬盘中。但是为什么spring初始化的时候报错?我想了一下,觉得可能是因为MemorySessionDAO所在的包并不是Spring监控的路径;于是我在CacheSessionDAO同级包体下添加了一个类,继承自MemeryCacheDAO;但是总是报错,Login在反射初始化SessionDAO的时候并没有找到一个类继承自SessionDAO;这就很奇怪了,因为我翻看了源码MemeryDAO是实现了这个SessionDAO;
  但是其实是此SessionDAO非彼SessionDAO。
  明白了;于是我又继承了这个SessionDAO,实现了两个函数,这次才成功。

12. 自动生成用户树

  Jeesite只要在指定类型里面指定为User,那么就可以直接在生成页面中生成下拉框中!完全不需要自己去写;这种体力活我一直很无奈,但是看到页面直接就生成了树状结构;整个人都惊讶。是的,Jeesite在这一点上面解决了我的痛点。默认如果在数据库字段定义名称为user_id,则再生成实体类的时候自动绑定到User,然后在生成页面的时候自动显示为树。
13. 对象实例化依赖于声明
  Jeesite诡异事件,dao竟然没有实例化!
  因为调用的时候采用的是直接new了一个类,而没有采用autowired的方式导致了,内部的类实例化有问题。