只要开启线程,都会开启一块对应的栈内存,然后进行同步执行.    -- 谈斌

线程是CPU用来处理程序的资源,线程的执行是抢占式的.

线程开启方式:

  1. 创建一个类,继承Thread类.
    重写 run(), 并在run方法内编写线程需要执行的内容

    创建

    /*
    * 线程开启方式1:
    * 1.创建一个类,继承Thread类.
    * 2.重写 run(), 并在run方法内编写线程需要执行的内容
    * */
    public class MyThread extends Thread {
    // 构造方法
    public MyThread() {
    super();
    } public MyThread(String name) {
    super(name);
    } // run方法里面写的是线程需要执行的内容
    @Override
    public void run() {
    for (int i = 0; i < 100; i++) {
    // 让线程睡2秒
    try {
    Thread.sleep(2000);
    } catch (InterruptedException e) { // 中途打断的意思
    // 输出异常的 名称,信息,位置
    e.printStackTrace();
    }
    System.out.println(getName()+"I Love Java");
    }
    }
    }
    /*
    * Runnable接口应该是被封装了开启cpu线程的C语言方法
    * */

    开启

    /*
    * 线程对象只能是
    * Thread类的对象或者是Thread类的子类对象
    *
    * 线程体系结构:
    * 根节点: Runnable()接口 --> 抽象方法run()
    * 实现类: Thread类
    * 想在执行线程任务的时候,带上线程名称:
    * String getName(): 获取线程名称
    * --> 直接调用getName()方法只能在Thread的子类中调用
    *
    * 想在执行任务的时候打印主线程的名称:(因为ThreadDemo01类不是Thread的子类)
    * Thread类中有:
    * static Thread currentThread():获取当前线程对象
    * void setName(String name):对线程对象的名字进行设置
    * static void sleep(long millis):让线程睡觉,睡多久: long millis决定
    * */
    public class ThreadDemo01 {
    public static void main(String[] args) throws InterruptedException {
    // 创建线程对象
    MyThread mt = new MyThread(); // 启动线程
    mt.start(); // run方法是由start方法帮忙调用的 for (int i = 0; i < 100; i++) {
    Thread.sleep(3000);
    System.out.println(Thread.currentThread().getName()+"I Love C^_^");
    }
    }
    }
  2. (把任务和线程分离) --> 推荐
    创建一个类, 实现Runnable接口
    并在这个类中重写run(), 写入线程需要执行的任务

    创建

    /*
    * 第二种线程开启方式: (把任务和线程分离) --> 推荐
    * 1. 创建一个类, 实现Runnable接口
    * 2. 并在这个类中重写run(), 写入线程需要执行的任务
    * */
    public class Target implements Runnable {
    @Override
    public void run() {
    for (int i = 0; i < 50; i++) {
    System.out.println(Thread.currentThread().getName() + ": Hello World !");
    }
    }
    }

    开启

    public class ThreadDemo02 {
    public static void main(String[] args) {
    // 创建线程的任务对象
    Target target = new Target(); // 任务
    Thread t0 = new Thread(target); // 雇佣兵
    // 设置线程名称
    t0.setName("stark");
    t0.start(); Thread t1 = new Thread(target); // 另一个雇佣兵
    // 两个雇佣兵在做相同的任务
    t1.setName("spider");
    t1.start(); // 匿名内部类开启线程方式,相当于开启方法一
    Thread t2 = new Thread(new Runnable() {
    @Override
    public void run() {
    for (int i = 0; i < 50; i++) {
    System.out.println(Thread.currentThread().getName()+": biu biu biu~");
    }
    }
    });
    t2.setName("captain");
    t2.start();
    }
    }

    我并不明白匿名内部类开启线程的方式算不算一种新的开启方式,你觉得呢?

     

线程安全问题 : 线程的抢占式执行机制, 造成了执行中出现的各种我们不想看到的问题.

  应对策略:   1.synchronized()

          2.Lock 锁

/*
* 同步代码块:
* synchronized(锁对象) {
* 你需要上锁(同步)的代码
* }
* 任意引用类型的对象,这个锁对象必须被所有的任务对象共享
*
* 总结:
* 同步:当一个线程正在执行任务的时候,不让其他线程进入!!
* 同步提高了代码的安全性,但是降低了代码的效率
* */
public class SellTicket implements Runnable{
// 成员变量
int ticket = 100; /*
* synchronized 里面可以放 Integer 却不能放 int 变量
* 这也许就是包装类吧
* */
Integer tic = new Integer(ticket); // 锁对象 --> 你请人帮你看门
Object obj = new Object(); @Override
public void run() {
// 电影院不关门
while (true) {
synchronized (this) {// 你进了厕所把门关了
if (ticket > 0) {
// 每半秒钟卖一张票
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 窗口名 + “卖了第” +ticket+ “张票~”
System.out.println(Thread.currentThread().getName() + "卖了第" + ticket + "张票~");
ticket--;
}
}
}
} /*
* 同步方法 --> 被 synchronized 修饰的方法
* 锁对象 this
*
* 静态同步方法 --> 锁对象: 类的字节码对象!
* */
// public synchronized void sell() {
// if (ticket > 0) {
// // 每半秒钟卖一张票
// try {
// Thread.sleep(500);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// // 窗口名 + “卖了第” +ticket+ “张票~”
// System.out.println(Thread.currentThread().getName() + "卖了第" + ticket + "张票~");
// ticket--;
// }
// }
}
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /*
* Lock 锁的使用步骤:
* 1. 创建锁对象:
* Lock lock = new ReentrantLock();
* 2. 在需要上锁的代码前面 上锁! --> lock.lock()
* */
public class SellTicket01 implements Runnable {
// 成员变量
int ticket = 100; /*
* synchronized 里面可以放 Integer 却不能放 int 变量
* 这也许就是包装类吧
* */
Integer tic = new Integer(ticket); // 创建锁对象
Lock lock = new ReentrantLock(); @Override
public void run() {
// 电影院不关门
while (true) {
lock.lock();
if (ticket > 0) {
try {
// 每半秒钟卖一张票
Thread.sleep(500);
// 窗口名 + “卖了第” +ticket+ “张票~”
System.out.println(Thread.currentThread().getName() + "卖了第" + ticket + "张票~");
ticket--;
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
}
}
}

最后附上线程的生老病死图, 希望大家月薪过万的时刻指日可待 ^^

 

1年之后的拿高工资的资本,Java线程的更多相关文章

  1. Python工资高还是Java?

    说起来,随着人工智能和大数据逐渐进入人们的眼中,越来越多的人看到互联网未来大好发展趋势,而想要学习一门技术来进入其中,以期分一杯羹.但是,作为人工智能和大数据的重要编程语言,Python和Java,该 ...

  2. 经过本人 6 年.net 工作经验证明 .net 工资确实比 Java 低

    很久没有逛博客园了,很久没有出来吐槽了.怎么呢?生活压力大,就需要找地方吐槽. 排版不怎么好,文笔不怎么好,可以看出我不是雇佣的写手. 工作到今夏,已经有6个年头了,一直使用着.net技术,但是最近工 ...

  3. 【转】这些编程语言程序员工资最高!Java才第四

    原文网址:http://tech.hexun.com/2016-07-18/185009761.html 在众多行业中,程序员属于高薪职业.无论是在国外还是国内,程序员的薪金水平普遍高于其他行业的工作 ...

  4. java 设计模式之模板方法

    模板方法的定义 定义了一个算法的骨架,并允许子类为一个或多个步骤提供实现. 模板方法使得子类在不改变算法结构的情况下,重新定义某些算法的步骤. 一次性实现一个算法不变的部分,把可变的行为留给子类实现. ...

  5. 为什么你作为一个.NET的程序员工资那么低?

    最近看到很多抱怨贴,也许有一定的道理,但是你想过没,为什么大部分.NET程序员工资相对低?我个人是这么看的: 大批半罐子水的程序员,永远被局限在.NET的原始的小圈圈里.前端不会(你放弃了一项很重要的 ...

  6. 为什么你作为一个.NET的程序员工资那么低?(转)

    最近看到很多抱怨贴,也许有一定的道理,但是你想过没,为什么大部分.NET程序员工资相对低?我个人是这么看的: 大批半罐子水的程序员,永远被局限在.NET的原始的小圈圈里.前端不会(你放弃了一项很重要的 ...

  7. Java面试大纲-java面试该做哪些准备,java开发达到这样的水平可以涨工资

    Java培训结束,面临的就是毕业找工作.在找工作时,就要针对性地做充分的面试准备.准备不充分的面试,完全是浪费时间,更是对自己的不负责. 上海尚学堂Java培训整理出Java面试大纲,其中大部分都是面 ...

  8. 我也谈谈.NET程序员工资低

    我从2011年下半年预谋转型,2012春季正式转型到iOS,看了<经过本人 6 年.net 工作经验证明 .net 工资确实比 Java 低>这篇文章,一下子有很多感慨. 我不好意思算我干 ...

  9. 【笔试题】某公司中有N名员工。给定所有员工工资的清单

    排列员工工资顺序(C++map解法) 题目描述:某公司中有N名员工.给定所有员工工资的清单,财务人员要按照特定的顺序排列员工的工资.他按照工资的频次降序排列.即给定清单中所有频次较高的工资将在频次较低 ...

随机推荐

  1. Windows Server 2016 Storage Replication

    Storage Replication是Windows Server 2016中新增的一项功能,它是利用windows server自带的块存储复制技术 首先,我们简答粗暴的交代一下部署需求: 1.该 ...

  2. Java基础】并发 - 多线程

    Java基础]并发 - 多线程 分类: Java2014-05-03 23:56 275人阅读 评论(0) 收藏 举报 Java   目录(?)[+]   介绍 Java多线程 多线程任务执行 大多数 ...

  3. SolrCloud(solr集群+zookeeper集群)

    一.集群介绍 1.  什么是SolrCloud SolrCloud(solr 云)是Solr提供的分布式搜索方案,当你需要大规模,容错,分布式索引和检索能力时使用 SolrCloud.当一个系统的索引 ...

  4. AJ学IOS 之微博项目实战(13)发送微博调用相机里面的图片以及调用相机

    AJ分享,必须精品 一:效果 二:代码 相机部分就简单多了,几行代码调用而已,但是如果你要是想实现更多丰富的功能,需要自己写.利用AssetsLibrary.framework,利用这个框架可以获得手 ...

  5. 手把手教Extjs-简单GridField示例讲解二

    使用的Extjs版本为4.2,示例是官方的版本,对里面的语法进行一句一句的学习研究.可以方便他人,又可以提升自己的理解.里面存在的问题,后期会一步一步改进.也欢迎各位指出. /* Extjs具有很庞大 ...

  6. golang slice 源码解读

    本文从源码角度学习 golang slice 的创建.扩容,深拷贝的实现. 内部数据结构 slice 仅有三个字段,其中array 是保存数据的部分,len 字段为长度,cap 为容量. type s ...

  7. A - Free DIY Tour HDU - 1224

    题目大意:每一个城市都有一定的魅力值,然后有一个有向图,根据这个有向图从1到n+1所获得的魅力的最大值,并输出路径(要求只能从编号娇小的城市到编号较大的城市). 题解:很容易想到最短路+路径纪录.但是 ...

  8. 腾讯云集群服务部署mysql并挂载到服务器

    一.背景 由于现在大部分的应用都是运行在云服务器上的,而现在大多数文章都是主要写如何在服务器上使用docker去运行mysql,比较少有介绍云服务器上的.再加上现在k8s比较火爆,而云厂商大多数都提供 ...

  9. winfrom 基础

    1 winfrom就是一种窗体开发端应用程序 2 窗体分类 1)记事本类:可以最大最小化,可以拖拽                                                窗体默 ...

  10. Acid靶机渗透

    Acid渗透靶机实战 攻击机:kali 192.168.41.147 靶机: acid 192.168.41.149 信息收集 ip发现 开启Acid靶机,通过nmap进行局域网存火主机扫描.![]( ...