static 静态方法 有什么优缺点 PHP中静态方法(static)与非静态方法的使用及区别

static 静态方法 有什么优缺点 PHP中静态方法(static)与非静态方法的使用及区别
  1. static方法是类中的一个成员方法,属于整个类,即使不用创建任何对象也可以直接调用!

  2. 静态方法效率上要比实例化高,静态方法的缺点是不自动进行销毁,而实例化的则可以做销毁。

  3. 静态方法和静态变量创建后始终使用同一块内存,而使用实例的方式会创建多个内存。

  4. C++中,若类的方法前加了static关键字,则该方法称为静态方法,反之为实例方法。静态方法为类所有,可以通过对象来使用,也可以通过类来使用。但一般提倡通过类名来使用,因为静态方法只要定义了类,不必建立类的实例就可使用。静态方法只能用类的静态成员。







static关键字用来修饰属性、方法,称这些属性、方法为静态属性、静态方法。

static关键字声明一个属性或方法是和类相关的,而不是和类的某个特定的实例相关,因此,这类属性或方法也称为“类属性”或“类方法”

如果访问控制权限允许,可不必创建该类对象而直接使用类名加两个冒号“::”调用。

static关键字可以用来修饰变量、方法。

不经过实例化,就可以直接访问类中static的属性和static的方法。

static 的属性和方法,只能访问static的属性和方法,不能类访问非静态的属性和方法。因为静态属性和方法被创建时,可能还没有任何这个类的实例可以被调

用。

static的属性,在内存中只有一份,为所有的实例共用。

使用self:: 关键字访问当前类的静态成员。

一个类的所有实例,共用类中的静态属性。

也就是说,在内存中即使有多个实例,静态的属性也只有一份。

下面例子中的设置了一个计数器$count属性,设置private 和 static 修饰。

这样,外界并不能直接访问$count属性。而程序运行的结果我们也看到多个实例在使用同一个静态的$count 属性。

<?php     
class user   
{     
    private static $count = 0 ; //记录所有用户的登录情况.     
    public function __construct() {     
        self::$count = self::$count + 1;     
    }     
    public function getCount() {       
        return self::$count;     
    }     
    public function __destruct() {     
        self::$count = self::$count - 1;     
    }     
}     
$user1 = new user();     
$user2 = new user();     
$user3 = new user();     
echo "now here have " . $user1->getCount() . " user";     
echo "<br />";     
unset($user3);     
echo "now here have " . $user1->getCount() . " user";     
?>    

静态属性直接调用
静态属性不需要实例化就可以直接使用,在类还没有创建时就可以直接使用。

使用的方式是: 类名::静态属性名

<?php     
class Math   
{     
    public static $pi = 3.14;     
}     
// 求一个半径3的园的面积。     
$r = 3;     
echo "半径是 $r 的面积是<br />";     
echo Math::$pi * $r * $r;     
echo "<br /><br />";     
//这里我觉得 3.14 不够精确,我把它设置的更精确。     
Math::$pi = 3.141592653589793;     
echo "半径是 $r 的面积是<br />";     
echo Math::$pi * $r * $r;      
?>    


类没有创建,静态属性就可以直接使用。那静态属性在什么时候在内存中被创建? 在PHP中没有看到相关的资料。引用Java中的概念,来解释应该也具有通用性

。静态属性和方法,在类被调用时创建。

静态方法
静态方法不需要所在类被实例化就可以直接使用。

使用的方式是类名::静态方法名

下面我们继续写这个Math类,用来进行数学计算。我们设计一个方法用来算出其中的最大值。既然是数学运算,我们也没有必要去实例化这个类,如果这个方法

可以拿过来就用就方便多了。我们这只是为了演示static方法而设计的这个类。在PHP提供了 max() 函数比较数值。

view plaincopy to clipboardprint?
<?php     
class Math   
{     
    public static function Max($num1, $num2) {     
        return $num1 > $num2 ? $num1 : $num2;     
    }          
}     
$a = 99;     
$b = 88;     
echo "显示 $a 和 $b 中的最大值是";     
echo "<br />";     
echo Math::Max($a, $b);     
echo "<br />";   
echo "<br />";   
echo "<br />";     
$a = 99;     
$b = 100;     
echo "显示 $a 和 $b 中的最大值是";     
echo "<br />";     
echo Math::Max($a,$b);     
?>   

静态方法如何调用静态方法
第一个例子,一个静态方法调用其它静态方法时,使用self:: 

<?php     
// 实现最大值比较的Math类。     
class Math   
{     
    public static function Max($num1, $num2) {     
        return $num1 > $num2 ? $num1 : $num2;     
    }     
    public static function Max3($num1, $num2, $num3) {     
        $num1 = self::Max($num1, $num2);     
        $num2 = self::Max($num2, $num3);     
        $num1 = self::Max($num1, $num2);             
        return $num1;     
    }     
}     
$a = 99;     
$b = 77;     
$c = 88;     
echo "显示 $a $b $c 中的最大值是";     
echo "<br />";     
echo Math::Max3($a, $b, $c);     
?> 



静态方法调用静态属性
使用self:: 调用本类的静态属性。

<?php     
//      
class Circle   
{     
    public static $pi = 3.14;     
    public static function circleAcreage($r) {     
        return $r * $r * self::$pi;     
    }     
}     
$r = 3;     
echo " 半径 $r 的圆的面积是 " . Circle::circleAcreage($r);     
?>    


静态方法不能调用非静态属性 。不能使用self::调用非静态属性。 

<?php     
// 这个方式是错误的     
class Circle   
{     
    public $pi = 3.14;     
    public static function circleAcreage($r) {     
        return $r * $r * self::pi;     
    }     
}     
$r = 3;     
echo " 半径 $r 的圆的面积是 " . Circle::circleAcreage($r);     
?>   


也不能使用 $this 获取非静态属性的值。

静态方法调用非静态方法
PHP5中,在静态方法中不能使用 $this 标识调用非静态方法。

<?php   
// 实现最大值比较的Math类。     
class Math   
{         
    public function Max($num1, $num2) {     
        echo "bad<br />";             
        return $num1 > $num2 ? $num1 : $num2;     
    }     
    public static function Max3($num1, $num2, $num3) {     
        $num1 = $this->Max($num1, $num2);     
        $num2 = $this->Max($num2, $num3);     
        $num1 = $this->Max($num1, $num2);             
        return $num1;     
    }     
}     
$a = 99;     
$b = 77;     
$c = 188;     
echo "显示 $a $b $c 中的最大值是";     
echo "<br />";     
echo Math::Max3($a, $b, $c);    //同样的这个会报错    
?>   

当一个类中有非静态方法被self:: 调用时,系统会自动将这个方法转换为静态方法。

<?php     
// 实现最大值比较的Math类。     
class Math   
{         
    public function Max($num1, $num2) {            
        return $num1 > $num2 ? $num1 : $num2;     
    }     
    public static function Max3($num1, $num2, $num3) {     
        $num1 = self::Max($num1, $num2);     
        $num2 = self::Max($num2, $num3);     
        $num1 = self::Max($num1, $num2);             
        return $num1;     
    }     
}     
$a = 99;     
$b = 77;     
$c = 188;     
echo "显示 $a $b $c 中的最大值是";     
echo "<br />";     
echo Math::Max3($a, $b, $c);     
?>

关于静态方法与非静态方法

09-18

各位大虾,下面这段程序是核心技术卷图形设计那一章的里的代码,我对NotHelloWorldFrame 类中构造器如此调用setTitle("NotHelloWorld")以及rnsetSize(DEFAULT_WIDTH, DEFAULT_HEIGHT)感到奇怪,根据我以前掌握的知识似乎只有静态方法才能如此调用,但是我在API里面查询到这两个方法并不是静态方法,而是实例方法,我想应该是 new JFrame().setTitle("NotHelloWorld") rnnew JFrame().setSize(DEFAULT_WIDTH, DEFAULT_HEIGHT)rnrnimport javax.swing.JFrame;rnimport javax.swing.JPanel;rnimport java.awt.Graphics;rnrnpublic class NotHelloWorldrnrn public static void main(String[] args)rn rn NotHelloWorldFrame frame = new NotHelloWorldFrame();rn frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);rn frame.setVisible(true);rn rnrnrn/**rn A frame that contains a message panelrn*/rnclass NotHelloWorldFrame extends JFramernrn public NotHelloWorldFrame()rn rn setTitle("NotHelloWorld");rn setSize(DEFAULT_WIDTH, DEFAULT_HEIGHT);rnrn // add panel to framern NotHelloWorldPanel panel = new NotHelloWorldPanel();rn add(panel);rn rnrn public static final int DEFAULT_WIDTH = 300;rn public static final int DEFAULT_HEIGHT = 200;rnrnrnclass NotHelloWorldPanel extends JPanelrnrn public void paintComponent(Graphics g)rn rn super.paintComponent(g);rn g.drawString("Not a Hello, World program", MESSAGE_X, MESSAGE_Y);rn rnrn public static final int MESSAGE_X = 75;rn public static final int MESSAGE_Y = 100;rn

多线程环境下的static静态方法和类方法

08-24

今天无意之中看到一篇讨论静态方法和类方法的帖子。于是就想着测一测,遇到了部分问题。希望有大神能给解答一下rnrn我的java代码:rnrn[color=#FF0000]包:[/color][color=#00FF00]isStatic[/color]rnMethod.javarn[code=java]rnpackage isStatic;rnrnimport java.io.FileReader;rnimport java.io.IOException;rnimport java.util.Properties;rnrnpublic class Method rn rn public static void littlelong1(String name) rn long i = 0;rn while (i ++ != 100000000l) rn //do nothing , but not sleeprn rn rnrn public void littlelong2(String name) rn long i = 0;rn while (i ++ != 100000000l) rn //do nothing , but not sleeprn rn rn /*这是为了我测试方便,可以删掉的*/rn public static int getConfig() rn Properties p = new Properties();rn try rn p.load(new FileReader("pro.properties"));rn catch (IOException e) rn e.printStackTrace();rn rn return Integer.parseInt(p.getProperty("LENGTH"));rn rnrn[/code]rnUseLittlelong.javarn[code=java]rnpackage isStatic;rnrnpublic class UseLittlelong rn rn private static int LENGTH = 20;rn private int[] avg_static = new int[LENGTH];rn private int[] avg_nonstatic = new int[LENGTH];rn //dividedrn private Method m = new Method();rn rn public static void main(String[] args) rn UseLittlelong ull = new UseLittlelong();rn ull.staticTest();rn ull.nonstaticTest();rn try rn Thread.sleep(2000);rn catch (InterruptedException e) rn e.printStackTrace();rn rn ull.avg();rn rn /*test static*/rn public void staticTest() rn int i = -1 ;rn /*原谅我选择了再循环中启动线程,只是这样显得更方便一些*/rn while (i ++ < LENGTH - 1) rn new Use(true,i).start();rn rn rn /*test non-static*/rn public void nonstaticTest() rn int i = -1 ;rn while (i ++ < LENGTH - 1) rn new Use(false,i).start();rn rn rn /*average count*/rn public void avg() rn long sta = 0,nonsta = 0;rn rn for (int i = 0 ; i < LENGTH; i++) rn sta += avg_static[i];rn nonsta += avg_nonstatic[i];rn rn rn System.out.println("static avg :" + sta / LENGTH);rn System.out.println("non-static avg :" + nonsta / LENGTH);rn rn rn /**rn * @author qyp199312rn */rn class Use extends Thread rn rn private boolean isStatic;rn private String threadName;rn private int threadId;rn rn public Use(boolean b,int threadId) rn this.isStatic = b;rn this.threadId = threadId;rn this.threadName = (b ? "[static-" : "[nonstatic-")+ threadId +"] ";rn rn rn public void run () rn if (isStatic) rn long begin = System.currentTimeMillis();rn Method.littlelong1(threadName);rn avg_static[threadId] = (int) (System.currentTimeMillis() - begin);rn// System.out.println(threadName + avg_static[threadId] );rn else rn long begin = System.currentTimeMillis();rn// new Method().littlelong2(threadName);rn m.littlelong2(threadName);rn avg_nonstatic[threadId] = (int) (System.currentTimeMillis() - begin);rn// System.out.println(threadName + avg_nonstatic[threadId] );rn rn rn rnrn[/code]rnpro.propertiesrn[code=java]rn#可在源码中置换为实际数字rnLENGTH = 20rn[/code]rn我在本机上(i5-4300U 1.9GHz 8G)跑出来的结果为:rn[code=java]rnstatic avg :602rnnon-static avg :280rn[/code]rn我在服务器上(Intel(R) Xeon(R) CPU E5520 @ 2.27GHz 48G)跑多次出来的结果为:rn[code=java]rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :438rnnon-static avg :505rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :439rnnon-static avg :480rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :486rnnon-static avg :526rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :337rnnon-static avg :448rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :460rnnon-static avg :517rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :369rnnon-static avg :466rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :402rnnon-static avg :441rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :462rnnon-static avg :498rn[tt ~]$ java isStatic/UseLittlelongrnstatic avg :372rnnon-static avg :433rn[/code]rnrn疑问:rn[color=#FF0000]1.修改线程数,得到的时间近似正比翻倍。而按理说排除掉多线程分担掉的cpu消耗,不管多少个线程执行平均时间应该是近似相等的。这是我的程序的BUG还是其他什么的原因?rn2.忽略掉本机和服务器性能差异,静态方法调用和非静态方法的调用时间占比应该差不多。而明显的,本机上静态方法调用时间占用较大,而服务器上类方法的调用时间占用较大。这又是为什么。rn服务器jdk:rnjava version "1.7.0_45"rnJava(TM) SE Runtime Environment (build 1.7.0_45-b18)rnJava HotSpot(TM) 64-Bit Server VM (build 24.45-b08, mixed mode)rn本机jdkrnjava version "1.7.0_40"rnJava(TM) SE Runtime Environment (build 1.7.0_40-b43)rnJava HotSpot(TM) 64-Bit Server VM (build 24.0-b56, mixed mode)[/color]rnrn集思广益,求大神们踊跃发言。让更多的人看见,让更多的人明白。

没有更多推荐了,返回首页

私密
私密原因:
请选择设置私密原因
  • 广告
  • 抄袭
  • 版权
  • 政治
  • 色情
  • 无意义
  • 其他
其他原因:
120
出错啦
系统繁忙,请稍后再试