项目开发笔记-传单下发 名片替换 文件复制上传/html静态内容替换/json解析/html解析
//////////////////////////// 注意: 此博客是个人工作笔记 非独立demo//////////////////////////////////
........................................................................................................................................................................................................................
业务介绍:
最近搞的新系统有一个传单发放的功能,因为要赶着新系统先上线使用,所以老大说先使用老系统的代码,就是先写个接口调用支撑一下 下面是新系统调用老系统接口的方法
URL json(gson) html(jsoup)文件 及 IO流的使用
下面首先是http远程调用的方法 先称之为工具类
1 public String postTelentService(String urlStr,String urlParameter) throws IOException{ 2 //java.net.URL 主要用于基于Internet的网络通信编程 3 URL url = new URL(urlStr);//URL的构造方法 4 //参数是值 请求的URL连接的地址(就是url) 字符串类型的 5 HttpURLConnection urlcon = (HttpURLConnection)url.openConnection();//打开这个url连接 此方法声明抛出IO异常 返回值是URLConnection类型的 连接对象 6 //我们强制转型为他的一个实现类:HttpURLConnection 7 urlcon.setDoOutput(true);//设置是否向connection输出 8 //因为我们是个post类型的请求 参数要放在http正文当中,所以我们要设置成true 9 urlcon.setDoInput(true);// read from the connection.Defult is true 10 urlcon.setRequestMethod("POST");//设置http请求的方法 默认是GET方式 11 12 urlcon.setUseCaches(false);//POST请求不能使用缓存 13 urlcon.setInstanceFollowRedirects(true);//前者设置所有的http连接是否自动处理重定向 14 15 16 urlcon.setRequestProperty("Content-Type","application/x-www-form-urlencoded");//配置本次连接的Content-type,配置为application/x-www-form-urlencoded的意思是 正文是urlencoded编码过的form参数 下面我们可以看到我们对正文内容使用URLEncoder.encode 进行编码 17 /*google大神解释: 18 * 这个是告诉服务器 你的客户端的配置/需求 19 比如说你要取某个文件的多少字节到多少字节就通过这个东西告诉服务器 20 你的客户端支持压缩,也可以告诉服务器 服务器会压缩传输 21 你的客户端支持什么编码 也可以告诉服务器 服务器会尽量按照你的编码传递数据 22 还有比如你的客户端是什么类型,IE,FIrefox之类,有的服务器会按照你的客户端类型给你传送文本 23 你啥都不告诉 服务器就按缺省配置传递内容给你的客户端 24 */ 25 /** 还有注意点: 26 * 1 连接,从openConnection()至此的相关配置设定 必须在connect()之前完成! 27 * 2 connection.getOutputStream会隐含的进行connect()方法 28 */ 29 urlcon.connect();//配置完后 (规定好格式的传输对象? 打开对象的连接?) 30 31 DataOutputStream out = new DataOutputStream(urlcon.getOutputStream());//获取 当前连接对象的输出流 然后包装成数据流 32 33 String content = urlParameter;//post方法的http请求正文起始跟 url中?后的参数字符串一致 34 out.writeBytes(content);//byte=8bit 以bate的形式讲内容写到向外的流内 35 out.flush();//强行刷入内存 36 out.close();//关闭输出流 37 38 InputStream is = urlcon.getInputStream();//为了获取结果的 输入流 原始流 39 BufferedReader buffer = new BufferedReader(new InputStreamReader(is));//包装成字符输出流 再转换成字节输出流 40 StringBuffer bs = new StringBuffer(); 41 String l = null; 42 while((l=buffer.readLine())!=null){ //读取一行 如果不为空则 in √ 43 bs.append(l).append("/n");//换行 44 } 45 String str1 = bs.toString();// 将Stringbuffer转换成字符串 46 str1 = new String(str1.getBytes("GBK"),"UTF-8");//gbk转成utf-8 47 String str2 = str1.subString(0,str1.length()-2);//获取字符串 截取后面最后一位 48 return str2; 49 }
好了 方法封装完毕 io流有些日子没写过 生疏了不少 下面是调用方法的时候的代码 简单的写写
1 String assUserIds = RequestUtil.getStringParam(request, "assUserIds"); 2 String urlStr="http://"+ Config.getMkpUrl()+"/marketa/remotePassFlyers.do"; // 调用其他系统的方法 基于http调用 路径配置 3 // urlStr+="?userId="+userId+"&maId="+maId+"&flyerId="+flyerId+"&userIds="+userIds+"&assUserIds="+assUserIds; 4 String urlParameter="userId="+userId+"&maId="+maId+"&flyerId="+flyerId+"&userIds="+userIds+"&assUserIds="+assUserIds; 5 // userId=8665&maId=1015&flyerId=38273&userIds=8665,8687,8688 6 String result = null; 7 inviteAllUsers(request); 8 try { 9 result = this.postTelentService(urlStr,urlParameter);//路径和参数传递 请求 10 } catch (IOException e) { 11 e.printStackTrace(); 12 }
//---------------------------------------------------------------------------------------------------------------------------------------------------------
好了新系统的 调用方法撰写完毕
//=============================================================================================//
新系统用的springMVC spring MyBatis
老系统相信通过那个url路径就能看出以前用的是struts
又因为老系统已经上线了几年了 老员工告诉我尽量不要动他们写好的代码 直接调用或者自己重写一个 到时候更新.class文件就行了
//============================================================================================//
分析业务逻辑 及 需求分析
涉及到两个表 第一个表 d_flyer是传单对应表 里面存放的有 传单的id 复制的谁的模板 更新时间 所属人和活动 等等
d_flyer_detail 里面存放了 对应的个人内容==
老系统传单模块分析: 首先由 主管 创建一个活动 然后为此活动创建一个相应的推广的传单
传单: 一个静态的html文件 里面有需要展示的内容 :
1 名片 ./
-------------------------------------
2 反馈表 用于采集有意向的用户的信息
当 主管 创建完成后 传单的名片(如果此电子传单有名片)内部是主管的相关的信息 然后 主管 邀请 组织内部人员参加此活动 并为人员下发此传单 因为主管点击邀请后 被邀请人需要登录账号 进入软件后接受邀请 然后主管才能给其发送传单 而且这样太繁琐
因为下发传单也涉及到了传单的复制(即 主管传单的静态html文件要复制一份给当前的用户) 他复制的地址一直是在老系统中 所以说只能让老系统提供给新系统一个调用接口 来进行调用
新系统对应需求是: 勾选相应的人员 给他们发送传单 跳过了邀请接受的繁琐
即:批量发送功能
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
由url的http请求内容(即参数)我们看出 调用老系统的参数有:
"userId=" userId //当前账号的id
"&maId=" maId //当前活动的id
"&flyerId=" flyerId //当前活动绑定传单的id
"&userIds=" userIds //被邀请人的id s 多个
"&assUserIds=" assUserIds //其他被邀请人的id s
//
//下面是老系统写的接口:
//刚才已经说了 老系统是拿struts spring 和JdbcTemplate写的
//本人并不知道jdbcTemplate的原理 但是感觉应该就是一个数据库框架 没有时间研究它了 同事也说已经过时很久了 不是主流了 只能先放着
//其他的不说了 struts是action了 接口如下:
1 public class Action{ 2 /** 3 * 这个方法是给另一个系统调用的 4 * 传递过来的参数分析: 业务分析: 5 * 1 userId 当前登录的用户的id 有一个单独的人(主管)专门做传单 传单完成后进行下发 6 * 所以说他就是在做传单的模板 7 * 2 maId 指的是当前活动的id 下发传单实际上是对现有(某个活动某个用户)模板传单进行复制 8 * 在表d_flyer中 9 * 进行了赋值 穿但是的user_id字段赋以接受传单人的id 10 * 3 flyerId 指的是传单id 它现在写的应该是错误的(但是有个逻辑就是 一个活动只有一个原本的传单模板) 11 * 4 userIds 指的是下发到的目标用户 本来是个数组 后来传过来的时候是用了 toString()方法 12 * 5 assUserIds 已经废弃 13 * 远程下发传单 14 * @return 15 */ 16 public String remotePassFlyers(){ 17 //远程调用的参数示例为: userId=8665&maId=1015&flyerId=38273&userIds=8665,8687,8688&assUserIds= 18 System.out.println("This is in:Action.remotePassFlyers() ; \n " + 19 "---wunian7yulian testing...-----The var is: null , And The value is: " + "--flag1--" + " \t ---------"); 20 Integer userId = Integer.parseInt(this.getRequest().getParameter("userId")); 21 Integer maId = Integer.parseInt(this.getRequest().getParameter("maId")); 22 Integer flyerId = Integer.parseInt(this.getRequest().getParameter("flyerId")); 23 24 String userIds = this.getRequest().getParameter("userIds"); 25 String assUserIds = this.getRequest().getParameter("assUserIds"); 26 27 if(isBlank(maId) || isBlank(flyerId)){ //如果没有活动参数或者传单参数 28 return "ERROR INPUT PARAM"; 29 } 30 //................................................主要方法....................... 31 this.getMarketaService().passFlyerToUsers(userId, maId, flyerId, userIds, assUserIds);//spring注入的service层实例对象 32 return SUCCESS; 33 // return maId+";"+flyerId+";"+userIds+";"+assUserIds; 34 } 35 36 private boolean isBlank(Integer i){ 37 boolean flag = true; 38 if(null!=i && 0!=i){ 39 flag = false; 40 } 41 return flag; 42 } 43 } 44 //=======================================================================================================
下面是service层的代码 有详细清楚的的注释:
1 //==============================================================service层===================================================================== 2 public class MarketaService extends BaseService{ 3 /** 4 * =====================================远程action层调用此方法发送传单================================================================== 5 */ 6 ///远程调用的参数为: userId=8665&maId=1015&flyerId=38273&userIds=8665,8687,8688&assUserIds= 7 /** 服务层对接控制层 调用数据层 8 * 对接 action中的 远程下发传单 方法 remotePassFlyers () 9 * 下发活动传单 10 * @param userId 当前用户的id 11 * @param maId 当前活动的id 12 * @param flyerId 传单的id 13 * @param userIds 目标用户的id 14 * @param assUserIds --null 15 * @return ProcResult --> 专门用于存放(封装)返回值的工具bean类 16 */ 17 public ProcResult passFlyerToUsers(Integer userId, Integer maId, Integer flyerId, String userIds, String assUserIds){ 18 /**辅助方法清单 方法一*/ 19 List<Map<String, Object>> users = getInvitedUserInfos(maId, flyerId, userIds, assUserIds);//获取用户的信息对应传单id 没有传单id的为null 就是指没有收到过传单 20 //里面都有 --- 用户id 和创建用户组织的id 和 传单id 21 if(null != users && !users.isEmpty()){//非空才进去 (只要userIds有值就行) 22 //声明两个list集合 分别用于存放: 23 List<Integer> hasSharedList = new ArrayList<Integer>();//已经被分享了的成员 24 List<Integer> notSharedList = new ArrayList<Integer>();//还没有被分享的成员 id的集合 25 26 for(int i=0; i<users.size(); i++){ 27 Map<String, Object> user = users.get(i); //每个用户遍历出来 取出来的个体对象都是一个map 28 //map的entity是[列名:值]的方式 29 // String : Object 30 //例{"user_id":8875,"id":9999} 31 Integer inviteUserId = Integer.parseInt(user.get("user_id").toString()) ;//用户的id 32 boolean isShared = user.get("id")!=null;// 传单id 传单id不为空则说明已经有了 就是已经复制了模板 33 if(isShared){ //已经分享了的 用户id ~s 34 hasSharedList.add(inviteUserId); 35 } else { //还没有传单的 用户的id ~s 36 notSharedList.add(inviteUserId);//没有分享的用户的id 37 } 38 } 39 System.out.println("This is in:com.chanjet.mkp.marketa.MarketaService.passFlyerToUsers() ; \n " + 40 "---wunian testing...-----The var is: 已经分享的:没有分享的 , And The value is: " + hasSharedList.size() +"//"+ notSharedList.size() + " \t ---------"); 41 //上面是对 已派发过的和未派发过的进行了 !!!分类!!! 整理到了两个list里面 42 /* ***************************************************************华丽丽的分割线***************************************************/ 43 /* ***************************************************************华丽丽的分割线***************************************************/ 44 /* ***************************************************************华丽丽的分割线***************************************************/ 45 46 //对所有已经派发过的批量修改 47 if(!hasSharedList.isEmpty()){//已经派发的 hasSharedList 数组里面保存的是 已经有的传单的user_id ~s 48 /**辅助方法清单:方法二*/ 49 List<Map<String, Object>> sharedFlyers = getSharedFlyerDetaiInfo(maId, flyerId);// 所有在活动下父传单为模板传单的 子传单的详细信息 询已经共享的传单的user_id 50 Integer[] sharedFlyerArray = new Integer[sharedFlyers.size()];//查看一共有多少个已经派送了的传单 以多少作为长度 声明一个Integer类型的数组 51 for(int i=0; i<sharedFlyers.size(); i++){ //循环遍历出相关的子传单的详细信息来 52 Map<String, Object> flyer = sharedFlyers.get(i); 53 Integer sFlyerId = (Integer) flyer.get("flyer_id");//传单的id 54 sharedFlyerArray[i] = sFlyerId;//这样把现有的传单的id 放在一个数组里面 叫做sharedFlyerArray 55 } 56 /**辅助方法清单:方法三*/ 57 String sharedUserIds = castArrayToString(hasSharedList.toArray());//把已经分享的用户(list集合)弄出来 转换成数组 变成字符串 被分享的用户的id ~s 58 String sharedFlyerIds = castArrayToString(sharedFlyerArray); // 把现有的传单的id的数组转换成一个字符串 59 // 对应的传单的id ~s 60 61 /**辅助方法清单:方法四*/ 62 String updateFlyerSql = getPassFlyerSql(1).replace("{user_ids}", sharedUserIds); //肯定是sql的in用 63 //更新了 已经被分享的用户的id的所有需要更新的相关信息的sql语句 64 // 65 String updateDlyerDetailSql = getPassFlyerSql(3).replace("{flyer_ids}", sharedFlyerIds);// same 66 //更新 已经被分享用户的传单的 详细表(子表)的相关详细信息的sql语句 67 68 Object[] flyerParam = new Object[]{flyerId, maId, flyerId}; 69 Object[] flyerDetailParam = new Object[]{flyerId}; 70 this.getDao().update(updateFlyerSql, flyerParam);//更新 d_flyer表 71 this.getDao().update(updateDlyerDetailSql, flyerDetailParam); // 更新了他的子表 72 } 73 //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 74 //对所有未发送过的批量添加 没有什么传单的 75 if(!notSharedList.isEmpty()){ // 没有发送过的 就是d_flyer 所属人 缺少穿过来的那些 的 76 /**辅助方法清单:方法三*/ 77 String notSharedUserIds = castArrayToString(notSharedList.toArray()); 78 /**辅助方法清单:方法四*/ 79 String addFlyerSql = getPassFlyerSql(5);// 多行插入 d_flyer 语句(复制传单语句) 80 addFlyerSql = addFlyerSql.replace("{parent_id}",flyerId+"").replace("{user_ids}", notSharedUserIds); 81 //填充数据 批量复制传单语句 最后一个parent_id字段要写上模板(父传单的id值 )的id值 82 Object[] addFlyerParams = new Object[]{flyerId,userId}; 83 this.getDao().update(addFlyerSql, addFlyerParams); 84 /**辅助方法清单:方法四*/ 85 List<Map<String, Object>> newAddFlyerList = getSharedFlyerInfoByUserIds(maId, flyerId, castArrayToString(notSharedList.toArray()));//查询出刚刚分发出的传单的 86 Object[] newAddFlyerIds = new Object[newAddFlyerList.size()]; //新增加的传单个数 声明一个数组 87 for(int i=0; i<newAddFlyerList.size(); i++){ // 88 Map<String, Object> newF = newAddFlyerList.get(i); 89 newAddFlyerIds[i] = newF.get("flyer_id"); //新传单的id存放到数组里面 获得传单id 并放到数组中 90 } 91 /**辅助方法清单:方法三*/ 92 String newAddFIds = castArrayToString(newAddFlyerIds); //数组转换成字符串 去掉了空格 93 /**辅助方法清单:方法四*/ 94 String addFlyerDetailSql = getPassFlyerSql(4).replace("{flyer_ids}", newAddFIds);// 插入对应的子表的数据 95 Object[] addFlyerDetailParams = new Object[]{flyerId}; 96 this.getDao().update(addFlyerDetailSql, addFlyerDetailParams);//因为关联到了子表 97 } 98 //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~相对应的数据库的所有表的查询 更新操作结束!!!~~~~~~~~~~~ 99 //获取 所有属于此活动的 父传单id为flyerId的 所有复制版本员工传单 100 // 返回数据字段有: 101 // 用户的id 传单id 用户对应组织id 传单详细信息表的内容 发布后的内容 102 // 邮箱 电话 qq 微信 名称 职位 103 /**辅助方法清单:方法二*/ 104 List<Map<String, Object>> allSharedFlyers = getSharedFlyerDetaiInfo(maId, flyerId);//插入完成后再次重新获取所有相关的传单 105 //重新上传html文件 106 try { 107 /**辅助方法清单:方法六*/ 108 refreshNameCard(allSharedFlyers); //更新传单名片信息(数据库中的content字段的更新) 109 //重新生成静态html页面 110 Integer orgId=0; 111 /**辅助方法清单:方法十一 */ 112 uploadShareFlyerHtmlFile(maId, flyerId, userId, orgId, allSharedFlyers); 113 } catch (Exception e) { 114 e.printStackTrace(); 115 } 116 } 117 return ProcResult.success(); 118 } 119 120 /** 121 * =====================================辅助方法清单:方法一================================================================== 122 * 返回值为 此活动 模板传单是相应传单模板的 未删除 的且user_id 在参数里面的所有满足条件的 用户id 传单id 123 * 即 在此活动被邀请 且有传单的所有人 124 * 传入的userIds作为了左表!!!!! (9876有无对应的fid都会有一条对应 生成 9876 null) 125 */ 126 private List<Map<String, Object>> getInvitedUserInfos(Integer maId, Integer flyerId, String userIds, String assUserIds){ 127 String sql = "select " + 128 " u.user_id as user_id,f.id as id " + //用户id 和创建用户组织的id 和 传单id 129 " from m_user u " + 130 " left join d_flyer f " + 131 " on u.user_id = f.user_id " + 132 " and f.ma_id=? " + // 所属活动限定 133 " and f.parent_id=? " + //这填充的是 传单模板(父传单)的id值 这样就又是一个小型的无线分类 知道了模板的来源 134 " and f.deleted = 0" + // 删除 ? 否限定 135 " where (u.user_id in ({userIds}) ) and del = 0 ";// 用户限定 穿过来的用户 136 /** 137 * select u.user_id as user_id,f.id as id 138 * from m_user u 139 * left join d_flyer f on u.user_id = f.user_id And f.ma_id =? and parent_id=? and f.deleted =0 140 * where (u.user_id in ({userIds}) ) and del =0; 141 由此看出是preparedstatement in 为什么用替换? 我试了一下用? 出错了 142 */ 143 sql = sql.replace("{userIds}",userIds);//参数填充 144 Object[] params = new Object[]{maId, flyerId}; //准备参数 145 System.out.println(sql + "=========>"); 146 System.out.println( 147 "==========\n"+"maId:"+maId +"\n flyerId:"+flyerId +"\n userIds:" + userIds); 148 List<Map<String, Object>> list = this.getDao().query(sql, params, null);// 149 System.out.println("查询列表长度为: "+ list.size() +"==="+list+ " \t ---------"); 150 return list; 151 } 152 //================================================================================================== 153 154 /** 155 * =====================================辅助方法清单:方法二================================================================== 156 */ 157 private List<Map<String, Object>> getSharedFlyerDetaiInfo(Integer maId, Integer parentFlyerId){ 158 List<Map<String, Object>> list = null; 159 String sql = "SELECT t.user_id, t.id AS flyer_id, t.orgid, d.content, d.updated_content," + 160 " m.email, m.mobile, m.phone, m.qq, m.weixin, m.user_name as realname ,mou.position AS job " + 161 " FROM d_flyer t" + 162 " left join d_flyer_detail d on t.id=d.flyer_id" + 163 " left join m_user m on t.user_id = m.user_id"+ 164 " left join m_org_user mou on mou.user_id = m.user_id and mou.dept_id !=0" + 165 " WHERE t.parent_id = ? AND t.ma_id = ? AND t.deleted = 0"; 166 /* 167 SELECT ? FROM d_flyer t 168 left join d_flyer_detail d on t.id=d.flyer_id 169 left join m_user m on t.user_id = m.user_id 170 left join m_org_user mou on mou.user_id = m.user_id and mou.dept_id !=0 171 WHERE t.parent_id = ? AND t.ma_id = ? AND t.deleted = 0 172 */ 173 list = this.getDao().query(sql, new Object[]{parentFlyerId, maId}, null); 174 return list; 175 } 176 /** 177 * =====================================辅助方法清单:方法三================================================================== 178 */ 179 /** 180 * 把数组拼接成以","隔开的字符串 181 * @param array 182 * @return 183 */ 184 private String castArrayToString(Object[] array){ 185 String str = ""; 186 if(null==array){ 187 return str; 188 } 189 switch (array.length){ 190 case 1: 191 str = array[0].toString(); 192 break; 193 default: 194 for(int i=0; i<array.length; i++){ 195 if(i==array.length-1){ 196 str += array[array.length-1].toString(); 197 break; 198 } 199 str += array[i].toString() + ","; 200 } 201 } 202 return str; 203 } 204 205 /** 206 * =====================================辅助方法清单:方法四================================================================== 207 */ 208 private String getPassFlyerSql(Integer type){ 209 String sql = ""; 210 switch (type) { 211 case 1: // updateFlyer 212 sql = "UPDATE d_flyer t1, d_flyer t2 SET t1.title = t2.title," + 213 " t1.status = t2.status," + 214 " t1.create_time = NOW()," + 215 " t1.notice = t2.notice," + 216 " t1.template = t2.template," + 217 " t1.update_time = NOW()," + 218 " t1.description = t2.description," + 219 " t1.beau_sel = t2.beau_sel, " + 220 " t1.refer_count = t2.refer_count," + 221 " t1.refer_id = t2.refer_id," + 222 " t1.isdraw = t2.isdraw," + 223 " t1.iscopy = t2.iscopy," + 224 " t1.is_marketa_enjoy = 0," + 225 " t1.showTitle = t2.showTitle," + 226 " t1.searchKeyword = t2.searchKeyword," + 227 " t1.music = t2.music," + 228 " t1.classify_id = t2.classify_id," + 229 " t1.template_status = t2.template_status," + 230 " t1.option_id = t2.option_id," + 231 " t1.approvedBy = t2.approvedBy," + 232 " t1.deleteReason = t2.deleteReason" + 233 " WHERE t1.user_id IN ({user_ids}) AND t1.parent_id= ? AND t1.ma_id= ? AND t2.id=?"; 234 break; 235 case 2:// insertFlyer 236 sql = "INSERT INTO d_flyer(title,status,create_time,notice,template,user_id,update_time,description,deleted,orgid,hit,notice_days,visitor_count,feedback_count,ishot,version,is_edit,is_add,beau_sel,ma_id,isorg,refer_count,refer_id,isdraw,iscopy,is_marketa_enjoy,showTitle,searchKeyword,music,classify_id,template_status,option_id,approvedBy,deleteReason,parent_id)" + 237 " SELECT t1.title,t1.status,NOW(),t1.notice,t1.template,t2.invite_user_id,NOW(),t1.description,0,t2.adscription_org,0,t1.notice_days,0,0,t1.ishot,2.00,t1.is_edit,t1.is_add,t1.beau_sel,t1.ma_id,t1.isorg,t1.refer_count,t1.refer_id,t1.isdraw,t1.iscopy,0,t1.showTitle,t1.searchKeyword,t1.music,t1.classify_id,t1.template_status,t1.option_id,t1.approvedBy,t1.deleteReason,{parent_id}" + 238 " FROM d_flyer t1," + 239 " (SELECT m.invite_user_id,m.adscription_org FROM m_market_invite_user m WHERE m.invite_user_id IN ({user_ids}) AND m.ma_id=?" + 240 " {add_admin_user}) t2 " + 241 " WHERE t1.id=? AND t1.ma_id=? AND t1.deleted=0 AND t1.user_id=?"; 242 break; 243 case 3:// updateFlyerDetail 244 sql = "UPDATE d_flyer_detail t1,d_flyer_detail t2" + 245 " SET t1.content = t2.content," + 246 " t1.updated_content = t2.updated_content," + 247 " t1.dsc=t2.dsc" + 248 " WHERE t1.flyer_id IN({flyer_ids}) AND t2.flyer_id = ?"; 249 break; 250 case 4:// insertFlyerDetail 251 sql = "INSERT INTO d_flyer_detail(flyer_id, content, updated_content, dsc) " + 252 " SELECT t2.id, t1.content, t1.updated_content, t1.dsc " + 253 " FROM d_flyer_detail t1,(SELECT id ,user_id FROM d_flyer WHERE id IN({flyer_ids})) t2" + 254 " WHERE t1.flyer_id=?"; 255 break; 256 case 5: 257 sql =" INSERT INTO d_flyer(title,status,create_time,notice,template,user_id,update_time,description,deleted,orgid,hit,notice_days,visitor_count,feedback_count,ishot,version,is_edit,is_add,beau_sel,ma_id,isorg,refer_count,refer_id,isdraw,iscopy,is_marketa_enjoy,showTitle,searchKeyword,music,classify_id,template_status,option_id,approvedBy,deleteReason,parent_id) " + 258 " (SELECT t1.title,t1.status,NOW(),t1.notice,t1.template,t2.user_id,NOW(),t1.description,0,t2.org_id,0,t1.notice_days,0,0,t1.ishot,2.00,t1.is_edit,t1.is_add,t1.beau_sel,t1.ma_id,0,t1.refer_count,t1.refer_id,t1.isdraw,t1.iscopy,0,t1.showTitle,t1.searchKeyword,t1.music,t1.classify_id,t1.template_status,t1.option_id,t1.approvedBy,t1.deleteReason, " + 259 " {parent_id} " + 260 " FROM d_flyer t1, " + 261 " (select DISTINCT user_id, org_id from m_org_user where user_id in({user_ids}) and del=0 and dept_id !=0 ) t2 " + 262 " WHERE t1.id=? AND t1.deleted=0 AND t1.user_id=?)"; 263 default: 264 265 } 266 267 return sql; 268 } 269 270 /** 271 * =====================================辅助方法清单:方法五================================================================== 272 */ 273 private List<Map<String, Object>> getSharedFlyerInfoByUserIds(Integer maId, Integer parentId, String userIds){ 274 List<Map<String, Object>> flyers = null; 275 String sql = "SELECT t.id AS flyer_id,t.user_id, t.orgid FROM d_flyer t WHERE t.user_id IN ({user_ids}) AND t.ma_id=? AND t.parent_id=?"; 276 sql = sql.replace("{user_ids}", userIds); 277 Object[] params = new Object[]{maId, parentId}; 278 flyers = this.getDao().query(sql, params, null); 279 return flyers; 280 } 281 /** 282 * =====================================辅助方法清单:方法六================================================================== 283 */ 284 private void refreshNameCard(List<Map<String, Object>> flyerDetails){ 285 if(null==flyerDetails || flyerDetails.isEmpty()){ //如果没有任何相关的表 286 return; 287 } 288 for(int i=0; i<flyerDetails.size(); i++){//获取 289 Map<String, Object> flyer = flyerDetails.get(i); 290 Integer userId = Integer.parseInt(flyer.get("user_id").toString());//获取传单所属用户 291 Integer flyerId = Integer.parseInt(flyer.get("flyer_id").toString());// 相关的传单id 292 String content = flyer.get("content").toString();// 传单设置的内容 293 //String updateContent = flyer.get("updated_content").toString();//传单发布后设置的内容 为空!!!!!!! 294 /**辅助方法清单:方法七*/ 295 Map<String, Object> m = JsonUtil.jsonToMap(content);/**content 格式化后长度过长 会在最下面展示*/ 296 //m 是json解析的总根 ~ 297 List<Map<String,String>> widgets = (List<Map<String, String>>) m.get("widgets"); 298 for(int j=0; j<widgets.size(); j++){ 299 Map<String, String> widget = widgets.get(j); 300 String wdUname = widget.get("wd_u_name"); 301 if("bio".equals(wdUname)){// 找到名片 复制了简单的我们需要的json数据 302 /**辅助方法清单:方法九*/ 303 Map<String, String> newCard = replaceCardInfo(widget, userId, flyerId, flyer);//生成用户 新的名片 304 widgets.set(j, newCard);//应该就一个 - -! 设置回去 305 } 306 } 307 m.put("widgets", widgets);//用新的替换了之前的 308 /**辅助方法清单:方法八*/ 309 String newContent = JsonUtil.objToJson(m);//重新转换成json字符串 310 //访问数据库,开始修改detail表的数据 311 /**辅助方法清单:方法十*/ 312 updateDetailContent(flyerId, newContent);// 更改详细信息表里面的content数据 313 314 } 315 /**辅助方法六调用结束 返回passFlyerToUsers() 方法的调用处(倒数7行左右)*/ 316 } 317 /** 318 * =====================================辅助方法清单:方法九================================================================== 319 */ 320 private Map<String, String> replaceCardInfo(Map<String, String> widget, Integer userId, Integer flyerId, Map<String, Object> userInfo){ 321 /*"wd_u_name":"bio","company":"贵州限责任公司","job":"职位","realname":"李小姐", 322 "qq":"1234578","tell":"085-1234567","sina":"http://webo.com/baogs/ome?vr=5","email":"255123456@qq.com", 323 "web":"http://www.e.com/","address":"贵阳号",*/ 324 Map<String, String> card = new HashMap<String, String>(); 325 Set<String> keySet = widget.keySet();// company job realname ==的key集合 326 System.out.println("----------Start replace mobile card info[userId="+userId+",flyerId="+flyerId+"]---------------"); 327 for(String key : keySet){ 328 Object userObj = null; 329 if("realname".equals(key)){ 330 userObj = userInfo.get("realname"); 331 } else if("qq".equals(key)){ 332 userObj = userInfo.get("qq"); 333 } else if("tell".equals(key)){ 334 userObj = userInfo.get("mobile"); 335 } else if("email".equals(key)){ 336 userObj = userInfo.get("email"); 337 } else if("phone".equals(key)){ 338 userObj = userInfo.get("phone"); 339 } else if("weixin".equals(key)){ 340 userObj = userInfo.get("weixin"); 341 } else if("job".equals(key)){ 342 userObj = userInfo.get("job") 343 } 344 System.out.println(key+" = "+String.valueOf(userObj)); 345 String value = ""; 346 if(null==userObj){ 347 value = String.valueOf(widget.get(key)); //没有的话 用原来的 348 } else { 349 value = String.valueOf(userObj); //有的话 设置进去 350 } 351 card.put(key, value); //放到map里面 352 } 353 System.out.println("----------End replace mobile card info---------------"); 354 return card; //返回回去 355 } 356 357 /** 358 * =====================================辅助方法清单:方法十================================================================== 359 */ 360 private void updateDetailContent(Integer flyerId, String content){ 361 String sql = "update d_flyer_detail set content = ? where flyer_id = ?"; 362 Object[] params = new Object[]{content, flyerId}; 363 this.getDao().update(sql,params); 364 } 365 366 /** 367 * =====================================辅助方法清单:方法十一================================================================== 368 */ 369 private void uploadShareFlyerHtmlFile(Integer maId, Integer flyerId, Integer userId, Integer orgId, List<Map<String, Object>> members) throws Exception{ 370 //本地html保存路径 371 //获取ServletAction上下文对象,getServletContext()获取Servlet上下文对象(项目地址),getRealPath("file")获取file的!绝对路径! 加上/加上flyer 372 String localHtmlRoot = ServletActionContext.getServletContext().getRealPath("file")+ File.separatorChar +"flyer"; 373 //本地共享传单路径 374 String shareHtmlPath = localHtmlRoot + File.separator + "html" + File.separator + userId+ File.separator; 375 //共享传单文件名称 376 String shareFileName = flyerId + ".html"; //被共享的传单的地址拼接完成 377 378 //上传路径 379 final String realPath = ServletActionContext.getServletContext().getRealPath("");//获取项目的绝对路径 380 File shareFile = new File(shareHtmlPath + shareFileName); //被共享传单的File对象 381 if(!shareFile.exists()){ // 如果不存在 则抛出异常 382 throw new RuntimeException("共享失败,共享传单不存在!"); 383 } 384 for(int i=0; i<members.size(); i++){ //为每一个用户复制被共享的一份传单 放到自己的/user_id/flyer_id下面 385 String uploadFilePath = ""; 386 String tmpFileName = ""; 387 Map<String, Object> mem = members.get(i); 388 Integer inviteUserId = (Integer) mem.get("user_id"); 389 Integer inviteUserOrg = (Integer) mem.get("orgid"); 390 Integer inviteUserFlyerId = (Integer) mem.get("flyer_id"); 391 tmpFileName = inviteUserFlyerId + ".html"; 392 String tmpFilePath = localHtmlRoot + File.separator + "html" +File.separator + inviteUserId+ File.separator; 393 String tmpFileFullName = tmpFilePath + tmpFileName; 394 //文件名 和文件路径判断 先有路径才能创建文件 395 File tmpFile = new File(tmpFileFullName); 396 File tmpDirs = new File(tmpFilePath); 397 if(!tmpDirs.exists()){//文件路径不存在的情况下 创建文件路径 398 tmpDirs.mkdirs(); 399 } 400 if(tmpFile.exists()){ //如果先前文件已经存在(未删除)则进行删除 401 tmpFile.delete(); 402 } 403 //复制共享传单 404 /**辅助方法清单:方法十二*/ 405 FileUtil.copyFile(shareFile, tmpFile);//建立流 复制传单 406 if(!tmpFile.exists()){ //再次判断是否复制了传单 407 throw new RuntimeException("共享失败,复制传单"+tmpFileName+"失败!"); 408 } 409 // 更改了成员传单的 名片 等详细信息 410 /**辅助方法清单:方法十三*/ 411 reloadCardHtml(tmpFile, mem); 412 //上传成员传单html文件 413 uploadFilePath = localHtmlRoot.replace(realPath, "") + File.separator + "html" + File.separator + inviteUserOrg + File.separator + tmpFileName; 414 try { 415 FileService.UpYunWriteFile(uploadFilePath, tmpFile, false); 416 } catch (Exception e) { 417 e.printStackTrace(); 418 } 419 }//for循环右括号 420 } 421 /** 422 * =====================================辅助方法清单:方法十三================================================================== 423 */ 424 /** 使用了 Jsoup解析了 html文档树*/ 425 private void reloadCardHtml(File html, Map<String, Object> userInfo){ 426 // FileWriter fw = null; 427 OutputStreamWriter writer = null; 428 boolean flag = false; 429 try { 430 Integer inviteUserId = (Integer) userInfo.get("user_id"); 431 Integer inviteUserOrg = (Integer) userInfo.get("orgid"); 432 Integer inviteUserFlyerId = (Integer) userInfo.get("flyer_id"); 433 System.out.println("==========Start replace html bio[userId="+inviteUserId+",flyerId="+inviteUserFlyerId+"]========================"); 434 //http://www.jb51.net/article/43485.htm Jsoup的使用 435 //<span class="realname" data-key="realname">“构享家”团队</span> 436 Document doc = Jsoup.parse(html, "UTF-8"); 437 438 Elements companies = doc.getElementsByAttributeValue("data-key", "company"); 439 Elements jobs = doc.getElementsByAttributeValue("data-key","job"); 440 Elements realname = doc.getElementsByAttributeValue("data-key","realname"); 441 Elements qq = doc.getElementsByAttributeValue("data-key","qq"); 442 Elements wechats = doc.getElementsByAttributeValue("data-key","weixin"); 443 Elements mobile = doc.getElementsByAttributeValue("data-key","tell"); 444 Elements phone = doc.getElementsByAttributeValue("data-key","mobile"); 445 Elements weibos = doc.getElementsByAttributeValue("data-key","sina"); 446 Elements emails = doc.getElementsByAttributeValue("data-key","email"); 447 Elements webs = doc.getElementsByAttributeValue("data-key","web"); 448 Elements address = doc.getElementsByAttributeValue("data-key","address"); 449 450 String userCompany = String.valueOf(userInfo.get("company")==null?"":userInfo.get("company")); 451 String userJob = String.valueOf(userInfo.get("job") == null ? "" : userInfo.get("job")); 452 String userQQ = String.valueOf(userInfo.get("qq")==null?"":userInfo.get("qq")); 453 String userMobile = String.valueOf(userInfo.get("mobile")==null?"":userInfo.get("mobile")); 454 String userPhone = String.valueOf(userInfo.get("phone")==null?"":userInfo.get("phone")); 455 String userRealname = String.valueOf(userInfo.get("realname")==null?"":userInfo.get("realname")); 456 String userEmail = String.valueOf(userInfo.get("email")==null?"":userInfo.get("email")); 457 String userWeixin = String.valueOf(userInfo.get("weixin")==null?"":userInfo.get("weixin")); 458 if(null!=realname && realname.size()>0){ 459 Element e = realname.get(0); 460 e.text(userRealname);//给元素内容设置数据 461 System.out.println("realname="+userRealname); 462 flag = true; 463 } 464 465 if(null!=phone && phone.size()>0){ 466 Element e = phone.get(0); 467 e.text(userPhone); 468 System.out.println("phone="+userPhone); 469 flag = true; 470 } 471 472 if(null!=mobile && mobile.size()>0){ 473 Element e = mobile.get(0); 474 e.text(userMobile); 475 System.out.println("mobile="+userMobile); 476 flag = true; 477 } 478 479 if(null!=qq && qq.size()>0){ 480 Element e = qq.get(0); 481 e.text(userQQ); 482 System.out.println("qq="+userQQ); 483 flag = true; 484 } 485 486 if(null!=emails && emails.size()>0){ 487 Element e = emails.get(0); 488 e.text(userEmail); 489 System.out.println("email="+userEmail); 490 flag = true; 491 } 492 493 if(null!=wechats && wechats.size()>0){ 494 Element e = wechats.get(0); 495 e.text(userWeixin); 496 System.out.println("weixin="+userWeixin); 497 flag = true; 498 } 499 System.out.println("==========End replace html bio[userId="+inviteUserId+",flyerId="+inviteUserFlyerId+"]========================"); 500 if(flag){ //如果有更改 in √ 501 String htmlContent = doc.html();//doc对象的内容已经变了? 现在实际上是jsoup一次性解析了html文件 502 // 生成 Document对象 一直在内存中 我们只是操作了对象 503 // String aaa = doc.data(); 测试代码 504 // String bbb = doc.html(); 505 // String ccc = doc.val(); 506 // fw = new FileWriter(html); 507 // fw.write(htmlContent); 508 writer = new OutputStreamWriter(new FileOutputStream(html), "UTF-8"); //开启了到更新文件的 输出流 509 writer.write(htmlContent); //把文档输出到里面去 内存到硬盘的写入 510 writer.close(); 511 } 512 } catch (Exception e) { 513 e.printStackTrace(); 514 } 515 } 516 }
两个工具类(不完全):
1 //--------------------------JsonUtil类---------使用了google提供的 gson类 解析json-a------------------------------------------------------ 2 import com.google.gson.Gson; 3 import com.google.gson.GsonBuilder; 4 import com.google.gson.reflect.TypeToken; 5 6 public class JsonUtil{ 7 private static Gson gson; 8 /** JSON字符串转化为HashMap对象*/ 9 /** 10 * =====================================辅助方法清单:方法七================================================================== 11 */ 12 public static Map<String, Object> jsonToMap(String string) { 13 Map<String, Object> obj = null; 14 try { //s使用 google的gson包 15 obj = JsonUtil.getGson().fromJson(string, new TypeToken<Map<String, Object>>(){}.getType()); 16 }catch (Exception e) { 17 e.printStackTrace(); 18 } 19 return obj; 20 } 21 /** 22 * =====================================辅助方法清单:方法八================================================================== 23 */ 24 public static String objToJson(Object data) { 25 try { 26 return JsonUtil.getGson().toJson(data); 27 } catch (Exception e) { 28 e.printStackTrace(); 29 } 30 return ""; 31 } 32 } 33 34 //--------------------------FileUtil类---------使用了google提供的 gson类 解析json-a------------------------------------------------------ 35 public class FileUtil { 36 /** 37 * =====================================辅助方法清单:方法十二================================================================== 38 */ 39 public static boolean copyFile(File sourceFile, File targetFile) { 40 try { 41 BufferedInputStream inBuff = null; 42 BufferedOutputStream outBuff = null; 43 try { 44 // 新建文件输入流并对它进行缓冲 45 inBuff = new BufferedInputStream(new FileInputStream(sourceFile)); 46 47 // 新建文件输出流并对它进行缓冲 48 outBuff = new BufferedOutputStream(new FileOutputStream(targetFile)); 49 // 缓冲数组 50 byte[] b = new byte[1024 * 100]; 51 int len; 52 while ((len = inBuff.read(b)) != -1) { 53 outBuff.write(b, 0, len); 54 } 55 // 刷新此缓冲的输出流 56 outBuff.flush(); 57 } finally { 58 // 关闭流 59 if (inBuff != null) 60 inBuff.close(); 61 if (outBuff != null) 62 outBuff.close(); 63 } 64 } catch (Throwable e) { 65 e.printStackTrace(); 66 return false; 67 } 68 return true; 69 } 70 71 }
/**~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ content内容展~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
1 /* 2 {"widgets": 3 [ 4 {"wd_u_name":"title", 5 "value":"找工作去劳联,电信送手机", 6 "href":"http://baidu.com", 7 "color":"#000000", 8 "colors":[ 9 "#000000", 10 "#ff7418", 11 "#f14444", 12 "#a25c18", 13 "#b62450"], 14 "fontSize":"25px", 15 "fontFamily":"Microsoft YaHei,Microsoft YaHei", 16 "marginTop":"10px", 17 "textAlign":"center"}, 18 19 {"wd_u_name":"map", 20 "value":"北京市海淀区上地信息路28号科实大厦", 21 "marginTop":"5px"}, 22 {"wd_u_name":"media", 23 "value":"http://player.youku.com/player.php/sid/XNjY0NDI4MjE2/v.swf", 24 "color":"#000000", 25 "marginTop":"5px"}, 26 27 {"wd_u_name":"feedback", //反馈表 28 "btnLabel":"抢购", 29 "items": 30 [ 31 {"name":"name", 32 "allowBlank":"false", 33 "label":"姓 名 "}, 34 {"name":"weibo", 35 "label":"微博", 36 "maxlength":64, 37 "input":"string", 38 "width":"100", 39 "allowBlank":"true"}, 40 {"name":"phone", 41 "label":"电话", 42 "maxlength":32, 43 "sortable":true, 44 "type":"string", 45 "width":"100", 46 "allowBlank":"false"}, 47 {"name":"mobile", 48 "label":"手 机", 49 "allowBlank":"false"}, 50 {"name":"appellation", 51 "label":"称呼", 52 "maxlength":32, 53 "input":"string", 54 "width":"100", 55 "allowBlank":"true"}, 56 {"name":"email", 57 "label":"邮 箱 ", 58 "allowBlank":"true"}, 59 {"name":"position", 60 "label":"经纪人", 61 "maxlength":32, 62 "input":"string", 63 "width":"100", 64 "allowBlank":"false"}, 65 {"name":"note", 66 "label":"备注", 67 "maxlength":200, 68 "type":"string", 69 "input":"textarea", 70 "width":"150"}, 71 {"name":"company", 72 "label":"公司", 73 "maxlength":64, 74 "sortable":true, 75 "input":"string", 76 "width":"150"}, 77 {"name":"qq", 78 "label":"QQ", 79 "allowBlank":"true"}, 80 {"name":"address", 81 "label":"地址", 82 "maxlength":128, 83 "type":"string", 84 "width":"100", 85 "allowBlank":"true"} 86 ], 87 "fontSize":"13px", 88 "marginTop":"10px"}, 89 90 -------------//名片表-----------------------主要是他更新 因为传单的名片应该是发到A的账号中去的时候就变成A的相关的信息 当然也可以编辑 91 {"wd_u_name":"bio", 92 "company":"贵州限责任公司", 93 "job":"职位", 94 "realname":"李小姐", 95 "qq":"", 96 "tell":"08516550", 97 "sina":"http://webo.com/baogs/ome?vr=5", 98 "email":"2556890@qq.com", 99 "web":"http://www.e.com/", 100 "address":"贵阳号", 101 "marginTop":"0px", 102 "imgsrc":"/file/flyer/tdcode/png201431118063553.png", 103 "status":2, 104 "fontSize":"18px", 105 "color":"rgb(0, 106 0, 107 0)", 108 "colors":["#000000", 109 "#7f8c2f", 110 "#85622a", 111 "#b66c2a", 112 "#a92e2e"]} 113 ], 114 "meta":{"color":"#fedac2", 115 "bgImg":"/flyer/images/flyerimg/flyer_bg_6.jpg", 116 "shadingImg":"/flyer/images/bgimg/06.png"}} 117 */
OK 结束 再次声明 此博客为个人笔记 并不是独立demo