一个由try...catch...finally引出的思考,在前面已经初步了解过不可变与可变、值传递与引用传递,在这里再次深入理解。

1.先看下面一个try..catch..finally的例子:

Person.java

  1. package cn.qlq.test;
  2.  
  3. public class Person {
  4. private int age;
  5. private String name;
  6.  
  7. public int getAge() {
  8. return age;
  9. }
  10.  
  11. public void setAge(int age) {
  12. this.age = age;
  13. }
  14.  
  15. public String getName() {
  16. return name;
  17. }
  18.  
  19. public void setName(String name) {
  20. this.name = name;
  21. }
  22.  
  23. @Override
  24. public String toString() {
  25. return "Person [age=" + age + ", name=" + name + "]";
  26. }
  27.  
  28. }
  1. package cn.qlq.test;
  2.  
  3. public class FinallyTest {
  4.  
  5. public static void main(String[] args) {
  6. System.out.println(test1());
  7. System.out.println(test2());
  8. }
  9.  
  10. public static String test1() {
  11. String s = "s1";
  12. try {
  13. int i = 1 / 0;
  14. s = "s2";
  15. return s;
  16. } catch (Exception e) {
  17. s = "s3";
  18. return s;
  19. } finally {
  20. s = "s4";
  21. }
  22. }
  23.  
  24. public static Person test2() {
  25. Person p = new Person();
  26. p.setName("old");
  27. try {
  28. int i = 1 / 0;
  29. return p;
  30. } catch (Exception e) {
  31. p.setName("exception");
  32. return p;
  33. } finally {
  34. p.setName("finally");
  35. }
  36. }
  37. }

结果:

s3
Person [age=0, name=finally]

总结:

  finally块的语句在try或catch中的return语句执行之后返回之前执行且finally里的修改语句可能影响也可能不影响try或catch中return已经确定的返回值,如果返回值类型为传址类型,则影响;传值类型(8种基本类型)与8种基本数据类型的包装类型与String(不可变类)不影响。若finally里也有return语句则覆盖try或catch中的return语句直接返回。

面试宝典解释的原因如下:

  程序在执行到return时首先会把返回值存到一个指定的位置(JVM中的slot),其次与执行finally块,最后再返回。如果finally中有return语句会以finally的return为主,相当于普通程序中的return结束函数。如果没有return语句,则会在finally执行完之后弹出slot存储的结果值并且返回,如果是引用类型则finally修改会影响结果,如果是基本数据类型或者不可变类不会影响返回结果。

补充:两个例子对上面很好的解释

(1)对不可变类不影响

  1. package cn.xm.exam.test;
  2.  
  3. public class Test2 {
  4. public static void main(String[] args) {
  5. System.out.println(changeInteger());
  6. }
  7.  
  8. private static Integer changeInteger() {
  9. Integer result = 1;
  10. try {
  11. int i = 1 / 0;
  12. return result;
  13. } catch (Exception e) {
  14. result = 2;
  15. return result;
  16. } finally {
  17. result = 3;
  18. }
  19. }
  20.  
  21. }

结果:  2

  1. public static void main(String[] args) {
  2. System.out.println(getValue());
  3. }
  4.  
  5. @SuppressWarnings("finally")
  6. private static int getValue() {
  7. try {
  8. return 0;
  9. } finally {
  10. return 1;
  11. }
  12. }

结果:

1

(2)对引用类型可变类影响结果

  1. package cn.xm.exam.test;
  2.  
  3. public class Test2 {
  4. public static void main(String[] args) {
  5. System.out.println(changeInteger());
  6. }
  7.  
  8. private static StringBuilder changeInteger() {
  9. StringBuilder stringBuilder = new StringBuilder();
  10. try {
  11. int i = 1 / 0;
  12. stringBuilder.append("1");
  13. return stringBuilder;
  14. } catch (Exception e) {
  15. stringBuilder.append("2");
  16. return stringBuilder;
  17. } finally {
  18. stringBuilder.append("3");
  19. }
  20. }
  21.  
  22. }

结果:

23

2.值传递与引用传递

1)值传递:方法调用时,实际参数把它的值传递给对应的形式参数,形式参数只是用实际参数的值初始化自己的存储单元内容,是两个不同的存储单元,所以方法执行中形式参数值的改变不影响实际参数的值。

2)引用传递(指针传递):也称为传地址。方法调用时,实际参数是对象(或数组),这时实际参数与形式参数指向同一个地址,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,所以方法执行中形式参数的改变将会影响实际参数。引用类型如果另一个采用new之后两者会指向不同的对象,也就不会再关联。

注意:

在Java中,原始数据类型在传递参数时都是按值传递,而包装类型在传递参数是是按引用传递,但包装类型在进行计算的时候会自动拆箱。

对象在函数调用传参的时候是引用传递(基本数据类型值传递),"="赋值也是引用传递(基本数据类型值传递)。

1.Integer采用引用传递

  由于8种基本数据类型和String的不可变性,加大了引用传值的理解程度,误认为"8种包装类型是“值传递",下面进行实例:

  1. public static void main(String[] args) {
  2. Integer a = 5;
  3. Integer b = a;
  4. b++;
  5. System.out.println(a);//
  6.  
  7. String s1="s1";
  8. String s2 = s1;
  9. s2 = "s2";
  10. System.out.println(s1);//s1
  11. }

  解释:实际Integer和String是采用引用传递,=的时候a和b,s1和s2指向同一个对象。执行b++之后由于Integer的不可变性,b指向一个新的对象,b与a已经没有关系;s2="s2"之后s2指向一个新的对象,也与s1没关系。

为了验证Integer是采用引用传递,我门做案例如下:

  1. package cn.qlq.test;
  2.  
  3. import java.util.concurrent.TimeUnit;
  4.  
  5. public class IntegerSyn {
  6.  
  7. public static void main(String[] args) throws InterruptedException {
  8. Integer index = 0;
  9. TestObject a = new TestObject(index);
  10. synchronized (index) {
  11. new Thread(a).start();
  12. index.wait();
  13. }
  14. System.out.println("end");
  15. }
  16. }
  17.  
  18. class TestObject implements Runnable {
  19. private Integer index;
  20.  
  21. public TestObject(Integer index){
  22. this.index = index;
  23. }
  24.  
  25. public void run() {
  26. try {
  27. //线程休眠的另一种方法
  28. TimeUnit.SECONDS.sleep(5);
  29. } catch (InterruptedException e) {
  30. e.printStackTrace();
  31. }
  32. synchronized (index) {
  33. index.notify();
  34. }
  35. }
  36. }

5s后打印end

解释:  在程序刚启动的时候把 Integer 的index 对象锁住 ,并且调用了 wait方法,释放了锁的资源,等待notify,最后过了5秒钟,等待testObject 调用notify 方法就继续执行了。大家都知道锁的对象和释放的对象必须是同一个,否则会抛出  java.lang.IllegalMonitorStateException 。由此可以证明 Integer作为参数传递的时候是地址传递,而非值传递。

2.数组采用引用传值

  其实数组也是对象类型,传递的时候也是采用引用传递,只是因为基本数据类型数的不可变性也增大了理解难度,例如:

  1. package cn.qlq.test;
  2.  
  3. import java.util.Arrays;
  4.  
  5. public class ArrayTest {
  6. public static void main(String[] args) {
  7. int a[] = { 10, 20 };
  8. test(a);
  9. System.out.println(Arrays.toString(a));
  10. }
  11.  
  12. public static void test(int arr[]) {
  13. arr[0] = 100;
  14. }
  15. }

结果:

[100, 20]

  数组实际是引用传递。(这点必须理解,因为String的不可变是基于char[]与深复制实现)。实际上数组是基于引用传递,不管是基本数据类型数组还是包装类型数组都是引用传递。测试代码如下:

  1. package cn.qlq.test;
  2.  
  3. import java.util.concurrent.TimeUnit;
  4.  
  5. public class ArraySyn {
  6.  
  7. public static void main(String[] args) throws InterruptedException {
  8. int index[] = {1,2};
  9. TestObject1 a = new TestObject1(index);
  10. synchronized (index) {
  11. new Thread(a).start();
  12. index.wait();
  13. }
  14. System.out.println("end");
  15. }
  16. }
  17.  
  18. class TestObject1 implements Runnable {
  19. private int[] index;
  20.  
  21. public TestObject1(int []index){
  22. this.index = index;
  23. }
  24.  
  25. public void run() {
  26. try {
  27. //线程休眠的另一种方法
  28. TimeUnit.SECONDS.sleep(5);
  29. } catch (InterruptedException e) {
  30. e.printStackTrace();
  31. }
  32. synchronized (index) {
  33. index.notify();
  34. }
  35. }
  36. }

5s后打印end,证明是引用传递。

另一种测试方法:传递引用类型数组

  1. package cn.qlq.test;
  2.  
  3. import java.util.Arrays;
  4.  
  5. public class ArrayTest {
  6. public static void main(String[] args) {
  7. Person[] p = new Person[2];
  8. Person p1 = new Person();
  9. p1.setName("p1");
  10. Person p2 = new Person();
  11. p2.setName("p2");
  12. p[0] = p1;
  13. p[1] = p2;
  14. testArr(p);
  15. System.out.println(Arrays.toString(p));
  16. }
  17.  
  18. private static void testArr(Person[] p) {
  19. p[0].setName("p1p1");
  20. }
  21. }

结果:

[Person [age=0, name=p1p1], Person [age=0, name=p2]]

总结一条:

  8种基本数据类型是值传递,8种基本数据类型与String与数组是引用传递,我们程序中的类也是引用传递,但是由于String与8种基本数据类型的不可变性,所以每次赋予新值的时候都是新指向一个对象。如果是函数调用是形参和实参指向同一个对象,所以改变实参的时候相当于新创一个对象并赋给形参,对实参不会造成影响。

3.String引用传递图解

更进一步的理解:"引用传值也是按值传递,只不过传的是对象的地址"。

比如下面一段代码:

  1. package cn.qlq.test;
  2.  
  3. import java.util.Arrays;
  4.  
  5. public class ArrayTest {
  6. public static void main(String[] args) {
  7. String s = "hello";
  8. System.out.println(s.hashCode());
  9.  
  10. test(s);
  11. System.out.println(s);
  12. }
  13.  
  14. public static void test(String s1) {
  15. System.out.println(s1.hashCode());
  16. s1 = "world";
  17. System.out.println(s1.hashCode());
  18. }
  19. }

结果:

99162322
99162322
113318802
hello

解释:调用test方法的时候采用引用传递(将s的地址传下去),执行s1="world"是新创一个"world"并赋值给s1,也就是s1此时已经指向其他对象,不再与s指向相同对象。

图解:

  

4.补充:

  引用类型如果另一个采用 = 赋值 改变引用之后两者会指向不同的对象(因为这个纠结了一下午,new相当于新创一个对象并且赋值给该变量,如果不想让其new可以采用final限制为引用不可变),而且将一个静态变量赋值给局部变量的时候改变局部变量的值也会影响static变量的值,所以如果要定义真正的不可变对象可以用final变量。

Person.java

  1. class Person {
  2. public static Person person = new Person("1", 1);
  3. private String name = "zhangsan";
  4. private int age = 25;
  5.  
  6. @Override
  7. public String toString() {
  8. return "Person [name=" + name + ", age=" + age + "]";
  9. }
  10.  
  11. public String getName() {
  12. return name;
  13. }
  14.  
  15. public void setName(String name) {
  16. this.name = name;
  17. }
  18.  
  19. public int getAge() {
  20. return age;
  21. }
  22.  
  23. public void setAge(int age) {
  24. this.age = age;
  25. }
  26.  
  27. public Person(String name, int age) {
  28. super();
  29. this.name = name;
  30. this.age = age;
  31. }
  32. }

(1)测试一:测试new不影响原对象

  1. /**
  2. * new之后是新创建一个对象并赋值给该变量,不会影响原来的变量
  3. */
  4. private static void test3() {
  5. Person p1 = Person.person;
  6. System.out.println("p1:"+p1);
  7. System.out.println("Person.person:"+Person.person);
  8. p1 = new Person("2", 2);
  9. System.out.println("--------p1重新new之后的值------");
  10. System.out.println("p1:"+p1);
  11. System.out.println("Person.person:"+Person.person);
  12. }

结果:

p1:Person [name=1, age=1]
Person.person:Person [name=1, age=1]
--------p1重新new之后的值------
p1:Person [name=2, age=2]
Person.person:Person [name=1, age=1]

(2)将static变量引用传递给局部变量,改变局部变量也可以影响static变量

  1. /**
  2. * 修改一个变量会影响static变量对应引用类型的值
  3. */
  4. private static void test4() {
  5. Person p1 = Person.person;
  6. System.out.println("p1:" + p1);
  7. System.out.println("Person.person:" + Person.person);
  8. p1.setName("20");
  9. p1.setAge(20);
  10. System.out.println("--------p1改变值之后------");
  11. System.out.println("p1:" + p1);
  12. System.out.println("Person.person:" + Person.person);
  13. }

结果:

p1:Person [name=1, age=1]
Person.person:Person [name=1, age=1]
--------p1改变值之后------
p1:Person [name=20, age=20]
Person.person:Person [name=20, age=20]

(3)测试2:(这个例子更加的可以理解)

  1. public class PlainTest {
  2. private Person per = Person.person;
  3.  
  4. public static void main(String[] args) {
  5. test1();
  6. }
  7.  
  8. private static void test1() {
  9. PlainTest p1 = new PlainTest();
  10. Person.person = new Person("2", 2);
  11. System.out.println("Person.person:" + Person.person);
  12. System.out.println("p1的per:" + p1.per);
  13.  
  14. PlainTest p2 = new PlainTest();
  15. System.out.println("p2的per:" + p2.per);
  16. }
  17. }

结果:

Person.person:Person [name=2, age=2]
p1的per:Person [name=1, age=1]
p2的per:Person [name=2, age=2]

test1解释:

  PlainTest p1 = new PlainTest();此时其实例变量per与Person的person指向堆中同一个对象。

  Person.person = new Person("2", 2);  此时新出那个键一个对象并且Person的person会指向该对象。但是p1.per指向的地址还是原来的地址。

  PlainTest p2 = new PlainTest();   此时新建一个PlainTest2,其成员变量与上面的Person.person指向同一对象(修改后的对象)

(4)测试3:(与上面例子3结合理解更好)

  1. public class PlainTest {
  2. private Person per = Person.person;
  3.  
  4. public static void main(String[] args) {
  5. test2();
  6. }private static void test2() {
  7. PlainTest p1 = new PlainTest();
  8. Person.person.setName("2");
  9. System.out.println("Person.person:" + Person.person);
  10. System.out.println("p1的per:" + p1.per);
  11.  
  12. PlainTest p2 = new PlainTest();
  13. System.out.println("p2的per:" + p2.per);
  14. }
  15. }

结果:

Person.person:Person [name=2, age=1]
p1的per:Person [name=2, age=1]
p2的per:Person [name=2, age=1]

  这个例子很好解释,三个person都指向同一对象,所以修改任何一个都会影响剩下两个。

5.如果一个对象被赋予null值,也相当于与原来的对象脱离关系,被赋予null会孤立原来堆中的对象,也就是会被GC,前提是原来堆中的对象没有被其他变量引用

  1. public static void main(String[] args) {
  2. Person p = new Person("张三", 0);
  3. Person p1 = p;
  4. p=null;
  5. System.out.println(p1+"\t"+p1.hashCode());
  6. System.out.println(p+"\t"+p.hashCode());
  7. }

结果:(被重新赋值为null不会影响与之指向同一对象的引用,只是自己不再指向堆中的对象。)

Person [name=张三, age=0] 1605870897
Exception in thread "main" java.lang.NullPointerException
at cn.qlq.test.PlainTest.main(PlainTest.java:11)

  

3.可变类与不可变类

不可变类:所谓的不可变类是指这个类的实例一旦创建完成后,就不能改变其成员变量值。如JDK内部自带的很多不可变类:Interger、Long和String(8种基本数据类型的包装类和String都是不可变类)等。不可变类的意思是一旦这个对象创建之后其引用不会改变,每次重新赋值会新增一个对象。不可变类是实例创建后就不可以改变成员遍历的值。这种特性使得不可变类提供了线程安全的特性但同时也带来了对象创建的开销,每更改一个属性都是重新创建一 个新的对象。例如String s = "s1",s = "s2"实际上是创建了两个对象,第二次将其值指向新的"s2".。

可变类:相对于不可变类,可变类创建实例后可以改变其成员变量值,开发中创建的大部分类都属于可变类。

关于更详细的介绍参考:https://www.cnblogs.com/qlqwjy/p/7944456.html

在这里我们只需要明白8种基本数据类型的包装类和String类型是不可变类,其余我们程序中的大部分类都是可变类。

不可变类的设计原则:

  1. 1. 类添加final修饰符,保证类不被继承。
  2.     如果类可以被继承会破坏类的不可变性机制,只要继承类覆盖父类的方法并且继承类可以改变成员变量值,那么一旦子类以父类的形式出现时,不能保证当前类是否可变。
  3.  
  4. 2. 保证所有成员变量必须私有,并且加上final修饰(不可变指的是引用不可变,也就是不可以重新指向其他对象)
  5.     通过这种方式保证成员变量不可改变。但只做到这一步还不够,因为如果是对象成员变量有可能再外部改变其值。所以第4点弥补这个不足。
  6.  
  7. 3. 不提供改变成员变量的方法,包括setter
  8.     避免通过其他接口改变成员变量的值,破坏不可变特性。
  9.  
  10. 4.通过构造器初始化所有成员,进行深拷贝(deep copy)
  11.  
  12. 如果构造器传入的对象直接赋值给成员变量,还是可以通过对传入对象的修改进而导致改变内部变量的值。例如:
  13.  
  14. public final class ImmutableDemo {
  15. private final int[] myArray;
  16. public ImmutableDemo(int[] array) {
  17. this.myArray = array; // wrong
  18. }
  19. }
  20. 这种方式不能保证不可变性,myArrayarray指向同一块内存地址,用户可以在ImmutableDemo之外通过修改array对象的值来改变myArray内部的值。
  21. 为了保证内部的值不被修改,可以采用深度copy来创建一个新内存保存传入的值。正确做法:
  22.  
  23. public final class MyImmutableDemo {
  24. private final int[] myArray;
  25. public MyImmutableDemo(int[] array) {
  26. this.myArray = array.clone();
  27. }
  28. }
  29. 5. getter方法中,不要直接返回对象本身,而是克隆对象,并返回对象的拷贝
  30.     这种做法也是防止对象外泄,防止通过getter获得内部可变成员对象后对成员变量直接操作,导致成员变量发生改变

string对象在内存创建后就不可改变,不可变对象的创建一般满足以上5个原则,我们看看String代码是如何实现的。

  1. public final class String
  2. implements java.io.Serializable, Comparable<String>, CharSequence
  3. {
  4. /** The value is used for character storage. */
  5. private final char value[];//数组是引用传递
  6. /** The offset is the first index of the storage that is used. */
  7. private final int offset;
  8. /** The count is the number of characters in the String. */
  9. private final int count;
  10. /** Cache the hash code for the string */
  11. private int hash; // Default to 0
  12. ....
  13. public String(char value[]) {
  14. this.value = Arrays.copyOf(value, value.length); // deep copy操作
  15. }
  16. ...
  17. public char[] toCharArray() {
  18. // Cannot use Arrays.copyOf because of class initialization order issues
  19. char result[] = new char[value.length];
  20. System.arraycopy(value, 0, result, 0, value.length);
  21. return result;
  22. }
  23. ...
  24. }

如上代码所示,可以观察到以下设计细节:

  1. String类被final修饰,不可继承
  2. string内部所有成员都设置为私有变量
  3. 不存在value的setter
  4. 并将value和offset设置为final。
  5. 当传入可变数组value[]时,进行copy而不是直接将value[]复制给内部变量.
  6. 获取value时不是直接返回对象引用,而是返回对象的copy.

这都符合上面总结的不变类型的特性,也保证了String类型是不可变的类。

补充:深复制与浅赋值区别:

浅复制:被赋值的对象与原对象都含有相同的值,而所有对其他对象的引用仍然指向原来的对象。换言之,浅复制仅仅赋值所考虑的对象,而不复制它所引用的对象。

深赋值:被复制的对象的所有变量都有与原对象相同的值,除去那些引用其他对象的变量。那些引用其他对象的变量将指向被赋值的新对象,而不再是原来的那些被引用的对象。换言之,深复制把复制的对象所引用的对象都复制了一遍。

如下图:

  总结:

  (1)关于finally:

  finally块的语句在try或catch中的return语句执行之后返回之前执行且finally里的修改语句可能影响也可能不影响try或catch中return已经确定的返回值,如果返回值类型为传址类型,则影响;传值类型(8种基本类型)与8种基本数据类型的包装类型与String(不可变类)不影响。若finally里也有return语句则覆盖try或catch中的return语句直接返回,相当于普通流程中的return语句。

面试宝典解释的原因如下:

  程序在执行到return时首先会把返回值存到一个指定的位置(JVM中的slot),其次与执行finally块,最后再返回。如果finally中有return语句会以finally的return为主,相当于普通程序中的return结束函数。如果没有return语句,则会在finally执行完之后弹出slot存储的结果值并且返回,如果是引用类型则finally修改会影响结果,如果是基本数据类型或者不可变类不会影响返回结果。

  (2)值传递与引用传递:

    =与函数调用是引用传递,8种基本数据类型采用值传递,其包装类型与String与其他我们手写的类都是引用传递。只是由于String和8种包装类型都是不可变类,所以每次操作都是新创一个对象并重新赋给引用;在函数调用的时候,如果形参是String或者8种包装类型,操作形参不会影响实参,操作形参相当于重新创建对象不会影响原实参。

  

  (3)可变与不可变

    String与8种包装类型、BigInteger、BigDecimal是不可变类,不可变的意思是每次更换值都会重新生成对象并赋给引用。不用考虑线程安全。我们也可以设计自己的不可变类。

    其他我们手写的一般是可变类。

  

JAVA不可变类与可变类、值传递与引用传递深入理解的更多相关文章

  1. Java中的值传递与引用传递

    1.基本类型和引用类型在内存中的保存 Java中数据类型分为两大类,基本类型和对象类型.相应的,变量也有两种类型:基本类型和引用类型. 基本类型的变量保存原始值,即它代表的值就是数值本身: 而引用类型 ...

  2. 第002弹:Java 中的值传递和引用传递

    在 Java 的代码开发过程中,为了尽可能提高方法的复用性,明确方法的作用,同时防止一个方法内部过于臃肿的问题,往往会创建许多方法,那么不可避免地会涉及到参数传递的问题.通常来说,我们将 Java 中 ...

  3. java 传参方式--值传递还是引用传递

    java 传参方式--值传递还是引用传递 参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递.写它是为了揭穿普遍存在的一种神话,即认为 Java 应用程序按引用 ...

  4. 一个随意list引发的惨案(java到底是值传递还是引用 传递?)

    前两天写了一个递归,因为太年轻,把一个递归方法需要用到的list定义该递归方法外了,结果开始断点测试的时候有点小问题 ,然后上线之后因为数据量太多导致了一个java.util.ConcurrentMo ...

  5. Java的参数传递是值传递还是引用传递?

    一.前言 首先先说结论,Java中方法参数传递方式是按值传递.如果参数是基本类型,传递的是基本类型的字面量值的拷贝.如果参数是引用类型,传递的是该参量所引用的对象在堆中地址值的拷贝. 接下来深入了解一 ...

  6. Java中引用类型变量,对象,值类型,值传递,引用传递 区别与定义

    一.Java中什么叫做引用类型变量?引用:就是按内存地址查询       比如:String s = new String();这个其实是在栈内存里分配一块内存空间为s,在堆内存里new了一个Stri ...

  7. java的值传递和引用传递

    昨天博主在对于值传递和引用传递这里栽了一个大坑啊,导致一下午时间都浪费在这里,我们先说下值传递和引用传递java官方解释: 值传递:(形式参数类型是基本数据类型):方法调用时,实际参数把它的值传递给对 ...

  8. 理解java值传递与引用传递

    1.基本类型和引用类型在内存中的保存 Java中数据类型分为两大类,基本类型和对象类型.相应的,变量也有两种类型:基本类型和引用类型.基本类型的变量保存原始值,即它代表的值就是数值本身:而引用类型的变 ...

  9. 一道笔试题来理顺Java中的值传递和引用传递

      题目如下: private static void change(StringBuffer str11, StringBuffer str12) { str12 = str11; str11 = ...

随机推荐

  1. 团队博客 Week14

    0. 在吹牛之前,先回答这个问题: 如果你的团队来了一个新队员,有一台全新的机器, 你们是否有一个文档,只要设置了相应的权限,她就可以根据文档,从头开始搭建环境,并成功地把最新.最稳定版本的软件编译出 ...

  2. visual studio-2013之单元测试

    安装个vs把一个寝室搞得欲仙欲死的,,已经安装好了vs2013,那些欲仙欲死的事就都不说了.开始我们的正式作业——单元测试. 要做单元测试前提有: 1.要有Unit Test Generator工具 ...

  3. Visual Studio2013安装过程

    Visual Studio是微软开发的一套基于组件的软件开发工具,我选择安装的是Visual Studio2013版本.首先, 第一步是要找到一个安装包: 我们可以直接百度MSDN,显示的第一条就是官 ...

  4. AD分辨率和精度区别

    最近做了一块板子,当然考虑到元器件的选型了,由于指标中要求精度比较高,所以对于AD的选型很慎重.很多人对于精度和分辨率的概念不清楚,这里我做一下总结,希望大家不要混淆.我们搞电子开发的,经常跟“精度” ...

  5. 从零开始学Kotlin-数据类型(2)

    从零开始学Kotlin基础篇系列文章 基本数据类型 Kotlin 的基本数值类型包括 Byte.Short.Int.Long.Float.Double 等: 数据-------位宽度 Double-- ...

  6. java 封装,继承,多态基础

    什么是封装? 1,对象数据和在.操作该对象的指令都是对象自身的一部分,能够实现尽可能对外部隐藏数据. 2,实际项目开发中,使用封装最多的就是实体类. 什么是继承? 1,继承是面向对象程序设计能提高效率 ...

  7. T检验在项目上的具体实施

    我觉得 T 检验,应该用在 判断某种仿真条件因素 对碳纳米管的随机性 是否有显著影响 上.所以不是针对<相同仿真条件对不同源的影响>这个表中的数据做 T 检验 如:判断 金属/半导体比率 ...

  8. Win2008r2 设置 多用户同时远程

    Study From http://blog.sina.com.cn/s/blog_7ebe66420101tfln.html 1. 启动远程桌面,关闭防火墙 略过不提 2. 添加远程服务角色, 打开 ...

  9. CentOS7 安装redis 并且设置成服务自动启动

    通过 博客园 https://www.cnblogs.com/zuidongfeng/p/8032505.html 学习以及记录 1. 下载redis 现在最新的stable版本是 4.0.10 wg ...

  10. Hbase远程连接:Can't get the locations

    当Java API远程连接出错:Can't get the locations 原先填入的是IP地址,后来改为HOSTS文件中配置的主机名问题解决,如下红色字体部分: conf.set("h ...