GWT(GXT)+OpenJPA使用RPC时Date类型出错的情况分析

先来一个完整的错误提示如下:

com.google.gwt.user.client.rpc.SerializationException: Type 'org.apache.openjpa.util.java$util$Date$proxy' was not included in the set of types which can be serialized by this SerializationPolicy or its Class object could not be loaded. For security purposes, this type will not be serialized.: instance = Wed Sep 21 00:00:00 CST 2011

 

错误里面废话很多,大概就是有个"org.apache.openjpa.util.java$util$Date$proxy "我RPC不认识。我当时就奇怪了,我在POJO的时候用的是java.util.Date,没有使用这个什么奇怪的数据类型。请出Google大神貌似也说不清什么原因。

 

再说说我的做法:

我使用OpenJPA构建来一个@Entity

@Entity

@Table(name = "t_wj")

public class TWj implements java.io.Serializable {

  private static final long serialVersionUID = -3474120905033096760L;

  private Date wjBy7; // 备用7

  @Temporal(TemporalType.TIMESTAMP)
  @Column(name = "wj_by7", length = 19)
  public Date getWjBy7() {
    return wjBy7;
  }

  public void setWjBy7(Date wjBy7) {
    this.wjBy7 = wjBy7;
  }
  // 其中省略无数代码
}

 

我再构造了一个Model:

public class MWj {

  private Date wjBy7; // 备用7

  public Date getWjBy7() {
    return wjBy7;
  }

  public void setWjBy7(Date wjBy7) {
    this.wjBy7 = wjBy7;
  }
  // 其中省略无数代码
}

 

最后使用Spring的BeanUtils.copyProperties将两个类的参数合并

最后神奇的事情发生了,在@Temporal(TemporalType.TIMESTAMP)的指引下,OpenJPA将wjBy7转换成了"org.apache.openjpa.util.java$util$Date$proxy "如果你将MWj通过RPC传回很不幸,GWT会无情的拒绝你

 

怎办?转换吧:

setWjBy7(new Date(twj.getWjBy7().getTime()))

 

OK,万事大吉

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我来给你讲一下具体的步骤。 1. 初始化项目 首先,我们需要使用 Vue CLI 4+ 创建一个新的项目,选择 TypeScript 作为语言: ``` vue create my-project ``` 然后,在项目根目录下执行以下命令安装 Element Plus、Vite、GWT 等依赖: ``` npm install element-plus vite vue-router vuex axios qs ``` 2. 配置 Vite 我们需要对 Vite 进行一些配置以实现换肤和全局 loading 控制。在项目根目录下创建一个 `vite.config.ts` 文件,添加以下内容: ```typescript import { defineConfig } from 'vite'; export default defineConfig({ css: { preprocessorOptions: { scss: { additionalData: ` @import "~element-plus/packages/theme-chalk/src/index"; @import "@/styles/variables.scss"; ` } } }, server: { proxy: { '/api': { target: 'http://localhost:3000', // GWT 后端地址 changeOrigin: true, rewrite: (path) => path.replace(/^\/api/, '') } } } }) ``` 其中,我们使用了 Element Plus 自带的 SCSS 变量来实现换肤,需要在 `variables.scss` 中定义相应的变量。`server.proxy` 则是用于配置 GWT 后端地址的代理。 3. 配置 Element Plus 在 `main.ts` 中,我们需要引入 Element Plus 并使用它的样式和组件: ```typescript import { createApp } from 'vue'; import ElementPlus from 'element-plus'; import App from './App.vue'; import router from './router'; import store from './store'; import 'element-plus/lib/theme-chalk/index.css'; const app = createApp(App); app.use(ElementPlus); app.use(router); app.use(store); app.mount('#app'); ``` 4. 配置路由和状态管理 我们使用 Vue Router 来实现路由,Vuex 来实现全局状态管理。在 `store` 目录下创建 `index.ts` 文件,添加以下内容: ```typescript import { createStore } from 'vuex'; export default createStore({ state: { loading: false }, mutations: { setLoading(state, payload) { state.loading = payload; } }, actions: { setLoading({ commit }, payload) { commit('setLoading', payload); } }, modules: { } }) ``` 在 `router` 目录下创建 `index.ts` 文件,添加以下内容: ```typescript import { createRouter, createWebHistory, RouteRecordRaw } from 'vue-router'; import Home from '../views/Home.vue'; const routes: Array<RouteRecordRaw> = [ { path: '/', name: 'Home', component: Home } ]; const router = createRouter({ history: createWebHistory(process.env.BASE_URL), routes }); export default router; ``` 5. 实现换肤功能 在 `styles` 目录下创建 `variables.scss` 文件,添加以下内容: ```scss /* 主题色 */ $--color-primary: #409EFF; /* 辅助色 */ $--color-success: #67C23A; $--color-warning: #E6A23C; $--color-danger: #F56C6C; $--color-info: #909399; /* 背景色 */ $--color-background: #f5f7fa; $--color-card: #fff; /* 字体 */ $--font-family: 'Helvetica Neue', Arial, 'PingFang SC', 'Microsoft YaHei', sans-serif; ``` 我们可以在 `App.vue` 中通过修改 `document.documentElement.style` 中的 CSS 变量来实现换肤。具体代码如下: ```html <template> <div id="app"> <!-- ... --> </div> </template> <script> import { defineComponent } from 'vue'; import { useStore } from 'vuex'; export default defineComponent({ setup() { const store = useStore(); function switchTheme(theme: string) { const app = document.documentElement; app.style.setProperty('--color-primary', theme); } return { switchTheme }; } }); </script> ``` 6. 实现全局 loading 控制 在 `App.vue` 中,我们可以使用 Vuex 中的 `loading` 状态来控制全局 loading。具体代码如下: ```html <template> <div id="app"> <div v-if="loading" class="loading-mask"> <div class="loading-spinner"></div> </div> <!-- ... --> </div> </template> <script> import { defineComponent } from 'vue'; import { useStore } from 'vuex'; export default defineComponent({ setup() { const store = useStore(); return { loading: store.state.loading }; } }); </script> <style scoped> .loading-mask { position: fixed; z-index: 9999; top: 0; left: 0; right: 0; bottom: 0; background-color: rgba(255, 255, 255, 0.5); } .loading-spinner { position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); border: 3px solid #ccc; border-top-color: #409EFF; border-radius: 50%; width: 30px; height: 30px; animation: spin 0.6s infinite linear; } @keyframes spin { to { transform: rotate(360deg); } } </style> ``` 然后,在需要控制 loading 的地方,我们可以使用以下代码来控制全局 loading: ```typescript import { useStore } from 'vuex'; const store = useStore(); // 显示 loading store.dispatch('setLoading', true); // 隐藏 loading store.dispatch('setLoading', false); ``` 到这里,我们就完成了一个基于 Vue 3、TypeScript、Element Plus、Vite 的代码框架,并且支持换肤和全局 loading 控制。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值