Blog

nosql入门概述-(1)-产生原因和发展简史

nosql发展简史

一、单机MySQL的美好年代

在90年代,一个网站的访问量一般都不大,用单个数据库完全可以轻松应付。
在那个时候,更多的都是静态网页,动态交互类型的网站不多。

上述架构下,我们来看看数据存储的瓶颈是什么?
1.数据量的总大小 一个机器放不下时
2.数据的索引(B+ Tree)一个机器的内存放不下时
3.访问量(读写混合)一个实例不能承受
如果满足了上述1 or 3个,进化……

二、 Memcached(缓存)+MySQL+垂直拆分

后来,随着访问量的上升,几乎大部分使用MySQL架构的网站在数据库上都开始出现了性能问题,web程序不再仅仅专注在功能上,同时也在追求性能。程序员们开始大量的使用缓存技术来缓解数据库的压力,优化数据库的结构和索引。开始比较流行的是通过文件缓存来缓解数据库压力,但是当访问量继续增大的时候,多台web机器通过文件缓存不能共享,大量的小文件缓存也带了了比较高的IO压力。在这个时候, Memcached就自然的成为一个非常时尚的技术产品。

 Memcached作为一个独立的分布式的缓存服务器,为多个web服务器提供了一个共享的高性能缓存服务,在Memcached服务器上,又发展了根据hash算法来进行多台Memcached缓存服务的扩展,然后又出现了一致性hash来解决增加或减少缓存服务器导致重新hash带来的大量缓存失效的弊端

三、Mysql主从读写分离

由于数据库的写入压力增加,Memcached只能缓解数据库的读取压力。读写集中在一个数据库上让数据库不堪重负,大部分网站开始使用主从复制技术来达到读写分离,以提高读写性能和读库的可扩展性。Mysqlmaster-slave模式成为这个时候的网站标配了。

四、分表分库+水平拆分+mysql集群

Memcached的高速缓存,MySQL的主从复制,读写分离的基础之上,这时MySQL主库的写压力开始出现瓶颈,而数据量的持续猛增,由于MyISAM使用表锁,在高并发下会出现严重的锁问题,大量的高并发MySQL应用开始使用InnoDB引擎代替MyISAM

 同时,开始流行使用分表分库来缓解写压力和数据增长的扩展问题。这个时候,分表分库成了一个热门技术,是面试的热门问题也是业界讨论的热门技术问题。也就在这个时候,MySQL推出了还不太稳定的表分区,这也给技术实力一般的公司带来了希望。虽然MySQL推出了MySQL Cluster集群,但性能也不能很好满足互联网的要求,只是在高可靠性上提供了非常大的保证。

五、今天是什么样子??

1、3V+3高:

大数据时代的3V  互联网需求的3高
1、海量Volume 1、高并发 【同时处理大数据】
2、多样Variety
【数据格式多种多样,如声音文字】
2、高可扩
【纵向扩展:单机版性能升级】
【横向扩展:多台电脑集群,负载均衡】
3、实时Velocity  3、高性能

2、分布式和集群:

分布式系统(distributed system)
由多台计算机和通信的软件组件通过计算机网络连接(本地网络或广域网)组成。分布式系统是建立在网络之上的软件系统。正是因为软件的特性,所以分布式系统具有高度的内聚性和透明性。因此,网络和分布式系统之间的区别更多的在于高层软件(特别是操作系统),而不是硬件。分布式系统可以应用在在不同的平台上如:Pc、工作站、局域网和广域网上等。

简单来讲:
(1)分布式:不同的多台服务器上面部署不同的服务模块(工程),他们之间通过Rpc/Rmi之间通信和调用,对外提供服务和组内协作。

(2)集群:不同的多台服务器上面部署相同的服务模块,通过分布式调度软件进行统一的调度,对外提供服务和访问。

2、MySQL的扩展性瓶颈

MySQL数据库也经常存储一些大文本字段,导致数据库表非常的大,在做数据库恢复的时候就导致非常的慢,不容易快速恢复数据库。比如10004KB大小的文本就接近40GB的大小,如果能把这些数据从MySQL省去,MySQL将变得非常的小。关系数据库很强大,但是它并不能很好的应付所有的应用场景。MySQL的扩展性差(需要复杂的技术来实现),大数据下IO压力大,表结构更改困难,正是当前使用MySQL的开发人员面临的问题。

3、为什么用NoSQL

今天我们可以通过第三方平台(如:Google,Facebook等)可以很容易的访问和抓取数据。用户的个人信息,社交网络,地理位置,用户生成的数据和用户操作日志已经成倍的增加。 我们如果要对这些用户数据进行挖掘,那SQL数据库已经不适合这些应用了 , NoSQL数据库的发展也却能很好的处理这些大的数据。

SSM小项目-(11)-项目总结与正确的Maven项目架构

编码管理方便统一:

标签中的类名:取-
标签中的id:取_

一、项目总结

 

二、Maven项目打包

Eclipse 中 右键项目 run as ==>  maven install ==>可以打包成 war 项目包。
但是因为之前设置的目录有问题,所以打包时,说找不到 web.xml 配置文件。
正确的maven项目目录是:

三、发布项目到Tomcat

四、WAR包的目录结构

首先:maven的web工程中:src/main下面有三个文件夹

1、项目根地址/src/main/java   里面包含的就是一些类文件

2、项目根地址/src/main/resource  里面包含的就是一些资源文件

3、项目根地址/src/main/webapp 里面包含的就是一些前端代码和资源

其中1和2 最终会整合到webapp/WEB-INF/classes目录下。并且项目依赖的jar包,最终会整合到webapp/WEB-INF/lib目录下。至于webapp下面的META-INF目录,里面更新的文件不是很重要。原本就有一个MANIFEST.MF(清单文件,内容如下)

Manifest-Version: 1.0
Class-Path:

后来maven打包后,又添加了一个maven文件夹,里面是公司名文件夹–>项目名文件夹–>【pom.properties,pom.xml】

pom.properties里面的具体内容如下:

#Generated by Maven
#Wed Jul 5 13:12:07 CST 2018
version=0.0.1-SNAPSHOT
groupId=yyy(也就是公司名)
artifactId=xxx(也就是项目名)

 

SSM小项目-(10)-删除雇员模块及本项目总结

注意在目前了解到的所有的js操作中:
//只有下面的方法,查找有  点+类名 ,比如【.edit-btn】  这个算标签内的自定义属性
//其他地方,查找类名,都不需要添加 . 号 【原因就是  .  代表查找同类名的所有元素 】
$(document).on("click",".edit-btn",function(){      });

一、单个删除

1、前端代码

// delete-btn 这个是在一开始展示表格信息的时候,就已经添加了。
$(document).on("click",".delete-btn",function(){
			
	//弹出是否确认删除对话框
	//alert($(this).parents("tr").find("td:eq(1)").text());
         var empName = $(this).parents("tr").find("td:eq(1)").text()
	 var empId = $(this).attr("delete-id");
         if(confirm("确认删除【"+empName+"】")){
				
		$.ajax({
				url:"${APP_PATH}/emp/"+empId,
				type:"delete",
				success:function(result){
					alert(result.msg);
					to_page(g_currentPage);
				}
			});
	}
		    
		  
});

2、后端代码

EmployeeController.java

@ResponseBody
@RequestMapping(value="/emp/{id}",method=RequestMethod.DELETE)
public Msg deleteByEmpId(@PathVariable("id") int id){
	employeeService.deleteEmp(id);
	return Msg.success();
}

EmployeeService.java

/**
 * 根据id 删除用户
 * @param id
 */
public void deleteEmp(int id) {
	// TODO Auto-generated method stub
	employeeMapper.deleteByPrimaryKey(id);	
}

二、批量删除

1、前端代码

(1)添加checkbox  勾选功能

<!-- 显示表格数据 -->
<div class="row">
	<div class="col-md-12">
		<table class="table table-hover" id="emps_table">
			<thead>
				<tr>
					<th> <input type="checkbox" id="check_all" /></th>
				        <th>#</th>
					<th>empName</th>
					<th>gender</th>
					<th>email</th>
					<th>deptName</th>
					<th>操作</th>
				</tr>
			</thead>
			<tbody>
				
			</tbody>
		</table>
	</div>
</div>
function build_emps_table(result) {
	//清空table表格,如果不请空,下一次请求时,数据会在原来的基础上,不断添加。
	$("#emps_table tbody").empty();
			
	var emps = result.extend.pageInfo.list;
	$.each(emps, function(index, item) {
		//alert(item.empName);
				
		//构建单元格
		var checkBoxTd = $("<td><input type='checkbox' class='check-item' /></td>");
		var empIdTd = $("<td></td>").append(item.empId);
		var empNameTd = $("<td></td>").append(item.empName);
		var genderTd = $("<td></td>").append(item.gender=='M'?"男":"女");
		var emailTd = $("<td></td>").append(item.email);
		var deptNameTd = $("<td></td>").append(item.department.deptName);				
		/**
		<button class="btn btn-primary btn-sm">
			<i class="icon-edit icon-large"></i> 编辑
		</button>
		<button class="btn btn-danger btn-sm">
			<i class="icon-trash icon-large"></i> 删除
		</button>
		**/
	var editBtn = $("<button></button>").addClass("btn btn-primary btn-sm edit-btn")
			.append($("<i></i>").addClass("icon-edit icon-large")).append(" 编辑");
        //为编辑按钮添加一个自定义的属性,来表示当前员工id
        editBtn.attr("edit-id",item.empId);
                
        var delBtn =  $("<button></button>").addClass("btn btn-danger btn-sm delete-btn")
				.append($("<i></i>").addClass("icon-trash icon-large")).append(" 删除");
        //为删除按钮添加一个自定义的属性来表示当前删除的员工id
        delBtn.attr("delete-id",item.empId);
        var btnTd = $("<td></td>").append(editBtn).append(" ").append(delBtn);

                //append方法执行完成以后还是返回原来的元素
	$("<tr></tr>")
                .append(checkBoxTd)
                .append(empIdTd)
		.append(empNameTd)
		.append(genderTd)
		.append(emailTd)
		.append(deptNameTd)
		//.append(editBtn)
		//.append(delBtn)
		.append(btnTd)
		.appendTo("#emps_table tbody");
				
	});
}

(2)添加勾选逻辑

//完成全选、全不选功能
$("#check_all").click(function(){
			
	/***
	//获取id为p的标签的属性值
        $('#p').attr('class')
        //如果存在就是修改,如果不存在就是添加
        $('#p').attr('class','red')
        // 删除某个属性
        $('#p').removeAttr('name')
        //添加class名称
        $('#p').addClass('blue')
        //删除class名称
        $('#p').removeClass('yellow')
        //有class就删除 没有就添加
        $('#p').toggleClass('asdf')
        //val()获取输入框内容
        var s = $('input[name="user"]').val()
	***/
			
	// attr获取checked是undefined;
	// alert($(this).attr("checked"));  弹出的是 undefined
	// 用prop修改和读取dom原生属性的值;用attr获取自定义属性的值;这样就不容易出错			
	// alert($(this).prop("checked")); 
        // 可以正常弹出结果,也就是说:$(this).prop("checked") 返回是否选中的  true or false
	// 设置 是否 checked  $(this).prop("checked",true/false)
			
	//设置全选和全不选
	$(".check-item").prop("checked",$(this).prop("checked"));
});
		
//check_item 触发回调函数
$(document).on("click",".check-item",function(){
	//判断当前选择中的元素是否5个
	var flag = $(".check-item:checked").length==$(".check-item").length;
	$("#check_all").prop("checked",flag);
});
		
//点击全部删除,就是批量删除
$("#emp_delete_all_btn").click(function(){			
			
	var empNames ="";
	//组织员工id字符串
	var delete_ids_str="";
			
	$.each($(".check-item:checked"),function(){
		empNames = empNames+$(this).parents("tr").find("td:eq(2)").text() +",";
	        delete_ids_str += $(this).parents("tr").find("td:eq(1)").text() +"-";
	});
			
	//去除empNames多余的,
	empNames = empNames.substring(0,empNames.length-1);
	//去除删除的id的多余的-
	delete_ids_str = delete_ids_str.substring(0,delete_ids_str.length-1);
			
	if(confirm("确认删除【"+empNames+"】?")){
	//发送ajax请求
	 $.ajax({
			url:"${APP_PATH}/emp/"+delete_ids_str,
			type:"delete",
			success:function(result){
				alert(result.msg);
				to_page(g_currentPage);
			}
		});
	}
});

(3)后端处理逻辑

EmployeeController.java

 /**
  * 单个、批量 删除二合一,
  * 批量删除 1-2-3
  * 单个删除 1
  * 
  * @param ids
  * @return
  */
@ResponseBody
@RequestMapping(value="/emp/{ids}",method=RequestMethod.DELETE)
public Msg deleteByEmpId(@PathVariable("ids") String ids){
	//批量删除
	if(ids.contains("-")){
	       List<Integer> delete_id = new ArrayList<>() ;
               String[] str_ids = ids.split("-");
               for (String str :str_ids) {
            	   delete_id.add(Integer.parseInt(str));
		}
		employeeService.deleteBatch(delete_id);

	}else{
		int id = Integer.parseInt(ids);
		employeeService.deleteEmp(id);
	}
	return Msg.success();
}

EmployeeService.java

/**
 * 
 * @param ids
 */
public void deleteBatch(List<Integer> ids) {
	
        // TODO Auto-generated method stub
	EmployeeExample example = new EmployeeExample();
	Criteria criteria = example.createCriteria();
        //delete  from  xxx where emp_id in (1,2,3)
	criteria.andEmpIdIn(ids);
        employeeMapper.deleteByExample(example);
		
}

 

springmvc原理-POJO对象是如何赋值的及PUT请求如何赋值

SpringMVC封装POJO对象的时候,会调用request.getParamter(“属性名”) 并赋值给 POJO中每个属性。


在Tomcat 服务器中:

如果是POST请求:
1、会将请求体中的数据,封装一个map。
2、request.getParameter(“empName”)就会从这个map中取值。
3、SpringMVC封装POJO对象的时候,会调用request.getParamter(“属性名”) 并赋值给 POJO中每个属性。

如果是PUT请求:
Tomcat一看是PUT不会封装请求体中的数据为map,只有POST形式的请求才封装请求体为map
所以请求体中的数据,request.getParameter(“empName”)拿不到。

下面是Tomcat源代码的处理逻辑,只有post请求,才能封装请求体为map对象。

 * org.apache.catalina.connector.Request(类名)--parseParameters(方法名) (3111行);
	 * 
	 * protected String parseBodyMethods = "POST";
	 * if( !getConnector().isParseBodyMethod(getMethod()) ) {
                success = true;
                return;
            }
	 *

 put传递数据解决方案

我们要能支持直接发送PUT之类的请求还要封装请求体中的数据
1、配置上HttpPutFormContentFilter;
2、他的作用;将请求体中的数据解析包装成一个map。
3、request被重新包装,request.getParameter()被重写,就会从自己封装的map中取数据

在web.xml中配置过滤器,

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

大致的原理就是:

HttpServletRequest 本来就是 一个接口,在之前的接口实现类中,我们又添加了HttpPutFormContentFilter对象,并重写了 之前接口实现类的getParameter方法,该方法内部调用的是 HttpPutFormContentFilter 里面的方法。

mybatis扩展-查询结果排序的两种方式

1.使用PageHelper排序(Pagehelper的版本需在5.1.2及以上)

<!– https://mvnrepository.com/artifact/com.github.pagehelper/pagehelper –>
<dependency>
<groupId>com.github.pagehelper</groupId>
<artifactId>pagehelper</artifactId>
<version>5.1.2</version>
</dependency>

PageHelper.startPage(pageNum , pageSize);PageHelper.orderBy(“A B”);
其中A为排序依据的字段名,B为排序规律,desc为降序,asc为升序

或者一步到位

String orderBy=”字段名 排序规律”;

PageHelper.startPage(pageNum, pageSize, orderBy);

2.使用Mybatis排序

XXXExample example = new XXXExample();

example.setOrderByClause(“字段名1 ASC/DESC,字段名2 ASC/DESC,…”);
———————

来自:https://blog.csdn.net/kalnon/article/details/79559627

bootstrap踩坑-简单复制粘贴修改出bug花时间

在bootstrap中 ,新建了一个模态框表格:

知道显示模态框需要:

$("#empUpdateModal").modal({
	backdrop:"static"
});

但是忘记了,点击提交按钮时,需要自己关闭模态框:

//点击更新,更新员工信息
$("#emp_update_done_btn").click(function(){

//1、员工修改成功,需要关闭模态框。 
//下面这句话,之前一直忘记添加了,所以一直不知道为什么提交了,却不自动关闭模态框。思考并找了很长时间,对框架执行不熟悉
$("#empUpdateModal").modal("hide");

}

新建的模态框表格如下:

	<!-- 员工修改的模态框 -->
<div class="modal fade" id="empUpdateModal" tabindex="-1" role="dialog" aria-labelledby="exampleModalCenterTitle" aria-hidden="true">
  <div class="modal-dialog modal-dialog-centered" role="document">
    <div class="modal-content">
      <div class="modal-header">
        <h5 class="modal-title" id="exampleModalLongTitle">员工修改</h5>
        <button type="button" class="close" data-dismiss="modal" aria-label="Close">
          <span aria-hidden="true">&times;</span>
        </button>
      </div>
      <div class="modal-body">

<form>
  <div class="form-group row">
    <label class="col-sm-3 col-form-label">empName</label>    
    <div class="col-sm-9">
      <input type="text" readonly class="form-control-plaintext" id="empName_update_input_disable" name="empName" value="none">
    </div>
    
  </div>
  
  <div class="form-group row">
    <label class="col-sm-3 col-form-label">email</label>
    <div class="col-sm-9">
      <input type="text"  class="form-control" id="email_update_input" name="email" placeholder="[email protected]">
          <div class="invalid-feedback"> 邮箱输入错误。</div>
    </div>

  </div>
  
  <div class="form-group row">
    <label  class="col-sm-3 col-form-label">gender</label>  
    <div class="col-sm-9">
<div class="form-check form-check-inline">
  <input class="form-check-input" type="radio" name="gender" id="gender1_update_input" checked value="M">
  <label class="form-check-label" >男</label>
</div>
<div class="form-check form-check-inline">
  <input class="form-check-input" type="radio" name="gender" id="gender2_update_input" value="F">
  <label class="form-check-label" >女</label>
</div>
    </div>  
  </div>
  
    <div class="form-group row">
    <label  class="col-sm-3 col-form-label">departName</label>  
    <div class="col-sm-6">
    <!-- 部门提交部门id即可 -->
    <select class="custom-select my-1 mr-sm-2" name="dId" id="dept_update_select">
    </select>
    </div>
    </div>
</form>

      </div>
      <div class="modal-footer">
        <button type="button" class="btn btn-secondary" data-dismiss="modal">关闭</button>
        <button type="button" class="btn btn-primary" id="emp_update_done_btn">更新</button>
      </div>
    </div>
  </div>
</div>

 

javascript原理-多事件响应时的执行流程

今天在表单中:

对一个输入框添加了邮箱格式校验:

//校验 email框
$("#email_update_input").change(function(){
	//校验表单
	validate_form_ele("#email_update_input",g_email_reg,g_email_valid,g_email_invalid_format)

console.log("变化方法---邮箱校验");
				
	});

同时在提交表单时:添加了事件

//点击更新,更新员工信息
$("#emp_update_done_btn").click(function(){
	
console.log("提交更新方法---邮箱校验");
		
	//验证邮箱是否合法
	if(!validate_form_ele("#email_update_input",g_email_reg,g_email_valid,g_email_invalid_format))
         return false;
			
	//2、发送ajax请求保存更新的员工数据
	$.ajax({
          ...
        });

}

 

经过测试,修改了邮箱表单并立刻提交时,会先后触发两个方法:
 变化方法—邮箱校验
 提交更新方法—邮箱校验

但是两个方法都打上断点并调试的时候,很奇怪,只调用了 变化方法—邮箱校验,没有调用   提交更新方法—邮箱校验  。所以网页没有进行提交更新。

SSM小项目-(9)-修改雇员模块

注意在目前了解到的所有的js操作中:
//只有下面的方法,查找有  点+类名 ,比如【.edit-btn】   这个算标签内的自定义属性。
//其他地方,查找类名,都不需要添加 . 号  【原因就是  .  代表查找同类名的所有元素 】
$(document).on("click",".edit-btn",function(){      });

一、思路概述:

0、新建了一个  全局变量 当前页码信息,当保存提交时,显示的 还是当前页码。

1、get deps方法中   ajax 的异步和同步问题  获取部门请求,会影响更新操作前 获取员工信息,并赋值给 select 的dep 部门。两个都是 异步操作,容易出现提前赋值 select ,后在出现 select 部门选项

2、//2、发送ajax请求保存更新的员工数据   添加 _method
			$.ajax({
				url:"${APP_PATH}/emp/"+$(this).attr("edit-id"),
				type:"post",
				data:$("#empUpdateModal form").serialize()+"&_method=put",
				success:function(result){
				
这样做会来到,web.xml 中的过滤器

<!-- 使用Rest风格的URI,将页面普通的post请求转为指定的delete或者put请求 -->
	<filter>
		<filter-name>HiddenHttpMethodFilter</filter-name>
		<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
	</filter>
	<filter-mapping>
		<filter-name>HiddenHttpMethodFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>



3、	@RequestMapping(value="/emp/{id}”,method=RequestMethod.PUT)
        改成
	@RequestMapping(value="/emp/{empId}",method=RequestMethod.PUT)

4、如果改成 ajax 的 put 方法,则不需要在data中添加 "&_method=put" ,但是  tomcat 无法封装 请求体,导致 报错。
* 原因:
	 * Tomcat:
	 * 		1、将请求体中的数据,封装一个map。
	 * 		2、request.getParameter("empName")就会从这个map中取值。
	 * 		3、SpringMVC封装POJO对象的时候。
	 * 				会把POJO中每个属性的值,request.getParamter("email");
	 * AJAX发送PUT请求引发的血案:
	 * 		PUT请求,请求体中的数据,request.getParameter("empName")拿不到
	 * 		Tomcat一看是PUT不会封装请求体中的数据为map,只有POST形式的请求才封装请求体为map
         *  查看下面的 tomcat 源码 解析  ,发现只有post 
	 * org.apache.catalina.connector.Request(类名)—parseParameters() (方法名)( 行号 3111);
	 * 
	 * protected String parseBodyMethods = "POST";
	 * if( !getConnector().isParseBodyMethod(getMethod()) ) {
                success = true;
                return;
            }
	 * 
	 * 
	 * 解决方案;
	 * 我们要能支持直接发送PUT之类的请求还要封装请求体中的数据
	 * 1、配置上HttpPutFormContentFilter;
	 * 2、他的作用;将请求体中的数据解析包装成一个map。
	 * 3、request被重新包装,request.getParameter()被重写,就会从自己封装的map中取数据
	 * 员工更新方法

二、添加模态框:

1、直接复制并修改之前的员工添加模态框:

		
<!-- 员工修改的模态框 -->
<div class="modal fade" id="empUpdateModal" tabindex="-1" role="dialog" aria-labelledby="exampleModalCenterTitle" aria-hidden="true">
  <div class="modal-dialog modal-dialog-centered" role="document">
    <div class="modal-content">
      <div class="modal-header">
        <h5 class="modal-title" id="exampleModalLongTitle">员工修改</h5>
        <button type="button" class="close" data-dismiss="modal" aria-label="Close">
          <span aria-hidden="true">&times;</span>
        </button>
      </div>
      <div class="modal-body">

<form>
  <div class="form-group row">
    <label class="col-sm-3 col-form-label">empName</label>    
    <div class="col-sm-9">
      <input type="text" readonly class="form-control-plaintext" id="empName_update_input_disable" name="empName" value="none">
    </div>
    
  </div>
  
  <div class="form-group row">
    <label class="col-sm-3 col-form-label">email</label>
    <div class="col-sm-9">
      <input type="text"  class="form-control" id="email_update_input" name="email" placeholder="[email protected]">
          <div class="invalid-feedback"> 邮箱输入错误。</div>
    </div>

  </div>
  
  <div class="form-group row">
    <label  class="col-sm-3 col-form-label">gender</label>  
    <div class="col-sm-9">
<div class="form-check form-check-inline">
  <input class="form-check-input" type="radio" name="gender" id="gender1_update_input" checked value="M">
  <label class="form-check-label" >男</label>
</div>
<div class="form-check form-check-inline">
  <input class="form-check-input" type="radio" name="gender" id="gender2_update_input" value="F">
  <label class="form-check-label" >女</label>
</div>
    </div>  
  </div>
  
    <div class="form-group row">
    <label  class="col-sm-3 col-form-label">departName</label>  
    <div class="col-sm-6">
    <!-- 部门提交部门id即可 -->
    <select class="custom-select my-1 mr-sm-2" name="dId" id="dept_update_select">
    </select>
    </div>
    </div>
</form>

      </div>
      <div class="modal-footer">
        <button type="button" class="btn btn-secondary" data-dismiss="modal">关闭</button>
        <button type="button" class="btn btn-primary" id="emp_update_done_btn">更新</button>
      </div>
    </div>
  </div>
</div>
		

2、添加点击出现模态框事件

注意绑定的时间,因为在生成button后,才能完成绑定。所以绑定的方法不是普通方法。[ edit-btn  这个伪类]

/*** 
1、我们是按钮创建之前就绑定了click,所以绑定不上。
$(".edit-btn").click(function(){
	alert("edit");
});
****/
//1)、可以在创建按钮的时候绑定。   
//2)、绑定点击  live() 方法,这个方法就算是后来添加的元素,也能绑定方法
/***
$(".edit-btn").live(function(){
	alert("edit");
});
***/
//但是新版jquery没有live方法,使用on方法进行替代
/*** 下面的写法 不对
$(".edit-btn").on("click",function(){
	alert("edit");
});
***/
// edit-btn 这个是在一开始展示表格信息的时候,就已经添加了。
//注意 这里是 .edit-btn   算标签内的自定义属性,只是这里使用时 有个点号
$(document).on("click",".edit-btn",function(){
			
	//0、清空表单样式和内容
	reset_form("#empUpdateModal form");
			
	//1、查出部门信息,并显示部门列表
	getDepts("#empUpdateModal select");
	//2、查询员工信息,并显示员工信息  // 坑爹 一开始写成了 attr(edit-id) 
	 getEmp($(this).attr("edit-id"));
			
	//3、把员工的id传递给模态框的更新按钮,为了发送ajax时传递id
	$("#emp_update_done_btn").attr("edit-id",$(this).attr("edit-id"));
			
	//alert("edit");
	$("#empUpdateModal").modal({
		backdrop:"static"
	});
			
			
});

3、将添加雇员信息用到的 getDepts() 方法,进行了抽取

async:false,   这里用到了同步请求方法,如果是异步方法,可能会出现第四步设置设置部门信息时,可能部门信息还没有获取到,就已经开始设置哪个部门是员工现在默认的了。

//1、查出部门信息,并显示部门列表
function getDepts(ele){
	$.ajax({
		url:"${APP_PATH}/depts",
		type:"GET",
		async:false, 
		success:function(result){
		     console.log(result);
	            //显示部门信息,在下拉列表中
	            // $("#dept_add_select") 换一种找法
	            //$("#empAddModal select")
	                
	            //清除之前留下的 option 标签
	             $(ele).empty();
	             $.each(result.extend.depts,function(){
	                     var optionEle = $("<option></option>").attr("value",this.deptId).append(this.deptName);
	                    optionEle.appendTo(ele);
	             });
		}
	});
}

4、显示员工信息方法

(1)前端请求代码:

function getEmp(id){
			
	$.ajax({
		url:"${APP_PATH}/emp/"+id,
		type:"get",
		success:function(result){
			console.log(result);
			var empData = result.extend.emp;
			$("#empName_update_input_disable").val(empData.empName);
			$("#email_update_input").val(empData.email);
			$("#empUpdateModal input[name=gender]").val([empData.gender]);
			$("#empUpdateModal select").val([empData.dId]);
		}
				
	})
}

(2)后端处理代码:

EmployeeController.java

/**
 * 根据id查询员工
 * @param id
 * @return
 */
@RequestMapping(value="/emp/{id}",method=RequestMethod.GET)
@ResponseBody
public Msg getEmp(@PathVariable("id")Integer id){
		
	Employee employee = employeeService.getEmp(id);
	return Msg.success().add("emp", employee);
}

EmployeeService.java

/**
 * 按照员工Id 查询员工
 * @param id
 * @return
 */
public Employee getEmp(Integer id) {
       // TODO Auto-generated method stub
       Employee employee = employeeMapper.selectByPrimaryKey(id);
       return employee;
}

三、保存员工修改信息

1、添加保存按钮点击事件

之前对前端框架不熟悉,忘记添加这句话了$("#empUpdateModal").modal("hide"); 
导致点击提交事件后,一直没有关闭模态框,我以为会自动关闭模态框,原因找了很久,后来才发现。

//点击更新,更新员工信息
$("#emp_update_done_btn").click(function(){
			
	console.log("提交更新方法---邮箱校验");		

	//验证邮箱是否合法
	if(!validate_form_ele("#email_update_input",g_email_reg,g_email_valid,g_email_invalid_format))
        return false;

			
	//2、发送ajax请求保存更新的员工数据
	$.ajax({
		url:"${APP_PATH}/emp/"+$(this).attr("edit-id"),
		type:"post",
		data:$("#empUpdateModal form").serialize()+"&_method=put",
		success:function(result){
			//alert(result);
			//console.log(result);
			if(result.code=100){
				//1、员工修改成功,需要关闭模态框。
				$("#empUpdateModal").modal("hide");
				//2、来到最后一页,显示刚才的数据;发送ajax请求,显示最后一页数据即可(用总记录数请求,保证是请求足够大,mybatis 分页插件 已经在mybatis-config.xml中配置了数据合法性校验,只会返回最后一页数据)
			        to_page(g_currentPage);
			}else{
				//后端校验  显示失败信息
				if(undefined!=result.extend.errorFileds.email){
		                 //显示邮箱错误信息
		    		show_validate_msg("#email_update_input","error",result.extend.errorFileds.email);
		                }
		        }
					
	      }//success 方法结束
							
			  
	});
});

 

特别注意:
1、to_page(g_currentPage); g_currentPage 是一个全局变量,我在显示分页信息时,将其赋了值。

2、保存时,ajax 发送的是post请求,然后用参数 data:$("#empUpdateModal form").serialize()+"&_method=put", 利用 web.xml的

<!-- 4、使用Rest风格的URI,将页面普通的post请求转为指定的delete或者put请求 -->
<filter>
	<filter-name>HiddenHttpMethodFilter</filter-name>
	<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
	<filter-name>HiddenHttpMethodFilter</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>

2、后端处理

EmployeeController.java

@ResponseBody
@RequestMapping(value="/emp/{empId}",method=RequestMethod.PUT)
public Msg saveEmp(Employee employee){
		
	System.out.println("将要更新的员工数据:"+employee);
	employeeService.updateEmp(employee);
	return Msg.success();
		
}

特别注意:value="/emp/{empId}"   ,不是 value="/emp/{id}"   因为empId要赋值给 employee 中的empId属性,前端的post 请求中,不包括empId。

前端put请求举例如下:
empName=adafaf&email=123123%4013.com&gender=F&dId=1&_method=put

EmployeeService.java

public void saveEmp(Employee employee) {
	// 下面这个方法 是全部插入 ,包括 连自增的id 都是
	// employeeMapper.insert(employee);
	// 下面这个方法是 有选择的插入,自增Id 被忽略 不会插入。
	employeeMapper.insertSelective(employee);
}

四、保存员工修改信息改良方法

1、前端直接用ajax的put请求:

$.ajax({			  
	url:"${APP_PATH}/emp/"+$(this).attr("edit-id"),
	type:"PUT",
	data:$("#empUpdateModal form").serialize(),
	success:function(result){
		//alert(result.msg);
		//1、关闭对话框
		$("#empUpdateModal").modal("hide");
		//2、回到本页面
		to_page(currentPage);
	}
});

2、后端解决方法

/**
 * 如果直接发送ajax=PUT形式的请求
 * 封装的数据
 * Employee 
 * [empId=1014, empName=null, gender=null, email=null, dId=null]
 * 
 * 问题:
 * 请求体中有数据;
 * 但是Employee对象封装不上;
 * update tbl_emp  where emp_id = 1014;
 * 
 * 原因:
 * Tomcat:
 * 		1、将请求体中的数据,封装一个map。
 * 		2、request.getParameter("empName")就会从这个map中取值。
 * 		3、SpringMVC封装POJO对象的时候。
 * 				会把POJO中每个属性的值,request.getParamter("email");
 * AJAX发送PUT请求引发的血案:
 * 		PUT请求,请求体中的数据,request.getParameter("empName")拿不到
 * 		Tomcat一看是PUT不会封装请求体中的数据为map,只有POST形式的请求才封装请求体为map
 * Tomcat 源码,查看 put 的处理逻辑
 * org.apache.catalina.connector.Request【类名】--parseParameters()【方法名】 (行号:3111);
 * 
 * protected String parseBodyMethods = "POST";
 * if( !getConnector().isParseBodyMethod(getMethod()) ) {
           success = true;
           return;
      }
 * 
 * 
 * 解决方案;
 * 我们要能支持直接发送PUT之类的请求还要封装请求体中的数据
 * 1、配置上HttpPutFormContentFilter;
 * 2、他的作用;将请求体中的数据解析包装成一个map。
 * 3、request被重新包装,request.getParameter()被重写,就会从自己封装的map中取数据
 * 员工更新方法
 * @param employee
 * @return
 */
	
@ResponseBody
@RequestMapping(value="/emp/{empId}",method=RequestMethod.PUT)
public Msg saveEmp(Employee employee,HttpServletRequest request){
	System.out.println("请求体中的值:"+request.getParameter("gender"));
	System.out.println("将要更新的员工数据:"+employee);
	employeeService.updateEmp(employee);
	return Msg.success()	;
}

注意一下:需要在 web.xml中添加

<!-- 5、直接支持put请求的拦截器,即在put请求时将请求体数据封装成map,并让springmvc能够通过request.getParameter(属性名) 来获取数据 -->
<filter>
	<filter-name>HttpPutFormContentFilter</filter-name>
		<filter-class>org.springframework.web.filter.HttpPutFormContentFilter</filter-class>
	</filter>
<filter-mapping>
	<filter-name>HttpPutFormContentFilter</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>

 

 

 

 

 

mybatis踩坑-自生成mapper映射文件的MBG的路径问题

Mybatis Generator代码不报错,但是没有生成文件

比如:之前是 targetProject=”.\src”  适用于windows,在当前目录的src创建文件。
targetProject=”./src”  适用于 linux ,在当前目录的 src 创建文件。
【Windows 无法识别  ./     同理 linux 也无法识别  .\】

<javaClientGenerator type="XMLMAPPER"
	targetPackage="com.mybatis.mapper"
	targetProject=".\src">
	<!-- enableSubPackages:是否让schema作为包的后缀 -->
	<property name="enableSubPackages" value="false" />
</javaClientGenerator>

MBG 配置目录的时候 targetProject=“./src”  需要知道 是windows环境还是mac环境。主要是 目录设置问题。windows环境:.\    mac环境:./ 

当然目录可以去掉前面./或.\  直接开始写文件夹就行了,这样子 mac和windows都可以运行了。