首頁 Java Java基礎 Java並發程式設計之介紹線程安全基礎

Java並發程式設計之介紹線程安全基礎

Feb 24, 2021 am 10:10 AM
java 線程安全

Java並發程式設計之介紹線程安全基礎

免費學習推薦:java基礎教學

  • #線程安全基礎
    • 1.線程安全問題
    • 2.帳戶取款案例
    • 3.同步程式碼區塊synchronized
    • #synchronized的理解
    java中有三大變數的執行緒安全性問題
  • 在實例方法上使用synchronized
  • 總結
  • #面試題
  • 4.死鎖
  • 5.開發中該怎麼解決執行緒安全性問題
  • 6.守護執行緒
# 7.定時器

8.實作執行緒的第三種方式:實作Callable介面

#9.Object類別中的wait和notify方法


10.生產者和消費者


Java並發程式設計之介紹線程安全基礎1.線程安全性問題

2.1、為什麼這個才是重點?
  以後在開發中,我們的專案都是運行在伺服器當中,而伺服器已經將執行緒的定義,執行緒物件的創建,執行緒的啟動等,都已經實現了。這些程式碼我們都不需要寫。
  最重要的是:你要知道,你寫的程式需要放到一個多執行緒的環境下運行,你更需要關注的是這些資料在多執行緒並發的環境下是否是安全的。 (重點:*****)
2.2、何時資料在多執行緒並發的環境下會存在安全性問題呢?

  三個條件:
  條件1:多執行緒並發。
  條件2:有共享資料。   條件3:共享資料有修改的行為。
  滿足以上3個條件之後,就會存在執行緒安全性問題。
2.3、怎麼解決執行緒安全問題呢?
當多執行緒並發的環境下,有共享數據,而這個數據還會被修改,此時就存在線程安全問題,怎麼解決這個問題?   執行緒排隊執行。 (不能併發)。
  用排隊執行解決線程安全問題。
  這種機制稱為:執行緒同步機制。   專業術語叫做:
執行緒同步,其實就是執行緒不能並發了,執行緒必須排隊執行

  怎麼解決線程安全性問題呀?
  使用「執行緒同步機制」。

  線程同步就是線程排隊了,線程排隊了就會犧牲一部分效率,沒辦法,資料安全第一位,只有資料安全了,我們才可以談效率。數據不安全,沒有效率的事兒。
2.4、說到線程同步這塊,涉及到這兩個專業術語:

非同步程式設計模型:
  線程t1和線程t2,各自執行各自的, t1不管t2,t2不管t1,   誰也不需要等誰,這種程式設計模型叫做:非同步程式設計模型。   其實就是:多執行緒並發(效率較高。)

同步程式設計模型:

  執行緒t1和執行緒t2,在執行緒t1執行的時候,必須等待t2線程執行結束,或者說在t2線程執行的時候,必須等待t1線程執行結束,兩個線程之間發生了等待關係,這就是同步編程模型。效率較低。線程排隊執行。

非同步就是並發。同步就是排隊。

Java並發程式設計之介紹線程安全基礎2.帳戶提款案例

Account類別

package ThreadSafe;public class Account {
	//账号
	private String actno;
	//余额
	private double balance;
	
	public Account(String actno, double balance) {
		super();
		this.actno = actno;
		this.balance = balance;
	}
	public String getActno() {
		return actno;
	}
	public void setActno(String actno) {
		this.actno = actno;
	}
	public double getBalance() {
		return balance;
	}
	public void setBalance(double balance) {
		this.balance = balance;
	}
	//取款的方法
	public void withdraw(double money){
		//t1和t2并发执行这个方法(t1和t2是两个栈 ,两个栈操作堆中同一个对象)
		//取款之前的余额
		double before=this.getBalance();
		//取款之后的余额
		double after=before-money;
		//模拟一下网络延迟,会出现问题
		try {
			Thread.sleep(1000);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		//更新余额
		//思考:t1执行到这里了,但还没有来得及执行这行代码,t2线程进来withdraw方法了,此时一定出问题
		this.setBalance(after);
	}}AccountThread类public class AccountThread extends Thread{
	//两个线程必须共享一个账户对象
	private Account act;
	//通过构造方法传递过来账户对象
	public AccountThread(Account act) {
		this.act = act;
	}
	@Override
	public void run() {
		//假设取款5000
		double money=5000;
		//多线程执行这个方法
		act.withdraw(money);
		System.out.println(Thread.currentThread().getName()+"账户"+act.getActno()+"取款成功"+act.getBalance());
	}}
登入後複製
Test類別

public class Test {
	public static void main(String[] args) {
		//创建账户对象
		Account act=new Account("act-001",10000);
		//创建两个线程
		Thread t1=new AccountThread(act);
		Thread t2=new AccountThread(act);
		
		t1.setName("t1");
		t2.setName("t2");
		//启动两个线程执行
		t1.start();
		t2.start();
	}}
登入後複製


#3.同步程式碼區塊synchronized

synchronized的理解
//以下這幾行程式碼必須是執行緒排隊的,不能並發
//
一個執行緒把這裡的程式碼全部執行結束後,另一個執行緒才能進來

/*
執行緒同步機制的語法是 synchronized( ){

//執行緒同步程式碼區塊。
}

synchronized後面小括號中傳的這個數據是相當關鍵的,

這個數據必須是多執行緒共享的數據,才能達到多執行緒排隊。

()中寫什麼? ### 那要看你想讓哪些線程同步### 假設t1、t2、t3、t4、t5有5個線程### 你只希望t1 t2 t3排隊,t4 t5不需要排隊,怎麼辦## # 你一定要在()中寫一個t1 t2 t3共享的對象,而這個對象對於t4 t5來說不是​​共享的### 這裡的共享對像是賬戶對象### 帳戶對像是共享的,this是賬戶物件### 有時不一定是this,這裡只要是多執行緒共享的那個物件就行###
		synchronized(this){
			double before=this.getBalance();
			double after=before-money;
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
			this.setBalance(after);
		}
登入後複製

Java並發程式設計之介紹線程安全基礎
在java语言中,任何对象都有一把锁,其实这把锁就是一个标记,(只是把它叫做锁)
100个对象,100个锁,1个对象1把锁。
以上代码的执行原理是什么呢?
1.假设t1和t2线程并发,开始执行以上代码的时候,肯定有一个先一个后,
2.假设t1先执行了,遇到了synchronized,这个时候自动找后面共享对象的对象锁,找到之后,并占有这把锁,然后执行同步代码块中的程序,在程序执行过程中一直都占有这把锁,直到同步代码块执行结束,这把锁才会释放。
3.假设t1已经占有这把锁,此时t2也遇到synchronized关键字,也会去占有后面共享对象的这把锁,结果这把锁被t1占有,t2只能在同步代码块外边等待t1的结束,直到t1把同步代码块执行结束了,t1会归还这把锁,此时t2终于等到这把锁,然后t2占有这把锁之后,进入同步代码块执行程序。
这样就达到了线程排队执行
这里需要注意的是:这个共享对象一定要选好了,这个共享对象一定是你需要排队执行的这些线程对象所共享的。
//对象
Object obj=new Object(); //实例变量(Account对象是多线程共享的,Account对象中的实例变量obj也是共享的)
synchronized(obj){}

Java並發程式設計之介紹線程安全基礎
括号里边只要是共享对象就行。
Object obj2=new Object(); //局部变量
synchronized(obj2){}
这样写就不安全了,因为obj2是局部变量,不是共享对象。
synchronized(“abc”){}
这样写时可以的。存在字符串常量池中
写"abc"的话所有线程都会同步

而如果是写synchronized(this){}的话,我们创建了一个新的对象act2可不用共享对象。
所以最好是写synchronized(this){},比如你要取款,要让其他取别的账户的人也要等吗?不应该,只有同时对你这1个账户取款的时候,需要等待,别人取钱的时候,需要从其他账户中取钱,就不需要等待。

java中有三大变量的线程安全问题

实例变量,在堆中
静态变量,在方法区
局部变量,在栈中

以上三大变量
局部变量永远不会存在线程安全问题
因为局部变量不共享(一个线程一个栈)
局部变量在栈中,所以局部变量永远都不会共享
实例变量在堆中,堆只有1个。
静态变量在方法区中,方法区只有1个

堆和方法区都是多线程共享的,所以可能存在线程安全问题

局部变量+常量:不会有线程安全问题。
成员变量:可能会有线程安全问题

同步代码块越小,效率越高。

//多线程执行这个方法//synchronized(this)//这里的this是AccountThread对象,这个对象不共享。synchronized(act){
			act.withdraw(money);
			System.out.println(Thread.currentThread().getName()+"账户"+act.getActno()+"取款成功,余额:"+act.getBalance());}
登入後複製

在实例方法上使用synchronized

在实例方法上可以使用synchronized吗?可以的。
synchronized出现在实例方法上,一定锁的是this
没得挑,只能是this,不能是其他的对象了
所以这种方式不灵活。
另外还有一个缺点:synchronized出现在实例方法上,表示整个方法体都需要同步
可能会扩大同步的范围,导致程序的执行效率降低。所以这种方式不常用
synchronized使用在实例方法上有什么优点?
代码写的少了,节俭了
如果共享的对象是this,并且需要同步的代码是整个方法体,建议使用这种方式
StringBuffer就是在每个方法上加了synchronized关键字

使用局部变量的话,最好使用StringBuilder
因为局部变量不存在线程安全问题,选择StringBuilder,StringBuffer效率比较低。
ArrayList是非线程安全的。
Vector是线程安全的。
HashMap HashSet是非线程安全的。
Hashtable是线程安全的

总结

synchronized有三种写法:

  第一种:同步代码块
    灵活
      synchronized(线程共享对象){
        同步代码块;
      }
  第二种:在实例方法上使用synchronized
      表示共享对象一定是this
      并且同步代码块是整个方法体。
  第三种:在静态方法上使用synchronized
      表示找类锁。
      类锁永远只有1把。
      就算创建了100个对象,那类锁也只有一把。

对象锁:1个对象1把锁,100个对象100把锁。
类锁:100个对象,也可能只是1把类锁。

面试题

//面试题:doother方法的执行需不需要等待dosome方法的结束。
//不需要,因为doother方法没有synchronized

public class exam01 {
	public static void main(String[] args) {
		MyClass mc=new MyClass();
		Thread t1=new MyThread(mc);
		Thread t2=new MyThread(mc);
		t1.setName("t1");
		t2.setName("t2");
		t1.start();
		try {
			Thread.sleep(1000);		//这个睡眠的作用是:为了保证t1线程先执行
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		t2.start();
	}}class MyThread extends Thread{
	private MyClass mc;
	public MyThread(MyClass mc) {
		super();
		this.mc = mc;
	}
	public void run(){
		if(Thread.currentThread().getName().equals("t1")){
			mc.dosome();
		}
		if(Thread.currentThread().getName().equals("t2")){
			mc.doOther();
		}
	}
	}class MyClass{
	public synchronized void dosome(){
		System.out.println("doSome begin");
		try {
			Thread.sleep(1000);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		System.out.println("doSome end");
	}
	public void doOther(){
		System.out.println("doOther begin");
		System.out.println("doOther end");
	}}
登入後複製

Java並發程式設計之介紹線程安全基礎
当在doother上面加了synchronized呢
//面试题:doother方法的执行需不需要等待dosome方法的结束。
//需要,因为doother方法没有synchronized

	public synchronized void doOther(){
		System.out.println("doOther begin");
		System.out.println("doOther end");
	}
登入後複製

//面试题:doother方法的执行需不需要等待dosome方法的结束。
//不用排队,谁也不用管谁

MyClass mc1=new MyClass();MyClass mc2=new MyClass();Thread t1=new MyThread(mc1);Thread t2=new MyThread(mc2);
登入後複製

//面试题:doother方法的执行需不需要等待dosome方法的结束。
//需要,因为静态方法是类锁,类锁不管创建了几个对象,类锁只有一把

MyClass mc1=new MyClass();MyClass mc2=new MyClass();Thread t1=new MyThread(mc1);Thread t2=new MyThread(mc2);public synchronized static void dosome(){
		System.out.println("doSome begin");
		try {
			Thread.sleep(1000);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		System.out.println("doSome end");
	}
	public synchronized static void doOther(){
		System.out.println("doOther begin");
		System.out.println("doOther end");
	}
登入後複製

这种锁叫排他锁:

4.死锁

Java並發程式設計之介紹線程安全基礎
synchronized在开发中最好不要嵌套使用,一不小心就可能导致死锁。
死锁代码要会写。
一般面试官要求你会写
只有会写的,才会在以后的开发中注意这个事儿
因为死锁很难调试。

public class DeadLock {
	public static void main(String[] args) {
		Object o1=new Object();
		Object o2=new Object();
		//t1线程和t2线程共享o1,o2
		Thread t1=new MyThread1(o1,o2);
		Thread t2=new MyThread2(o1,o2);
		t1.start();
		t2.start();
	}}class MyThread1 extends Thread{
	Object o1;
	Object o2;
	public MyThread1(Object o1,Object o2){
		this.o1=o1;
		this.o2=o2;
	}
	@Override
	public void run() {
		synchronized (o1) {
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
			synchronized (o2) {
				
			}
		}
	}}class MyThread2 extends Thread{
	Object o1;
	Object o2;
	public MyThread2(Object o1,Object o2){
		this.o1=o1;
		this.o2=o2;
	}
	@Override
	public void run() {
		synchronized (o2) {
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
			synchronized (o1) {
				
			}
		}
	}}
登入後複製

5.开发中应该怎么解决线程安全问题

聊一聊,我们以后开发中应该怎么解决线程安全问题?
  是一上来就选择线程同步吗?synchronized
  不是,synchronized会让程序的执行效率降低,用户体验不好。
  系统的用户吞吐量降低。用户体验差。在不得已的情况下再选择
  线程同步机制。
  第一种方案:尽量使用局部变量代替“实例变量和静态变量”。
  第二种方案:如果必须是实例变量,那么可以考虑创建多个对象,这样实例变量的内存就不共享了。(一个线程对应1个对象,100个线程对应100个对象,对象不共享,就没有数据安全问题了。)
  第三种方案:如果不能使用局部变量,对象也不能创建多个,这个时候就只能选择synchronized了。线程同步机制
线程这块还有那些内容呢?列举一下
7.1、守护线程
7.2、定时器
7.3、实现线程的第三种方式:FutureTask方式,实现Callable接口。(JDK8新特性。)
7.4、关于Object类中的wait和notify方法。(生产者和消费者模式!)

6.守护线程

守护线程
java语言中线程分为两大类:
  一类是:用户线程
  一类是:守护线程(后台线程)

其中具有代表性的就是:垃圾回收线程(守护线程)。
  守护线程的特点:
  一般守护线程是一个死循环,所有的用户线程只要结束,
  守护线程自动结束。
  注意:主线程main方法是一个用户线程。
  守护线程用在什么地方呢?
  每天00:00的时候系统数据自动备份。
  这个需要使用到定时器,并且我们可以将定时器设置为守护线程。
  一直在那里看着,没到00:00的时候就备份一次。所有的用户线程如果结束了,守护线程自动退出,没有必要进行数据备份了。

package testThread;/*
实现守护线程
 * */
	public class ThreadTest13 {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Thread t=new BakDataThread();
		t.setName("备份数据的线程");
		//启动之前,将线程设置为守护线程
		t.setDaemon(true);
		t.start();
		//主线程:主线程是用户线程
		for(int i=0;i"+i);
			try {
				Thread.sleep(1000);
			} catch (Exception e) {
				// TODO: handle exception
			}
		}
	}}class BakDataThread extends Thread{
	@Override
	public void run() {
		int i=0;
		//即使是死循环,但由于该线程是守护者,当用户线程结束,守护线程自动终止
		while(true){
			System.out.println(Thread.currentThread().getName()+"---->"+(++i));
			try {
				Thread.sleep(1000);
			} catch (Exception e) {
				// TODO: handle exception
			}
			
		}
	}}
登入後複製

7.定时器

定时器的作用:
  间隔特定的时间,执行特定的程序。
  每周要进行银行账户的总账操作。
  每天要进行数据的备份操作。
  在实际的开发中,每隔多久执行一段特定的程序,这种需求是很常见的,
那么在java中其实可以采用多种方式实现:
  可以使用sleep方法,睡眠,设置睡眠时间,没到这个时间点醒来,执行任务。这种方式是最原始的定时器。(比较low)
  在java的类库中已经写好了一个定时器:java.util.Timer,可以直接拿来用。不过,这种方式在目前的开发中也很少用,因为现在有很多高级框架都是支持定时任务的。

  在实际的开发中,目前使用较多的是Spring框架中提供的SpringTask框架,这个框架只要进行简单的配置,就可以完成定时器的任务。

8.实现线程的第三种方式:实现Callable接口

实现线程的第三种方式:实现Callable接口。(JDK8新特性。)
这种方式实现的线程可以获取线程的返回值
之前讲解的那两种方式是无法获取线程返回值的,因为run方法返回void。
思考:
系统委派一个线程去执行一个任务,该线程执行完任务之后,可能会有一个执行结果,我们怎么能拿到这个执行结果呢
使用第三种方式:实现Callable接口方式

public class ThreadTest14 {
	public static void main(String[] args) throws Exception, ExecutionException {
		//第一步:创建一个未来任务类对象
		//参数非常重要,需要给一个callable接口的实现类对象
		FutureTask task=new FutureTask(new Callable(){
			@Override
			//call方法相当于是run方法,只不过这个有返回值,线程执行一个任务,执行之后可能会有一个执行结果。
			public Object call() throws Exception {
				System.out.println("call method begin");
				Thread.sleep(1000);
				System.out.println("call method begin");
				int a=100;
				int b=200;
				return a+b;			//自动装箱
			}		
		});
		//创建线程对象
		Thread t=new Thread(task);
		
		//启动线程
		t.start();
		
		//这里是main方法,这是在主线程中
		//在线程中,怎么获取t线程的执行结果
		//get方法的执行会导致当前线程阻塞
		Object obj=task.get();
		System.out.println("线程执行结果"+obj);
		//main方法这里的程序要想执行必须等待get()方法的结束
		//而get方法可能需要很久。因为get()方法是为了拿另一个线程的执行结果。
		//另一个线程的执行是需要时间的
		System.out.println("hello,world");
	}}
登入後複製

这种方式的优点:可以获取到线程的执行结果
这种方式的缺点:效率比较低,在获取t线程执行结果的时候,当前线程受阻塞,效率较低。

9.Object类中的wait和notify方法

(生产者和消费者模式!)
  第一:wait和notify方法不是线程对象的方法,是java中任何一个java对象都有的方法,因为这两个方式是Object类中自带的。
  wait方法和notify方法不是通过线程对象调用,

  不是这样的:t.wait(),也不是这样的:t.notify()…不对。
  第二:wait()方法作用?
    Object o = new Object();
    o.wait();
  表示:
  让正在o对象上活动的线程进入等待状态,无期限等待,直到被唤醒为止。
  o.wait();方法的调用,会让“当前线程(正在o对象上活动的线程)”进入等待状态
  第三:notify()方法作用?
    Object o = new Object();
    o.notify();
  表示:
  唤醒正在o对象上等待的线程
  还有一个notifyAll()方法:
  这个方法是唤醒o对象上处于等待的所有线程。

Java並發程式設計之介紹線程安全基礎

10.生产者和消费者

Java並發程式設計之介紹線程安全基礎
1.使用wait方法和notify方法实现生产者和消费者模式
2.什么是生产者和消费者模式?
生产线程负责生产,消费线程负责消费
生产线程和消费线程要达到均衡
这是一种特殊的业务需求,在这种特殊的情况下需要使用wait方法和notify方法
3.wait和notify方法不是线程对象的方法,是普通java对象都有的方法
4.wait方法和notify方法是建立在线程同步的基础之上。因为多线程要同时操作一个仓库,有线程安全问题
5.wait方法作用:o.wait()让正在o对象上活动的线程t进入等待状态,并且释放掉t线程之前占有的o对象的锁
6.notify方法的作用:o.notify()让正在o对象上等待的线程唤醒,只是5通知,不会释放o对象上之前占有的锁
7.模拟这样一个需求:
仓库我们采用list集合
list集合中假设只能存储1个元素
1个元素就表示仓库满了
如果list集合中的元素个数是0,就表示仓库空了。
保证list集合中永远都是最多存储1个元素
必须做到这种效果,生产1个消费1个。

public class ThreadTest15 {
	public static void main(String[] args) {
		//创建一个仓库独享,共享的
		List list=new ArrayList();
		//创建两个线程对象
		//生产者线程
		Thread t1=new Thread(new Producer(list));
		//消费者线程
		Thread t2=new Thread(new Consumer(list));
		t1.setName("生产者线程");
		t2.setName("消费者线程");
		t1.start();
		t2.start();
	}}//生产线程class Producer implements Runnable{
	//仓库
	private List list;
	public Producer(List list) {
		this.list = list;
	}
	public void run() {
		//一直生产
		while(true){
			//给仓库对象list加锁
			synchronized (list) {
				if(list.size()>0){	//大于0说明仓库中已经有1个元素了
					//当前线程进入等待状态,并且释放list集合的锁
					try {
						list.wait();
					} catch (InterruptedException e) {
						// TODO Auto-generated catch block
						e.printStackTrace();
					}
				}
				//程序能够执行到这里说明仓库是空的,可以生产
				Object obj =new Object();
				list.add(obj);
				System.out.println(Thread.currentThread().getName()+"---->"+obj);
				//唤醒消费者进行消费
				list.notifyAll();
			}
			
		}
	}}//消费线程class Consumer implements Runnable{
	//仓库
	private List list;
	public Consumer(List list) {
		this.list = list;
	}
	public void run() {
		//一直消费
		while(true){
			//给仓库对象list加锁
			synchronized (list) {
				if(list.size()==0){	//仓里已经空了
					try {
						//仓库已经空了
						//消费者线程等待,并释放掉list集合锁
						list.wait();
					} catch (InterruptedException e) {
						// TODO Auto-generated catch block
						e.printStackTrace();
					}
				}
			}
			Object obj=list.remove(0);
			System.out.println(Thread.currentThread().getName()+"---->"+obj);
			//唤醒生产者生产
			list.notifyAll();
		}
	}}
登入後複製

相关学习推荐:java基础

以上是Java並發程式設計之介紹線程安全基礎的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

突破或從Java 8流返回? 突破或從Java 8流返回? Feb 07, 2025 pm 12:09 PM

Java 8引入了Stream API,提供了一種強大且表達力豐富的處理數據集合的方式。然而,使用Stream時,一個常見問題是:如何從forEach操作中中斷或返回? 傳統循環允許提前中斷或返回,但Stream的forEach方法並不直接支持這種方式。本文將解釋原因,並探討在Stream處理系統中實現提前終止的替代方法。 延伸閱讀: Java Stream API改進 理解Stream forEach forEach方法是一個終端操作,它對Stream中的每個元素執行一個操作。它的設計意圖是處

PHP:網絡開發的關鍵語言 PHP:網絡開發的關鍵語言 Apr 13, 2025 am 12:08 AM

PHP是一種廣泛應用於服務器端的腳本語言,特別適合web開發。 1.PHP可以嵌入HTML,處理HTTP請求和響應,支持多種數據庫。 2.PHP用於生成動態網頁內容,處理表單數據,訪問數據庫等,具有強大的社區支持和開源資源。 3.PHP是解釋型語言,執行過程包括詞法分析、語法分析、編譯和執行。 4.PHP可以與MySQL結合用於用戶註冊系統等高級應用。 5.調試PHP時,可使用error_reporting()和var_dump()等函數。 6.優化PHP代碼可通過緩存機制、優化數據庫查詢和使用內置函數。 7

PHP與Python:了解差異 PHP與Python:了解差異 Apr 11, 2025 am 12:15 AM

PHP和Python各有優勢,選擇應基於項目需求。 1.PHP適合web開發,語法簡單,執行效率高。 2.Python適用於數據科學和機器學習,語法簡潔,庫豐富。

PHP與其他語言:比較 PHP與其他語言:比較 Apr 13, 2025 am 12:19 AM

PHP適合web開發,特別是在快速開發和處理動態內容方面表現出色,但不擅長數據科學和企業級應用。與Python相比,PHP在web開發中更具優勢,但在數據科學領域不如Python;與Java相比,PHP在企業級應用中表現較差,但在web開發中更靈活;與JavaScript相比,PHP在後端開發中更簡潔,但在前端開發中不如JavaScript。

Java程序查找膠囊的體積 Java程序查找膠囊的體積 Feb 07, 2025 am 11:37 AM

膠囊是一種三維幾何圖形,由一個圓柱體和兩端各一個半球體組成。膠囊的體積可以通過將圓柱體的體積和兩端半球體的體積相加來計算。本教程將討論如何使用不同的方法在Java中計算給定膠囊的體積。 膠囊體積公式 膠囊體積的公式如下: 膠囊體積 = 圓柱體體積 兩個半球體體積 其中, r: 半球體的半徑。 h: 圓柱體的高度(不包括半球體)。 例子 1 輸入 半徑 = 5 單位 高度 = 10 單位 輸出 體積 = 1570.8 立方單位 解釋 使用公式計算體積: 體積 = π × r2 × h (4

PHP與Python:核心功能 PHP與Python:核心功能 Apr 13, 2025 am 12:16 AM

PHP和Python各有優勢,適合不同場景。 1.PHP適用於web開發,提供內置web服務器和豐富函數庫。 2.Python適合數據科學和機器學習,語法簡潔且有強大標準庫。選擇時應根據項目需求決定。

創造未來:零基礎的 Java 編程 創造未來:零基礎的 Java 編程 Oct 13, 2024 pm 01:32 PM

Java是熱門程式語言,適合初學者和經驗豐富的開發者學習。本教學從基礎概念出發,逐步深入解說進階主題。安裝Java開發工具包後,可透過建立簡單的「Hello,World!」程式來實踐程式設計。理解程式碼後,使用命令提示字元編譯並執行程序,控制台上將輸出「Hello,World!」。學習Java開啟了程式設計之旅,隨著掌握程度加深,可創建更複雜的應用程式。

PHP:許多網站的基礎 PHP:許多網站的基礎 Apr 13, 2025 am 12:07 AM

PHP成為許多網站首選技術棧的原因包括其易用性、強大社區支持和廣泛應用。 1)易於學習和使用,適合初學者。 2)擁有龐大的開發者社區,資源豐富。 3)廣泛應用於WordPress、Drupal等平台。 4)與Web服務器緊密集成,簡化開發部署。

See all articles