线程问题<1>

实现一个容器,提供两个方法,add,size
写两个线程,线程1添加10个元素到容器中,线程2实现监控元素的个数,当个数到5个时,线程2给出提示并结束


1.第一种方法是使用while一直监听容器的数量变化,个数到达5就退出

缺点: list添加volatile之后,第二个线程能够接到通知,但是,线程的死循环很浪费cpu

public class Test1 {
	
	private volatile List list = new ArrayList();//线程可见
	public void add(Object o) {list.add(o);}
	public int size() {return list.size();}
	
	public static void main(String[] args) {
		final Test1 t = new Test1();
		new Thread() {
			public void run() {
				for (int i = 0; i < 10; i++) {
					t.add(i);
					System.out.println("线程2增加了" + i);
					try {
						Thread.sleep(1000);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			}
		}.start();
		
		new Thread() {
			public void run() {
				System.out.println("线程2开始");
				while(true) {
					if(t.size() == 5) {
						System.out.println("线程2退出");
						break;
					}
				}
			}
		}.start();
	}
}


2.第二种方法是使用等待通知机制,这里使用wait和notify做到,wait会释放锁,而notify不会释放锁

第一个线程判断个数不到5就wait 该线程停止执行并释放锁,第二个线程得以执行 把容器个数增加到5就调用notify唤醒线程1,这时候第二个线程要wait释放锁,让第一个线程执行,第一个线程执行结束前,要调用notify唤醒第二个线程,让第二个线程得以继续执行

public class Test2 {
	private volatile List list = new ArrayList();
	public void add(Object o) {
		list.add(o);
	}
	public int size() {
		return list.size();
	}
	public static void main(String[] args) {
		final Test2 t = new Test2();
		final Object object = new Object();
		new Thread() {
			public void run() {
				System.out.println("线程2开始");
				if(t.size() != 5) {
					synchronized (object) {
						try {
							object.wait();//个数不到5 调用wait()方法,释放锁
							object.notify();
						} catch (InterruptedException e) {
							e.printStackTrace();
						}
					}
				}
				System.out.println("线程2结束");
			}
		}.start();

		
		new Thread() {
			public void run() {
				System.out.println("线程1开始");
				synchronized (object) {
					for (int i = 0; i < 10; i++) {
						t.add(i);
						System.out.println("线程1添加" + i);
						try {
							Thread.sleep(100);
						} catch (InterruptedException e1) {
							e1.printStackTrace();
						}
						if(t.size() == 5) {
							object.notify();
							try {
								object.wait();
							} catch (InterruptedException e) {
								e.printStackTrace();
							}
						}
					}
				}
				System.out.println("线程1结束");
			}
		}.start();
	}
	
}


3.使用Latch(门闩)替代wait notify来进行通知

好处是通信方式简单,同时也可以指定等待时间

使用await和countdown方法替代wait和notify

CountDownLatch不涉及锁定,当count的值为零时当前线程继续运行
当不涉及同步,只是涉及线程通信的时候,用synchronized + wait/notify就显得太重了
这时应该考虑countdownlatch/cyclicbarrier/semaphore

public class Test3 {

	private volatile List list = new ArrayList();

	public void add(Object o) {
		list.add(o);
	}
	public int size() {
		return list.size();
	}
	public static void main(String[] args) {
		
		final Test3 t = new Test3();
		final CountDownLatch latch = new CountDownLatch(1);//这里是闩上一个门
		
		new Thread() {
			public void run() {
				System.out.println("线程1启动");
				if(t.size() != 5) {
					try {
						latch.await();//调用该方法,下面的代码停止执行,被闩住了
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
				System.out.println("线程1结束");
			}
		}.start();
		
		new Thread() {
			public void run() {
				for (int i = 0; i < 10; i++) {
					t.add(i);
					System.out.println("线程2添加了" + i);
					if(t.size() == 4) {
						// 打开门闩,让另一个线程得以执行
						latch.countDown();
					}
				}
			}
		}.start();
	}
}



运行的结果

线程1启动
线程2添加了0
线程2添加了1
线程2添加了2
线程2添加了3
线程2添加了4
线程1结束
线程2添加了5
线程2添加了6
线程2添加了7
线程2添加了8
线程2添加了9


kernel<<<>>>用法是指在CUDA编程中,使用<<<>>>符号来定义并行执行的内核函数。内核函数是在GPU上执行的函数,每个线程都会执行一次该函数。<<<>>>符号中的参数表示线程块(block)和线程(thread)的数量。例如,kernel<<<block, thread>>>()表示在block个线程块中,每个线程块中有thread个线程。 在上述示例中,kernel2和kernel3是两个并行执行的内核函数,<<<grid, block>>>表示在grid个线程块中,每个线程块中有block个线程。这样就可以同时启动多个线程块执行相同的内核函数。 在CUDA编程中,还可以使用__syncthreads()函数来同步线程。这个函数会让所有的线程在同一点等待,直到所有线程都执行到这个点。这可以用来确保线程之间的同步和协作。 另外,在示例中的cudaMemcpy()函数用于在主机和设备之间进行内存的数据传输。该函数的参数包括目标内存地址、源内存地址、数据数量和传输方向。例如,cudaMemcpy(dst, src, count, cudaMemcpyKind)表示将count个数据从src内存地址复制到dst内存地址。 综上所述,kernel<<<>>>用法是在CUDA编程中定义并行执行的内核函数,通过设置线程块和线程的数量来控制并发执行的规模。同时可以使用__syncthreads()函数来同步线程,以及使用cudaMemcpy()函数来进行主机和设备之间的数据传输。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [「并行学习」CUDA](https://blog.csdn.net/weixin_41468462/article/details/103378541)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值