本篇博客是对在线人员列表逻辑混乱反例进行的总结和进行的改进。
目的是将经历转变为自己的经验。通过博客的方式分享给大家,大家一起共同进步和提高。
上面的问题主要分为四类问题:
由于线程共享全局变量,并且在实际业务的过程中对全局变量进行了操作(修改),所以在高并发、多线程的条件下会产生公共变量导致数据混乱的现象。
ThreadLocal叫做线程变量,该变量对于其他线程而言是隔离的,该变量是当前该线程独有的变量。
它与普通变量的区别在于,每个使用该变量的线程都会初始化一个完全独立的实例副本。正是由于每个线程都有自己的实例副本,且与其他线程不可访问,那就不存在多线程共享该变量的问题。
ThreadLocal 常见使用场景
1、每个线程需要有自己单独的实例
2、实例需要在多个方法中共享,但不希望被多线程共享
多线程环境下未使用ThreadLocal出现数据混乱的问题
public class Client {public static void main(String[] args) {// 定义线程实现接口Runnable runnable = new Runnable(){Counter counter = new Counter();@Overridepublic void run() {counter.count();}};// 启动10个线程for( int i= 0;i< 10;i++) {new Thread(runnable).start();}}}
public class Counter{private int number;public void count(){for (int i = 0; i <=10 ; i++) {number=number+i;}System.out.println(Thread.currentThread().getName()+ "--"+number;}
}
实现结果
出现数据混乱的问题,正确计算结果应该都为55
多线程环境下使用ThreadLocal解决数据混乱的问题
public class Client {public static void main(String[] args) {// 定义线程实现接口Runnable runnable = new Runnable(){Counter counter = new Counter();@Overridepublic void run() {counter.count();}};// 启动10个线程for( int i= 0;i< 10;i++) {new Thread(runnable).start();}}}
public class Counter {private static ThreadLocal number = new ThreadLocal() {// 重写这个方法,可以修改“线程变量”的初始值,默认是null@Overrideprotected Integer initialValue() {return 0;}};//计数方法public void count() {for( int i= 0;i<= 10;i++) {number.set(number.get()+i);}System.out.println(Thread.currentThread().getName()+ "--"+number.get());}
}
实现结果
public void count(){int number = 0;for (int i = 0; i <=10 ; i++) {number=number+i;}System.out.println(Thread.currentThread().getName()+ "--"+number);}}
实现效果
解决思路:由于这里查询的入参带了createDate,如果该学生从第一天到第二天一直没有下课,那么你这儿查询的话由于带了createDate就查询不到该学生的在线数据,在后续的流程中就会重复插入,该学生就会有两条数据,查询的时候就会有数据问题。
解决思路: 对于isOnline参数,上面paramUser中有该参数。不必再进行赋值。这样的好处是参数值保持了一致,降低了复杂度。
**解决思路:**这里主要是对在线人员进行下线处理,1.如果在这里更新了update_time那么就相当于这个课程班级中的所有学生的最近一次登陆时间都是一致的,并且最近一次登陆时间与实际不符。2.如果where条件上加上了create_date作为条件那么如果过了十二点结束课程,那么所有在线的人就无法进行下线处理了。
正确的SQL语句:
解决思路: 1.子类可以继承父类的所有非private修饰的属性和方法 2.this.方法的问题
实例结构
动物类
public class Animal {protected String name="动物" ;protected String age="18";public void printName(){System.out.println(name);}public void printAge(){System.out.println(age);}
}
猫类
public class Cat extends Animal {private String name="猫";public void printName(){System.out.println(this.name);System.out.println(super.name);}public void println(){this.printName();super.printName();}}
调用类
public class Client {public static void main(String[] args) {Cat cat =new Cat();cat.printName();cat.printAge();cat.println();}
}
实现结果
结论:
java中继承关系的父子类,相同的方法会被重写
当子类父类中的成员没有重名时,子类都可以通过this去调用。
当成员方法 重名,子类就会将父类中的方法进行重写。如果还想调用父类中的方法只能通过 super去进行调用。
当时成员变量不存在覆盖重写:在子类中只能通过super调用父类的
在实际的项目开发过程中,需要特别注意关联字段的数据类型是否一致。吐过不一致将导致索引失效,出现mansql的情况。
索引失效的问题复现
EXPLAIN SELECTaui.ding_phone,aui.ding_name,aui.chaoxing_name,aui.chaoxing_phoneFROM(SELECT info_id FROM `arpro_user_course_info` WHERE course_id = '223667994' AND class_id = '55801765' AND is_delete = 0 GROUP BY id)as aucileft JOIN arpro_user_info as aui on auci.info_id=aui.id
arpro_user_course_info与arpro_user_info通过info_id与aui.id进行主外键的关联。
arpro_user_info的id类型为varchar类型
arpro_user_course_info的info_id类型为bigint类型
通过EXPLAIN进行sql语句的性能分析
发现arpro_user_info的为全部查询,索引失效了。
将关联字段的类型修改为一致再次进行测试
测试结果
索引生效
下一篇:前缀和 及其优化技巧