A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

©   /  2013-3-22 16:48  /  2186 人查看  /  8 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

看看我的学习日志,希望对你有帮助

3.6   static(静态)关键字
static(静态)关键字:
用于修饰成员(成员变量和成员函数)
被修饰后的成员有以下特点:
•随着类的加载而加载
        也就说:静态会随着类的消失而消失。说明它的生命周期最长。
•优先于对象存在
        明确一点:静态是先存在。对象是后存在的。
•被所有对象所共享
•可以直接被类名调用
静态方法使用注意:
1,静态方法只能访问静态成员。
        非静态方法既可以访问静态也可以访问非静态。
2,静态方法中不可以定义this,super关键字。
        因为静态优先于对象存在。所以静态方法中不可以出现this。
3,主函数是静态的。
实例变量和类变量的区别:
1,存放位置。
        类变量随着类的加载而存在于方法区中。
        实例变量随着对象的建立而存在于堆内存中。
2,生命周期:
        类变量生命周期最长,随着类的消失而消失。
        实例变量生命周期随着对象的消失而消失。
静态的利弊
利处:对对象的共享数据进行单独空间的存储,节省空间。没有必要每一个对象中都存储一份。可以直接被类名调用。
弊端:生命周期过长。访问出现局限性。(静态虽好,但是只能访问静态。)
什么使用静态?
要从两方面下手:
因为静态修饰的内容有成员变量和函数。分析这两种情况。
什么时候定义静态变量(类变量)呢?
当对象中出现共享数据时,该数据被静态所修饰。(所有对象都有这个共性所以共享)
对象中的特有数据要定义成非静态存在于堆内存中。(就是每个对象不同的没有对象的内容,不能定义成静态)
什么时候定义静态函数呢?(静态函数只能定义静态)
当功能内部没有访问到非静态数据(对象的特有数据,如不同人的名字),那么该功能可以定义成静态的。
演示代码:
class Person
{
        String name;//成员变量,实例变量。没有被静态修饰,对象建立前不存在
        static String country = "CN";//静态的成员变量,属于类变量。因为每个Person都有CN的国籍,所以直接把这个变量被静态修饰,这样被所用对象共享,随着类的加载而加载并且优先于对象存在(对象建立前就有,建立后随着对象加载而加载),可以直接被类名调用
        public static void show()//被静态修饰的方法。为什么可以用静态修饰,因为功能内部没有访问到非静态数据(对象的特有数据,如不同人的名字),如果下面的输出语句里面有name值的输出,函数就不能用static修饰
        {
                System.out.println("::::");//括号内不能+name如("::::"+name),这样语法错误。因为name默认为this.name,this.super语句不能用于静态语句中。原因是静态在String name之前存在于方法区之中,不能调用非静态的变量。
                //this.haha();语法错误,因为this.super语句不能用于静态语句中。
        }
        public void haha()
        {}
}

class  StaticDemo
{
        public static void main(String[] args)
        {
                Person p = new Person();
                //p.name = "zhangsan";
                //p.show();
                //System.out.println(p.country);
                System.out.println(Person.country); //没有对象,但是county是静态变量,所以可以直接被类名调用。
                Person.show();//匿名函数调用
        }
}
回复 使用道具 举报
还有一些~~~

扩展知识:
堆区:
1.存储的全部是对象,每个对象都包含一个与之对应的class的信息。(class的目的是得到操作指令)
2.jvm只有一个堆区(heap)被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身
栈区:
1.每个线程包含一个栈区,栈中只保存基础数据类型的对象和自定义对象的引用(不是对象),对象都存放在堆区中
2.每个栈中的数据(原始类型和对象引用)都是私有的,其他栈不能访问。
3.栈分为3个部分:基本类型变量区、执行环境上下文、操作指令区(存放操作指令)。
方法区:
1.又叫静态区,跟堆一样,被所有的线程共享。方法区包含所有的class和static变量。
2.方法区中包含的都是在整个程序中永远唯一的元素,如class,static变量。
被静态修饰的变量存在于方法区
Main主函数:
主函数:是一个特殊的函数。作为程序的入口,可以被jvm调用。
定义:
主函数的定义:
public:代表着该函数访问权限是最大的。(需要被虚拟机)
static:代表主函数随着类的加载就已经存在了。(调用对象之前就已经存在)
void:主函数没有具体的返回值。
main:不是关键字,但是是一个特殊的单词,可以被jvm识别。
(String[] args):函数的参数,参数类型是一个数组,该数组中的元素是字符串。字符串类型的数组。args是名字,写这个是因为早期写arguments,简写成args。这个是可以改的。因为大家都这样写,为了可读性,就写成args了。
主函数是固定格式的:目的是被jvm识别。Jvm只识别这个。
class MainDemo
{
        public static void main(String[] args)//args是一个数组。
        {
                System.out.println(args);//输出的是一个哈希值,显示的是args在内存中的地址值。里面每一个值都是null
System.out.println(args.length);//输出数组的长度,为0
//由此可见,main被调用的时候,jvm在调用主函数时,传入的是new String[0]
                String[] arr = {"hah","hhe","heihei","xixi","hiahia"};//给数组装数据。
                MainTest.main(arr);//执行MainTest中的main函数
        }
}
class MainTest
{
        public static void main(String[] args)//虽然有两个主函数,但是在两个类中,所以语句合法
        {
                for(int x=0; x<args.length; x++)
                        System.out.println(args[x]);//依次打印出数组中的每个数组值。
        }
}
静态使用的工具应用(ArrayTool):
每一个应用程序中都有共性的功能,可以将这些功能进行抽取,独立封装。以便复用。
虽然可以通过建立ArrayTool的对象使用这些工具方法,对数组进行操作。
发现了问题:
1,对象是用于封装数据的,可是ArrayTool对象并未封装特有数据。
2,操作数组的每一个方法都没有用到ArrayTool对象中的特有数据。
这时就考虑,让程序更严谨,是不需要对象的。
可以将ArrayTool中的方法都定义成static的。直接通过类名调用即可。
public class ArrayTool
{
        private ArrayTool(){}//私有化构造函数,不让外界使用。将方法都静态后,可以方便于使用,但是该类还是可以被其他程序建立对象的。为了更为严谨,强制让该类不能建立对象。可以通过将构造函数私有化完成。记住这个私有化类的格式,是用构造函数来做的。
        public static int getMax(int[] arr)//定义一个数组求最大值。被修饰成静态是因为没有访问到对象的特有属性。好处是随着类的加载而加载,优先于对象存在,被所有对象所共享,可以直接被类名调用。
        {
                int max = 0;
                for(int x=1; x<arr.length; x++)
                {
                        if(arr[x]>arr[max])
                                max = x;
                }
                return arr[max];
        }
        public static int getMin(int[] arr)// 每一个应用程序中都有共性的功能,可以将这些功能进行抽取,独立封装。以便复用。
        {
                int min = 0;
                for(int x=1; x<arr.length; x++)
                {
                        if(arr[x]<arr[min])
                                min = x;
                }
                return arr[min];
        }
        public static void selectSort(int[] arr)
        {
                for (int x=0; x<arr.length-1 ; x++ )
                {
                        for(int y=x+1; y<arr.length; y++)
                        {
                                if(arr[x]>arr[y])
                                {
                                        swap(arr,x,y);
                                }
                        }
                }
        }
        public static void bubbleSort(int[] arr)
        {
                for (int x=0; x<arr.length-1 ; x++ )
                {
                        for(int y=0; y<arr.length-x-1; y++)
                        {
                                if(arr[y]>arr[y+1])
                                {
                                        swap(arr,y,y+1);
                                }
                        }
                }
        }
        private  static void swap(int[] arr,int a,int b)
        {
                int temp = arr[a];
                arr[a] = arr[b];
                arr[b] = temp;
        }
        public static void printArray(int[] arr)
        {
                System.out.print("[");
                for(int x=0; x<arr.length; x++)
                {
                        if(x!=arr.length-1)
                                System.out.print(arr[x]+", ");
                        else
                                System.out.println(arr[x]+"]");
                }
        }
}
接下来,将ArrayTool.class文件发送给其他人,其他人只要将该文件设置到classpath路径下,就可以使用该工具类。直接调用,如ArrayTool.getMax(arr);可以直接调用
class ArrayToolDemo
{
        public static void main(String[] args)
        {
                int[] arr = {3,1,87,32,8};
                int max = ArrayTool.getMax(arr);
                System.out.println("max="+max);
                /*
                ArrayTool tool = new ArrayTool();
                int max = tool.getMax(arr);//上一个演示代码中构造函数ArrayTool()已经被private私有化,不能简历对象了,所以这样的语句会编译失败。只能直接使用匿名函数调用ArrayTool.getMax(arr);
                System.out.println("max="+max);
                int min = tool.getMin(arr);
                System.out.println("min="+min);
                tool.printArray(arr);
                tool.selectSort(arr);
                tool.printArray(arr);
                int[] arr1 = {};
                ArrayTool tool1 = new ArrayTool();
                */
        }
}

点评

不错不错..  发表于 2013-3-22 21:50
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马