Java学习笔记——并发编程(一)

写在前面
      建议学习Java并发编程前,先学习一下jvm,这样可以理解的更清楚些。

一、共享带来的问题

1.上下文切换——分析

Java体现
      两个线程对初始值为0的静态变量一个做自增,一个做自减,各做5000次,结果是0么?

static int counter = 0;
public static void main(String[] args) throws InterruptedException{
   
	Thread t1 = new Thread(()->{
   
		for(int i=0;i<5000;i++){
   
			counter++;
		}
	},"t1");
	
	Thread t2 = new Thread(()->{
   
		for(int i=0;i<5000;i++){
   
			counter--;
		}
	},"t2");
	
	t1.start();
	t2.start();
	t1.join();
	t2.join();
	log.debug("{}",counter);
}

问题分析
      例如对于i++而言(i为静态变量),实际会产生如下的JVM字节码指令:

getstatic   i  //获取静态变量i的值
iconst_1       //准备常量1
iadd           //自增
putstatic   i  //将修改后的值存入静态变量i

      而对于i–也是类似:

getstatic   i  //获取静态变量i的值
iconst_1       //准备常量1
isub           //自减
putstatic   i  //将修改后的值存入静态变量i
  • 1为-1的情况
    在这里插入图片描述
  • 为1的情况
    在这里插入图片描述

2.临界区与竞态条件

临界区Critical Section

  • 一个程序运行多个线程本身是没有问题的
  • 问题出在多个线程访问共享资源
    • 多个线程读共享资源其实也没有问题
    • 在多个线程对共享资源读写操作时发生指令交错,就会出现问题
  • 一段代码块内如果存在对共享资源的多线程读写操作,称这段代码块为临界区

例如,下面代码中的临界区

static int counter = 0;

static void increment()
//临界区
{
   
	counter++;
}
static void decrement()
//临界区
{
   
	counter--;
}

竞态条件Race Condition
      多个线程在临界区内执行,由于代码的执行序列不同而导致结果无法预测,称之为发生了竞态条件。

二、synchronized解决方案

为了避免临界区的竞态条件发生,有多种手段可以达到目的。

  • 阻塞式的解决方案:synchronized,Lock
  • 非阻塞式的解决方案:原子变量

      本节先讲解使用synchronized来解决,其余方法将在后续讲到。加了synchronized的Java代码如下:

static int counter = 0;
static Object lock = new Object();
public static void main(String[] args) throws InterruptedException{
   
	Thread t1 = new Thread(()->{
   
		for(int i=0;i<5000;i++){
   
			synchronized(lock){
   
				counter++;
			}
		}
	},"t1");
	
	Thread t2 = new Thread(()->{
   
		for(int i=0;i<5000;i++){
   
			synchronized(lock){
   
				counter--;
			}
		}
	},"t2");
	
	t1.start();
	t2.start();
	t1.join();
	t2.join();
	log.debug("{}",counter);
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
文件上传是Web开发中常见的功能之一,Java中也提供了多种方式来实现文件上传。其中,一种常用的方式是通过Apache的commons-fileupload组件来实现文件上传。 以下是实现文件上传的步骤: 1.在pom.xml文件中添加以下依赖: ```xml <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.3.3</version> </dependency> ``` 2.在前端页面中添加文件上传表单: ```html <form method="post" enctype="multipart/form-data" action="upload"> <input type="file" name="file"> <input type="submit" value="Upload"> </form> ``` 3.在后台Java代码中处理上传文件: ```java // 创建一个DiskFileItemFactory对象,用于解析上传的文件 DiskFileItemFactory factory = new DiskFileItemFactory(); // 设置缓冲区大小,如果上传的文件大于缓冲区大小,则先将文件保存到临时文件中,再进行处理 factory.setSizeThreshold(1024 * 1024); // 创建一个ServletFileUpload对象,用于解析上传的文件 ServletFileUpload upload = new ServletFileUpload(factory); // 设置上传文件的大小限制,这里设置为10MB upload.setFileSizeMax(10 * 1024 * 1024); // 解析上传的文件,得到一个FileItem的List集合 List<FileItem> items = upload.parseRequest(request); // 遍历FileItem的List集合,处理上传的文件 for (FileItem item : items) { // 判断当前FileItem是否为上传的文件 if (!item.isFormField()) { // 获取上传文件的文件名 String fileName = item.getName(); // 创建一个File对象,用于保存上传的文件 File file = new File("D:/uploads/" + fileName); // 将上传的文件保存到指定的目录中 item.write(file); } } ``` 以上代码中,首先创建了一个DiskFileItemFactory对象,用于解析上传的文件。然后设置了缓冲区大小和上传文件的大小限制。接着创建一个ServletFileUpload对象,用于解析上传的文件。最后遍历FileItem的List集合,判断当前FileItem是否为上传的文件,如果是,则获取文件名,创建一个File对象,将上传的文件保存到指定的目录中。 4.文件上传完成后,可以给用户一个提示信息,例如: ```java response.getWriter().write("File uploaded successfully!"); ``` 以上就是使用Apache的commons-fileupload组件实现文件上传的步骤。需要注意的是,文件上传可能会带来安全隐患,因此在处理上传的文件时,需要进行严格的校验和过滤。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值