黑马程序员--多线程

------Java培训、Android培训、iOS培训、.Net培训、期待与您交流! -------

多线程的概述

多线程:就是指应用程序有多条执行路径。
  进程:正在运行的应用程序。
  线程:进程的执行单元,一条执行路径。
注意:线程不能脱离进程而单独存在也就是线程是依赖于进程存在

多线程:多条线程同时处理数据,可以提高效率,但是同时也会消耗CPU资源,所以项目中需仔细斟酌取舍有当


多线程类:Thread
注意:java是不能操作CPU的,所以java提供了一个Thread类,但是底层开线程的东西肯定C或者C++写的,比如native


多线程的实现方式有两种:

方式1:继承Thread类。
A:定义一个类继承Thread类。
B:子类要重写Thread类的run()方法。
C:让线程启动并执行。
  注意:调用start()方法,切记不是调用run方法这个方法,
其实做了两件事情,第一,让线程启动。第二,自动调用run()方法。


方式2:实现Runnable接口
A:创建一个类实现Runnable接口
B:重写run()方法
C:创建类的实例
D:把类的实例作为Thread的构造参数传递,创建Thread对象,让线程启动并执行
注意:既然有了继承Thread类的方式,为什么还要有实现Runnable接口的方式?
1):避免的单继承的局限性
2):实现接口的方式,只创建了一个资源对象,更好的实现了数据和操作的分离。
   一般我们选择第二种方式。


这两种方式所用到的方法:
public final String getName():获取线程对象的名称。默认情况下,名字的组成 Thread-编号(编号从0开始)
  public final void setName(String name):设置线程名称。
public static Thread currentThread():返回当前正在执行的线程对象引用


多线程经典案例卖票案例:
方式一继承Thread:
public class TicketThread extends Thread {
private static int tickets = 200;//这个地方一定要注意,因为每个TicketThread实例调用的自己的
//run方法,所以tickets要写成大家共享的

public void run() {

while (true) {//由于这个地方有个无限循环,所以说练习时候请强制关闭
if (tickets > 0) {
System.out.println(getName() + "正在出售第" + (tickets--) + "张票");
}

}
}
}


方式二实现Runnable:
public class TicketRunnable implements Runnable {


private int tickets = 200;//请注意这个地方不用加static,因为 多个线层用的都是同一个TicketRunnable
 //所以只要把tickets写到成员位置就ok

public void run() {
while (true) {//由于这个地方有个无限循环,所以说练习时候请强制关闭
if (tickets > 0) {
System.out.println(Thread.currentThread().getName() + "正在出售第"
+ (tickets--) + "张票");
}
}
}


}



多线程同步机制:
   
1):这时候我如果把
while (true) {
if (tickets > 0) {
System.out.println(getName() + "正在出售第" + (tickets--) + "张票");
}

}
变成
while (true) {
try {
Thread.sleep(200); 
} catch (InterruptedException e) {
e.printStackTrace();
}
if (tickets > 0) {
System.out.println(getName() + "正在出售第" + (tickets--) + "张票");
}

}
目的是让线程拖延的时间更长一些,这样一个线程在执行这段代码的时候时间变长了,其他线程和这个线程执行同一段代码的
几率就大一些。



2) 用synchronized (obj){}给代码加锁
如何确定多线程存在不同步问题(多线程出问题的判断条件)
  A:看有没有共享数据
  B:看对共享数据的操作是不是多条语句
  C:看是不是在多线程程序中
  找到后,就把同时满足这三个条件的代码给锁起来。


synchronized (obj) {//小括号呢 可以接收Object及其子类的一个对象,而且多个线层一定要用同一个所对象,
   //就相当于多个线程必须走同一个门才行,否则他还是会从另外的门走这个程序
//锁对象的状态:开,关
//t1进来了,给外界了一个关的状态。
if (tickets > 0) {
try {
//t1睡了
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在出售第"
+ (tickets--) + "张票");
}
}  //t1把状态修改为开


3)请问synchronized (obj){}小括号里面的对象传什么好?

1)
synchronized (this){}
小括号里面写this

2)静态方法的锁对象是是当前类的字节码文件对象。
  类名.class - Class类型的对象
比如:synchronized (TicketRunnable.class) {}

3)用同步代码块和同步方法
因为使用同步机制的话,计算机会重新开辟新的空间来管理同步机制,影响效率
所以被同步的内容越少越好,所以,一般使用同步代码块。
如果一个方法内部全部都被同步了,那么,可以考虑使用同步方法。


4):那代码被锁住了,会不会出现代码块一直被锁住的情况
public void run() {
if (flag) {
synchronized (MyLock.objA) { 
System.out.println("true -- objA");//d1--stop
synchronized (MyLock.objB) { //d1
System.out.println("true -- objB");
}
}
} else {
synchronized (MyLock.objB) {
System.out.println("false -- objB");//d2
synchronized (MyLock.objA) { //d2
System.out.println("false -- objA");
}
}
}
}

注意:1)wait()和notify()方法是Object的,必须由锁对象调用


     2)wait()和notify() 必须写在同步代码块或者同步方法里面,
也正是因为写在里面 才能有锁对象,才能让锁对象调用这两个方法


     3)wait()方法 让线程退出同步代码块等待 ,所以其他线程就可以进这个同步代码块了,(这也就是
好多人常说的 释放锁对象,其实是退出到了同步代码块外面去了,所以其他线程可以进入了)


     4)notify()方法 唤醒线程队列中的随机一个处于等待状态的的线程


     5)wait()和sleep(Long time)的区别
wait():是Object类的方法,可以不用传递参数。释放锁对象。
用锁对象来调用,需要锁对象调用notify来唤醒
wait()必须写在同步代码块或者同步方法里面
sleep():是Thread类的静态方法,需要传递参数。不释放锁对象。
Thread直接调用sleep即可,自动睡眠一段时间,不需要唤醒,一段时间后自动睡醒
一般写在run方法里面,(也可以写到其他任何地方,因为每个程序都是最起码有一个线程)


线程的优先级
试线程的优先级问题:
 
线程默认优先级是5。范围是1-10。
public final int getPriority():获取线程优先级
public final void setPriority(int newPriority):更改线程的优先级。
 
注意:优先级可以在一定的程度上,让线程获较多的执行机会。



Delphi 12.3 作为一款面向 Windows 平台的集成开发环境,由 Embarcadero Technologies 负责其持续演进。该环境以 Object Pascal 语言为核心,并依托 Visual Component Library(VCL)框架,广泛应用于各类桌面软件、数据库系统及企业级解决方案的开发。在此生态中,Excel4Delphi 作为一个重要的社区开源项目,致力于搭建 Delphi 与 Microsoft Excel 之间的高效桥梁,使开发者能够在自研程序中直接调用 Excel 的文档处理、工作表管理、单元格操作及宏执行等功能。 该项目以库文件与组件包的形式提供,开发者将其集成至 Delphi 工程后,即可通过封装良好的接口实现对 Excel 的编程控制。具体功能涵盖创建与编辑工作簿、格式化单元格、批量导入导出数据,乃至执行内置公式与宏指令等高级操作。这一机制显著降低了在财务分析、报表自动生成、数据整理等场景中实现 Excel 功能集成的技术门槛,使开发者无需深入掌握 COM 编程或 Excel 底层 API 即可完成复杂任务。 使用 Excel4Delphi 需具备基础的 Delphi 编程知识,并对 Excel 对象模型有一定理解。实践中需注意不同 Excel 版本间的兼容性,并严格遵循项目文档进行环境配置与依赖部署。此外,操作过程中应遵循文件访问的最佳实践,例如确保目标文件未被独占锁定,并实施完整的异常处理机制,以防数据损毁或程序意外中断。 该项目的持续维护依赖于 Delphi 开发者社区的集体贡献,通过定期更新以适配新版开发环境与 Office 套件,并修复已发现的问题。对于需要深度融合 Excel 功能的 Delphi 应用而言,Excel4Delphi 提供了经过充分测试的可靠代码基础,使开发团队能更专注于业务逻辑与用户体验的优化,从而提升整体开发效率与软件质量。 资源来源于网络分享,仅用于学习交流使用,请勿用于商业,如有侵权请联系我删除!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值