Java web上传文件源码

文件名:UploadExample.jsp

<%@ page contentType="text/html;charset=gb2312"%>
<html>
<title><%= application.getServerInfo() %></title>
<body>
上传文件程序应用示例
<form action="doUpload.jsp" method="post" enctype="multipart/form-data">
<%-- 类型enctype用multipart/form-data,这样可以把文件中的数据作为流式数据上传,不管是什么文件类型,均可上传。--%>
请选择要上传的文件<input type="file" name="upfile" size="50">
<input type="submit" value="提交">
</form>
</body>
</html>

文件名:doUpload.jsp

<%@ page contentType="text/html; charset=GBK" %>
<%@ page import="java.io.*"%>
<%@ page import="java.util.*"%>
<%@ page import="javax.servlet.*"%>
<%@ page import="javax.servlet.http.*"%>
<html><head><title>upFile</title></head>
<body bgcolor="#ffffff">
<%
//定义上载文件的最大字节
int MAX_SIZE = 102400 * 102400;
// 创建根路径的保存变量
String rootPath;
//声明文件读入类
DataInputStream in = null;
FileOutputStream fileOut = null;
//取得客户端的网络地址
String remoteAddr = request.getRemoteAddr();
//获得服务器的名字
String serverName = request.getServerName();

//取得互联网程序的绝对地址
String realPath = request.getRealPath(serverName);
realPath = realPath.substring(0,realPath.lastIndexOf("\"));
//创建文件的保存目录
rootPath = realPath + "\\upload\";
//取得客户端上传的数据类型
String contentType = request.getContentType();
try{
if(contentType.indexOf("multipart/form-data") >= 0){
//读入上传的数据
in = new DataInputStream(request.getInputStream());
int formDataLength = request.getContentLength();
if(formDataLength > MAX_SIZE){
out.println("<P>上传的文件字节数不可以超过" + MAX_SIZE + "</p>");
return;
}
//保存上传文件的数据
byte dataBytes[] = new byte[formDataLength];
int byteRead = 0;
int totalBytesRead = 0;
//上传的数据保存在byte数组
while(totalBytesRead < formDataLength){
byteRead = in.read(dataBytes,totalBytesRead,formDataLength);
totalBytesRead += byteRead;
}
//根据byte数组创建字符串
String file = new String(dataBytes);
//out.println(file);
//取得上传的数据的文件名
String saveFile = file.substring(file.indexOf("filename="") + 10);
saveFile = saveFile.substring(0,saveFile.indexOf("\n"));
saveFile = saveFile.substring(saveFile.lastIndexOf("\") + 1,saveFile.indexOf("""));
int lastIndex = contentType.lastIndexOf("=");
//取得数据的分隔字符串
String boundary = contentType.substring(lastIndex + 1,contentType.length());
//创建保存路径的文件名
String fileName = rootPath + saveFile;
//out.print(fileName);
int pos;
pos = file.indexOf("filename="");
pos = file.indexOf("\n",pos) + 1;
pos = file.indexOf("\n",pos) + 1;
pos = file.indexOf("\n",pos) + 1;
int boundaryLocation = file.indexOf(boundary,pos) - 4;
//out.println(boundaryLocation);
//取得文件数据的开始的位置
int startPos = ((file.substring(0,pos)).getBytes()).length;
//out.println(startPos);
//取得文件数据的结束的位置
int endPos = ((file.substring(0,boundaryLocation)).getBytes()).length;
//out.println(endPos);
//检查上载文件是否存在
File checkFile = new File(fileName);
if(checkFile.exists()){
out.println("<p>" + saveFile + "文件已经存在.</p>");
}
//检查上载文件的目录是否存在
File fileDir = new File(rootPath);
if(!fileDir.exists()){
fileDir.mkdirs();
}
//创建文件的写出类
fileOut = new FileOutputStream(fileName);
//保存文件的数据
fileOut.write(dataBytes,startPos,(endPos - startPos));
fileOut.close();
out.println(saveFile + "文件成功上载.</p>");
}else{
String content = request.getContentType();
out.println("<p>上传的数据类型不是multipart/form-data</p>");
}
}catch(Exception ex){
throw new ServletException(ex.getMessage());
}
%>
</body>
</html>

SpringMVC使用AJAX跨域请求

Spring MVC 4.2 增加 CORS 支持

正如大家所知,出于安全考虑,浏览器会限制脚本中发起的跨站请求。比如,使用 XMLHttpRequest 对象发起 HTTP 请求就必须遵守同源策略(same-origin policy)。 具体而言,Web 应用程序能且只能使用 XMLHttpRequest 对象向其加载的源域名发起 HTTP 请求,而不能向任何其它域名发起请求。为了能开发出更强大、更丰富、更安全的Web应用程序,开发人员渴望着在不丢失安全的前提下,Web 应用技术能越来越强大、越来越丰富。比如,可以使用 XMLHttpRequest 发起跨站 HTTP 请求。(这段描述跨域不准确,跨域并非浏览器限制了发起跨站请求,而是跨站请求可以正常发起,但是返回结果被浏览器拦截了。最好的例子是crsf跨站攻击原理,请求是发送到了后端服务器无论是否跨域!

@CrossOrigin 这个就是重要的注解

/**
  * 通过ID 查询所有
  * @param user_id
  * @return
  */
 @CrossOrigin("http://127.0.0.1:8020")
 @RequestMapping("/findUserById")
 @ResponseBody
 public Msg findUserById(Integer user_id,HttpServletRequest request, HttpServletResponse response){
     User user=userService.findUserById(user_id);
        return Msg.success().add("user",user);
 }

还有就是设置mvc XML 填入这段 再测试阶段可以这样使用。

<!-- 设置跨域不限制-->
<mvc:cors>
   <mvc:mapping path="/**" />
   </mvc:cors>

这还有更加细微的方式

<mvc:cors>
    <mvc:mapping path="/api/**"
        allowed-origins="http://domain1.com, http://domain2.com"
        allowed-methods="GET, PUT"
        allowed-headers="header1, header2, header3"
        exposed-headers="header1, header2" allow-credentials="false"
        max-age="123" />
    <mvc:mapping path="/resources/**"
        allowed-origins="http://domain1.com" />
</mvc:cors>

Java精确运算(BigDecimal)

如果需要精确计算,非要用String来够造BigDecimal不可!
实现方案

import java.math.BigDecimal;
public class MathExtend
{
  //默认除法运算精度  
  private static final int DEFAULT_DIV_SCALE = 10;
 /** 
  * 提供精确的加法运算。 
  * @param v1 
  * @param v2 
  * @return 两个参数的和 
  */
  public static double add(double v1, double v2)
  {
      BigDecimal b1 = new BigDecimal(Double.toString(v1));
      BigDecimal b2 = new BigDecimal(Double.toString(v2));
      return b1.add(b2).doubleValue();
  }
  /** 
   * 提供精确的加法运算 
   * @param v1   
   * @param v2 
   * @return 两个参数数学加和,以字符串格式返回 
   */
  public static String add(String v1, String v2)
  {
      BigDecimal b1 = new BigDecimal(v1);
      BigDecimal b2 = new BigDecimal(v2);
      return b1.add(b2).toString();
  }
 /** 
  * 提供精确的减法运算。 
  * @param v1 
  * @param v2 
  * @return 两个参数的差 
  */
  public static double subtract(double v1, double v2)
  {
      BigDecimal b1 = new BigDecimal(Double.toString(v1));
      BigDecimal b2 = new BigDecimal(Double.toString(v2));
      return b1.subtract(b2).doubleValue();
  }
  /** 
   * 提供精确的减法运算 
   * @param v1 
   * @param v2 
   * @return 两个参数数学差,以字符串格式返回 
   */
  public static String subtract(String v1, String v2)
  {
      BigDecimal b1 = new BigDecimal(v1);
      BigDecimal b2 = new BigDecimal(v2);
      return b1.subtract(b2).toString();
  }
  /** 
  * 提供精确的乘法运算。 
  * @param v1 
  * @param v2 
  * @return 两个参数的积 
  */
  public static double multiply(double v1, double v2)
  {
      BigDecimal b1 = new BigDecimal(Double.toString(v1));
      BigDecimal b2 = new BigDecimal(Double.toString(v2));
      return b1.multiply(b2).doubleValue();
  }
  /** 
   * 提供精确的乘法运算 
   * @param v1 
   * @param v2 
   * @return 两个参数的数学积,以字符串格式返回 
   */
  public static String multiply(String v1, String v2)
  {
      BigDecimal b1 = new BigDecimal(v1);
      BigDecimal b2 = new BigDecimal(v2);
      return b1.multiply(b2).toString();
  }
  /** 
  * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到 
  * 小数点以后10位,以后的数字四舍五入,舍入模式采用ROUND_HALF_EVEN 
  * @param v1 
  * @param v2 
  * @return 两个参数的商 
  */
  public static double divide(double v1, double v2)
  {
      return divide(v1, v2, DEFAULT_DIV_SCALE);
  }
  /** 
   * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指 
   * 定精度,以后的数字四舍五入。舍入模式采用ROUND_HALF_EVEN 
   * @param v1 
   * @param v2 
   * @param scale 表示需要精确到小数点以后几位。 
   * @return 两个参数的商 
   */
  public static double divide(double v1,double v2, int scale)
  {
      return divide(v1, v2, scale, BigDecimal.ROUND_HALF_EVEN);
  }
  /** 
   * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指 
   * 定精度,以后的数字四舍五入。舍入模式采用用户指定舍入模式 
   * @param v1 
   * @param v2 
   * @param scale 表示需要精确到小数点以后几位 
   * @param round_mode 表示用户指定的舍入模式 
   * @return 两个参数的商 
   */
  public static double divide(double v1,double v2,int scale, int round_mode){
          if(scale < 0)
          {
              throw new IllegalArgumentException("The scale must be a positive integer or zero");
          }
          BigDecimal b1 = new BigDecimal(Double.toString(v1));
          BigDecimal b2 = new BigDecimal(Double.toString(v2));
          return b1.divide(b2, scale, round_mode).doubleValue();
  }
  /** 
   * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到 
   * 小数点以后10位,以后的数字四舍五入,舍入模式采用ROUND_HALF_EVEN 
   * @param v1 
   * @param v2 
   * @return 两个参数的商,以字符串格式返回 
   */
  public static String divide(String v1, String v2)
  {
      return divide(v1, v2, DEFAULT_DIV_SCALE);
  }
  /** 
   * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指 
   * 定精度,以后的数字四舍五入。舍入模式采用ROUND_HALF_EVEN 
   * @param v1 
   * @param v2 
   * @param scale 表示需要精确到小数点以后几位 
   * @return 两个参数的商,以字符串格式返回 
   */
  public static String divide(String v1, String v2, int scale)
  {
      return divide(v1, v2, DEFAULT_DIV_SCALE, BigDecimal.ROUND_HALF_EVEN);
  }
  /** 
   * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指 
   * 定精度,以后的数字四舍五入。舍入模式采用用户指定舍入模式 
   * @param v1 
   * @param v2 
   * @param scale 表示需要精确到小数点以后几位 
   * @param round_mode 表示用户指定的舍入模式 
   * @return 两个参数的商,以字符串格式返回 
   */
  public static String divide(String v1, String v2, int scale, int round_mode)
  {
      if(scale < 0)
      {
          throw new IllegalArgumentException("The scale must be a positive integer or zero");
      }
      BigDecimal b1 = new BigDecimal(v1);
      BigDecimal b2 = new BigDecimal(v2);
      return b1.divide(b2, scale, round_mode).toString();
  }
  /** 
   * 提供精确的小数位四舍五入处理,舍入模式采用ROUND_HALF_EVEN 
   * @param v 需要四舍五入的数字 
   * @param scale 小数点后保留几位 
   * @return 四舍五入后的结果 
   */
  public static double round(double v,int scale)
  {
      return round(v, scale, BigDecimal.ROUND_HALF_EVEN);
  }
  /** 
   * 提供精确的小数位四舍五入处理 
   * @param v 需要四舍五入的数字 
   * @param scale 小数点后保留几位 
   * @param round_mode 指定的舍入模式 
   * @return 四舍五入后的结果 
   */
  public static double round(double v, int scale, int round_mode)
  {
     if(scale<0)
     {
         throw new IllegalArgumentException("The scale must be a positive integer or zero");
     }
     BigDecimal b = new BigDecimal(Double.toString(v));
     return b.setScale(scale, round_mode).doubleValue();
  }
  /** 
   * 提供精确的小数位四舍五入处理,舍入模式采用ROUND_HALF_EVEN 
   * @param v 需要四舍五入的数字 
   * @param scale 小数点后保留几位 
   * @return 四舍五入后的结果,以字符串格式返回 
   */
  public static String round(String v, int scale)
  {
    return round(v, scale, BigDecimal.ROUND_HALF_EVEN);
  }
  /** 
   * 提供精确的小数位四舍五入处理 
   * @param v 需要四舍五入的数字 
   * @param scale 小数点后保留几位 
   * @param round_mode 指定的舍入模式 
   * @return 四舍五入后的结果,以字符串格式返回 
   */
  public static String round(String v, int scale, int round_mode)
  {
     if(scale<0)
     {
         throw new IllegalArgumentException("The scale must be a positive integer or zero");
     }
     BigDecimal b = new BigDecimal(v);
     return b.setScale(scale, round_mode).toString();
  }
}

实现方案

现在我们已经知道怎么解决这个问题了,原则上是使用BigDecimal(String)构造器,我们建议,在商业应用开发中,涉及金额等浮点数计算的数据,全部定义为String,数据库中可定义为字符型字段,在需要使用这些数据进行运算的时候,
使用BigDecimal(String)构造BigDecimal对象进行运算,保证数据的精确计算。