JAVA自定义监听器(通过新线程+while+violate)

 

前言

大一时,我学了Java的awt,一直觉得监听器是个很神奇方便的东西(监听各种事件,如鼠标点击事件等)一直想要写一个自己的监听器,也在百度查过监听器的实现原理,但是因为当时太菜,也忘了是看不懂还是查不到.一直没有成功.直到过了一段时间,应该是大一下学期,我在尝试自己写一个单片机流水灯驱动的时候.突发灵感,可以使用while循环便利的方法实现自定义的监听器,(比如监听一个变量从5变成3),由于当时对Java底层的线程处理并不清楚,更加不知道Java本地方法栈获取堆中成员变量的具体过程,并不了解volatile关键字,所以导致的想法的失败,而且苦苦找不到原因.因为逻辑上是没有错误的. 直到最近深入学习了netty,又接触到监听器和回调等概念,想起了曾经的理想,动手自定义一个简单的入门监听器


 

一、需求

模拟一个工作状态,工作时,顺便监听一个变量, 如果他的值变成负数,在控制台打印警报,

 

二、步骤

1.定义一个类,并且声明他的静态变量A 初始化为1,同时先准备好main做程序入口

public class Lei {
	
	static int A = 1;

	public static void main(String[] args) {
		

	}

}

 

2.在该文件中创建另外一个监听类 继承Thread, 作为守护线程监听变量A

class jt extends Thread {

	@Override
	public void run() {
		System.out.println("默默监听");
		while(true) {
			if(Lei.A < 0) {
				System.out.println("监听到异常");
				break;
			}
		}
	}
	
}

 

3.在main方法中开启作业 通过不断输入的方式不断改变A的值

public static void main(String[] args) {
		new jt().start();
		
		Scanner sc = new Scanner(System.in);
		while(true) {
			A = sc.nextInt();
		}
		
	}

三、试验和改良

理论上,根据我们的逻辑,jt线程会一直监听变量A的值,如果A的值变为负数,会立刻报警,而实际上.

我们发现就算我们给他赋值负数,他也不监听到,我那时候结果几次冥想后认定是java的bug,直到后来我看了<<深入了解java虚拟机>>后才知道,我缺少了volatile关键字, 我们把A变量加上volatile关键字,全部代码如下

import java.util.Scanner;

public class Lei {
	
	static volatile int A = 1;

	public static void main(String[] args) {
		new jt().start();
		
		Scanner sc = new Scanner(System.in);
		while(true) {
			A = sc.nextInt();
		}
		
	}

}

class jt extends Thread {

	@Override
	public void run() {
		System.out.println("默默监听");
		while(true) {
			if(Lei.A < 0) {
				System.out.println("监听到异常");
				break;
			}
		}
	}
	
}

试验结果成功

 

 

四、总结和后语

就这样我们成功实现了一个自定义的监听器,监听A变为负数, 如果我们需要多几个监听器,为了提高程序效率,我们可以在jt类的run方法中通过多个if来监听多个事件,而不要创建多个线程,这也是nio思想的一个小细节

public void run() {
		System.out.println("默默监听");
		while(true) {
			if(Lei.A < 0) {
				System.out.println("监听到异常");
				break;
			}

           if(...){...};

           if(...){...};

           if(...){...};
		}
	}

 

至于volatile的作用,简单来说就是保证一个变量在多个线程的可见性,有兴趣的可以百度一下,有很多相关的资料,不过个人还是建议不懂的人能够体系地学一下java的多线程知识点,包括线程池,原子类,线程虚拟机层次的指令原理等等,这样有助于我们更好地了解Java地特性,如果有更好地方法,欢迎大神指教和提意见.

 

 

 

 

 

 

 

 

  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Java中,编写比较方法时需要遵守“通用契约”,这是为了保证排序算法的正确性。然而,有时候会出现“java时间排序comparison method violate lts general contract”的错误。这是因为在比较方法中违反了通用契约的规定。 通用契约中指定了三个规则: 1. 反对称性(Antisymmetry):如果两个对象进行比较,其中一个大于另一个,那么另一个一定小于前者。如果两个对象相等,它们之间的比较结果应为0。 2. 传递性(Transitivity):如果对象A大于对象B,B大于对象C,那么A一定大于C。同样地,如果A等于B,B等于C,那么A一定等于C。 3. 一致性(Consistency):在比较过程中,如果两个对象发生变化,它们之间的比较结果应该随之改变。 当我们在比较Java中的时间时,经常使用的是Date类。要满足通用契约,我们需要按照时间顺序进行比较。然而,有时候我们在编写比较方法时可能没有正确地实现这些规则,导致了“java时间排序comparison method violate lts general contract”错误。 要解决这个问题,我们需要回顾一下比较方法的实现,并确保满足通用契约的规定。首先,我们应该确保在比较中使用的属性是可比较的,并且没有为null的情况。然后,我们应该检查比较方法是否正确地比较了两个对象的时间顺序,并根据比较结果返回正确的值。 此外,还有一种可能导致此错误的情况是,排序算法中使用了不支持通用契约的比较方法。在这种情况下,我们应该更改排序算法或者使用其他可靠的比较方法。 总之,解决“java时间排序comparison method violate lts general contract”错误,首先我们需要检查并修复比较方法中的错误。确保按照通用契约的规定进行比较,并且没有使用不支持通用契约的比较方法。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值