2021-09-10

synchronized笔记

老韩零基础学Java,作业2



线程的同步
    线程同步机制
        1.在多线程编程,一些敏感数据不允许被多个线程同时访问,此时就使用同步访问技术,
        	保证数据在任何同一时刻, 最多有一个线程访问,以保证数据的完整性。
        2.也可以理解为:线程同步,即当有一个线程在对内存进行操作时,
        	其他线程都不可以对这个内存地址进行操作,
            直到该线程完成操作,其他线程才能对该内存地址进行操作

    同步具体方法-Synchronized
        1.同步代码块
            synchronized(对象){ //得到对象的锁,才能操作同步代码
                // 需要被同步代码;
            }

        2. synchronized 还可以放在方法声明中,表示整个方法-为同步方法
            public synchronized void m(String name){
                // 需要被同步的代码
            }
            
	互斥锁 :synchronized
    基本介绍:
    1.在java语言中,引入了对象互斥锁的概念,来保证共享数据操作的完整性
    2.每个对象都对应于一个可称为 "互斥锁" 的标记,这个标记用来保证在任意时刻,只能有一个线程访问对象
    3.关键字synchronized 来与对象的互斥锁联系。当某个对象用synchronized 修饰时,
    	表明该对象在任意时刻只能由一个线程访问
    4.同步的局限性:导致程序的执行效率要降低
    5.同步方法(非静态的)的锁可以是this,也可以是其他对象(要求是同一对象)
    6.同步方法(静态的)的锁为当前类本身


    注意事项
    1.同步方法如果没有使用static修饰:默认锁对象为this
    2.如果方法使用static修饰,默认锁对象:当前类.class

    3.实现的落地步骤
        需要先分析上锁的代码
        选择同步代码块或同步方法
        要求多个线程的锁对象为同一个即可
        
通过两种方式,锁同一个对象,避免超支取钱
package com.hspedu.homework_;

/**
 * @author rowling
 * @version 1.0
 */
 
public class HomeWork02 {


    public static void main(String[] args) {
        User user = new User();
        User2 user2 = new User2();
        user.start();
        user2.start();

//        User3 user3 = new User3();
//        Thread t1 = new Thread(user3);
//        t1.setName("t1");
//        t1.start();
//        Thread t2 = new Thread(user3);
//        t2.setName("t1");
//        t2.start();
    }
}

class User extends Thread{
    public static Integer money = 10000;
    private boolean loop = true;

    public void sell(){
        synchronized (money){
            if(money <= 0){
                System.out.println(Thread.currentThread().getName() + "没有余额...");
                loop = false;
                return ;
            }
            System.out.println(Thread.currentThread().getName() + "取钱后的金额还有:"
            	+ (money-=1000));
            try {
                Thread.sleep(50);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

    }
    @Override
    public void run() {
        while (loop){
            sell();
        }
    }
}



class User2 extends Thread{
    private boolean loop = true;

    public void sell(){
        synchronized (User.money){
            if(User.money <= 0){
                System.out.println(Thread.currentThread().getName() + "没有余额...");
                loop = false;
                return ;
            }
            System.out.println(Thread.currentThread().getName() + "取钱后的金额还有:" 
            	+ (User.money-=1000));
            try {
                Thread.sleep(50);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

    }

    @Override
    public void run() {
        while (loop){
            sell();
        }
    }
}


class User3 implements Runnable{
    public  int money = 10000;

    @Override
    public void run() {
        while (true){
            //解读:
            //1. 这里使用 synchronized 实现了线程同步
            //2. 当多个线程执行到这里时,就会去争夺 this 对象锁
            //3. 那个线程争夺(获取)this 对象锁,就执行 代码块,执行完后,会释放this对象锁
            //4. 争夺不到this 对象锁, 就 blocked, 准备继续争夺
            //5. this 对象锁非公平
            synchronized (this){
                if(money <= 0){
                    System.out.println(Thread.currentThread().getName() + "没有余额...");
                    break ;
                }
                System.out.println(Thread.currentThread().getName() + "取钱后的金额还有:"
                	+ (money-=1000));
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用python中的pymsql完成如下:表结构与数据创建 1. 建立 `users` 表和 `orders` 表。 `users` 表有用户ID、用户名、年龄字段,(id,name,age) `orders` 表有订单ID、订单日期、订单金额,用户id字段。(id,order_date,amount,user_id) 2 两表的id作为主键,`orders` 表用户id为users的外键 3 插入数据 `users` (1, '张三', 18), (2, '李四', 20), (3, '王五', 22), (4, '赵六', 25), (5, '钱七', 28); `orders` (1, '2021-09-01', 500, 1), (2, '2021-09-02', 1000, 2), (3, '2021-09-03', 600, 3), (4, '2021-09-04', 800, 4), (5, '2021-09-05', 1500, 5), (6, '2021-09-06', 1200, 3), (7, '2021-09-07', 2000, 1), (8, '2021-09-08', 300, 2), (9, '2021-09-09', 700, 5), (10, '2021-09-10', 900, 4); 查询语句 1. 查询订单总金额 2. 查询所有用户的平均年龄,并将结果四舍五入保留两位小数。 3. 查询订单总数最多的用户的姓名和订单总数。 4. 查询所有不重复的年龄。 5. 查询订单日期在2021年9月1日至9月4日之间的订单总金额。 6. 查询年龄不大于25岁的用户的订单数量,并按照降序排序。 7. 查询订单总金额排名前3的用户的姓名和订单总金额。 8. 查询订单总金额最大的用户的姓名和订单总金额。 9. 查询订单总金额最小的用户的姓名和订单总金额。 10. 查询所有名字中含有“李”的用户,按照名字升序排序。 11. 查询所有年龄大于20岁的用户,按照年龄降序排序,并只显示前5条记录。 12. 查询每个用户的订单数量和订单总金额,并按照总金额降序排序。
06-03
这里是使用 Python 的 PyMySQL 完成上述任务的代码: ```python import pymysql # 连接数据库 conn = pymysql.connect(host='localhost', port=3306, user='root', password='123456', database='test') # 创建游标 cur = conn.cursor() # 创建 users 表 cur.execute(''' CREATE TABLE users ( id INT PRIMARY KEY, name VARCHAR(20), age INT ) ''') # 创建 orders 表 cur.execute(''' CREATE TABLE orders ( id INT PRIMARY KEY, order_date DATE, amount INT, user_id INT, FOREIGN KEY (user_id) REFERENCES users(id) ) ''') # 插入数据 users_data = [ (1, '张三', 18), (2, '李四', 20), (3, '王五', 22), (4, '赵六', 25), (5, '钱七', 28) ] orders_data = [ (1, '2021-09-01', 500, 1), (2, '2021-09-02', 1000, 2), (3, '2021-09-03', 600, 3), (4, '2021-09-04', 800, 4), (5, '2021-09-05', 1500, 5), (6, '2021-09-06', 1200, 3), (7, '2021-09-07', 2000, 1), (8, '2021-09-08', 300, 2), (9, '2021-09-09', 700, 5), (10, '2021-09-10', 900, 4) ] for user in users_data: cur.execute('INSERT INTO users VALUES (%s, %s, %s)', user) for order in orders_data: cur.execute('INSERT INTO orders VALUES (%s, %s, %s, %s)', order) # 提交事务 conn.commit() # 查询语句 # 1. 查询订单总金额 cur.execute('SELECT SUM(amount) FROM orders') print(cur.fetchone()[0]) # 2. 查询所有用户的平均年龄,并将结果四舍五入保留两位小数。 cur.execute('SELECT ROUND(AVG(age), 2) FROM users') print(cur.fetchone()[0]) # 3. 查询订单总数最多的用户的姓名和订单总数。 cur.execute(''' SELECT users.name, COUNT(*) AS total_orders FROM users JOIN orders ON users.id = orders.user_id GROUP BY users.id ORDER BY total_orders DESC LIMIT 1 ''') print(cur.fetchone()) # 4. 查询所有不重复的年龄。 cur.execute('SELECT DISTINCT age FROM users') print([row[0] for row in cur.fetchall()]) # 5. 查询订单日期在2021年9月1日至9月4日之间的订单总金额。 cur.execute('SELECT SUM(amount) FROM orders WHERE order_date BETWEEN "2021-09-01" AND "2021-09-04"') print(cur.fetchone()[0]) # 6. 查询年龄不大于25岁的用户的订单数量,并按照降序排序。 cur.execute(''' SELECT users.name, COUNT(*) AS total_orders FROM users JOIN orders ON users.id = orders.user_id WHERE age <= 25 GROUP BY users.id ORDER BY total_orders DESC ''') print(cur.fetchall()) # 7. 查询订单总金额排名前3的用户的姓名和订单总金额。 cur.execute(''' SELECT users.name, SUM(amount) AS total_amount FROM users JOIN orders ON users.id = orders.user_id GROUP BY users.id ORDER BY total_amount DESC LIMIT 3 ''') print(cur.fetchall()) # 8. 查询订单总金额最大的用户的姓名和订单总金额。 cur.execute(''' SELECT users.name, SUM(amount) AS total_amount FROM users JOIN orders ON users.id = orders.user_id GROUP BY users.id ORDER BY total_amount DESC LIMIT 1 ''') print(cur.fetchone()) # 9. 查询订单总金额最小的用户的姓名和订单总金额。 cur.execute(''' SELECT users.name, SUM(amount) AS total_amount FROM users JOIN orders ON users.id = orders.user_id GROUP BY users.id ORDER BY total_amount ASC LIMIT 1 ''') print(cur.fetchone()) # 10. 查询所有名字中含有“李”的用户,按照名字升序排序。 cur.execute('SELECT * FROM users WHERE name LIKE "%李%" ORDER BY name ASC') print(cur.fetchall()) # 11. 查询所有年龄大于20岁的用户,按照年龄降序排序,并只显示前5条记录。 cur.execute('SELECT * FROM users WHERE age > 20 ORDER BY age DESC LIMIT 5') print(cur.fetchall()) # 12. 查询每个用户的订单数量和订单总金额,并按照总金额降序排序。 cur.execute(''' SELECT users.name, COUNT(*) AS total_orders, SUM(amount) AS total_amount FROM users JOIN orders ON users.id = orders.user_id GROUP BY users.id ORDER BY total_amount DESC ''') print(cur.fetchall()) # 关闭游标和连接 cur.close() conn.close() ``` 注意:在运行代码之前,需要先安装 PyMySQL 模块,可以使用以下命令进行安装: ``` pip install pymysql ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值