• 重新绘制TabControl的Tabpage标签,添加图片及关闭按钮


    1 using System;
      2 using System.Collections.Generic;
      3 using System.ComponentModel;
      4 using System.Data;
      5 using System.Drawing;
      6 using System.Linq;
      7 using System.Text;
      8 using System.Windows.Forms;
      9 
     10 
     11 namespace TabControlTest
     12 {
     13     public partial class Form1 : Form
     14     {
     15         public Form1()
     16         {
     17             InitializeComponent();
     18         }
     19 
     20         const int CLOSE_SIZE = 15;
     21 //tabPage标签图片
     22         Bitmap image = new Bitmap("E:\1\2.jpg");
     23 //绘制“X”号即关闭按钮
     24         private void MainTabControl_DrawItem(object sender, DrawItemEventArgs e)
     25         {
     26 
     27             try
     28             {
     29                 Rectangle myTabRect = this.MainTabControl.GetTabRect(e.Index);
     30 
     31                 //先添加TabPage属性   
     32                 e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text
     33                 , this.Font, SystemBrushes.ControlText, myTabRect.X + 2, myTabRect.Y + 2);
     34 
     35                 //再画一个矩形框
     36                 using (Pen p = new Pen(Color.White))
     37                 {
     38                     myTabRect.Offset(myTabRect.Width - (CLOSE_SIZE + 3), 2);
     39                     myTabRect.Width = CLOSE_SIZE;
     40                     myTabRect.Height = CLOSE_SIZE;
     41                     e.Graphics.DrawRectangle(p, myTabRect);
     42                  
     43                 }
     44 
     45                 //填充矩形框
     46                 Color recColor = e.State == DrawItemState.Selected ? Color.White : Color.White;
     47                 using (Brush b = new SolidBrush(recColor))
     48                 {
     49                     e.Graphics.FillRectangle(b, myTabRect);
     50                 }
     51 
     52                 //画关闭符号
     53                 using (Pen objpen = new Pen(Color.Black))
     54                 {
     55                     //""线
     56                     Point p1 = new Point(myTabRect.X + 3, myTabRect.Y + 3);
     57                     Point p2 = new Point(myTabRect.X + myTabRect.Width - 3, myTabRect.Y + myTabRect.Height - 3);
     58                     e.Graphics.DrawLine(objpen, p1, p2);
     59 
     60                     //"/"线
     61                     Point p3 = new Point(myTabRect.X + 3, myTabRect.Y + myTabRect.Height - 3);
     62                     Point p4 = new Point(myTabRect.X + myTabRect.Width - 3, myTabRect.Y + 3);
     63                     e.Graphics.DrawLine(objpen, p3, p4);
     64                       ////=============================================
     65                     Bitmap bt = new Bitmap(image);
     66                     Point p5 = new Point(myTabRect.X-50, 4);
     67                     e.Graphics.DrawImage(bt, p5);
     68                     //e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text, this.Font, objpen.Brush, p5);
     69                 }
     70 
     71 
     72                 //绘制小图标              
     73                 //==============================================================================
     74                 //Bitmap bt = new Bitmap("E:\1\2.jpg");
     75                 //Point p5 = new Point(4, 4);
     76                 ////e.Graphics.DrawImage(bt, e.Bounds);
     77                 //e.Graphics.DrawImage(bt, p5);
     78                 //Pen pt = new Pen(Color.Red);
     79                 ////e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text, this.Font, pt.Brush, e.Bounds);
     80                 //e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text, this.Font, pt.Brush, p5);
     81 
     82                 e.Graphics.Dispose();
     83             }
     84             catch (Exception)
     85             {
     86 
     87             }
     88 
     89 
     90         }
     91         //=======================================================================
     92      
     93 //关闭按钮功能
     94         private void MainTabControl_MouseDown(object sender, MouseEventArgs e)
     95         {
     96             if (e.Button == MouseButtons.Left)
     97             {
     98                 int x = e.X, y = e.Y;
     99 
    100                 //计算关闭区域   
    101                 Rectangle myTabRect = this.MainTabControl.GetTabRect(this.MainTabControl.SelectedIndex);
    102 
    103                 myTabRect.Offset(myTabRect.Width - (CLOSE_SIZE + 3), 2);
    104                 myTabRect.Width = CLOSE_SIZE;
    105                 myTabRect.Height = CLOSE_SIZE;
    106 
    107                 //如果鼠标在区域内就关闭选项卡   
    108                 bool isClose = x > myTabRect.X && x < myTabRect.Right
    109                  && y > myTabRect.Y && y < myTabRect.Bottom;
    110 
    111                 if (isClose == true)
    112                 {
    113                     this.MainTabControl.TabPages.Remove(this.MainTabControl.SelectedTab);
    114                 }
    115             }
    116 
    117 
    118         }
    119 //初始化页面
    120         private void Form1_Load(object sender, EventArgs e)
    121         {
    122             //清空控件
    123             //this.MainTabControl.TabPages.Clear();
    124             //绘制的方式OwnerDrawFixed表示由窗体绘制大小也一样
    125             this.MainTabControl.DrawMode = TabDrawMode.OwnerDrawFixed;
    126             this.MainTabControl.Padding = new System.Drawing.Point(CLOSE_SIZE, CLOSE_SIZE);
    127             this.MainTabControl.DrawItem += new DrawItemEventHandler(this.MainTabControl_DrawItem);
    128             this.MainTabControl.MouseDown += new System.Windows.Forms.MouseEventHandler(this.MainTabControl_MouseDown);
    129         }
    130 
    131  //添加新的tabPage并修改所有tabPage标签上面的图片
    132         private void button1_Click(object sender, EventArgs e)
    133         {
    134             TabPage tabtage=new TabPage ();
    135             MainTabControl.TabPages.Add(tabtage);
    136             MainTabControl.SelectedTab = tabtage;
    137             image = new Bitmap("E:\1\3.jpg");
    138 
    139 
    140         }
    141 //关闭选中的tabPage
    142         private void button2_Click(object sender, EventArgs e)
    143         {
    144             MainTabControl.TabPages.Remove(MainTabControl.SelectedTab);
    145         }
    146 
    147      
    148 
    149     }
    150 }
    View Code

    制作个性Tabcontrol控件(收藏~~)

    有的人认为,Blend主要是用来修改一下颜色,调整一下布局之类的,大部分的SL开发还是用VS搞定;这样的结果我们可以预测,他的UI一定是很普通的,也许布局颜色搭配之类算不上难看,符合一般审美,但是绝不是高水平的UI,也不会是销售人员满意的设计;
     有的人认为,Blend太难学了,总是搞不懂应该怎么去操作,感觉跟其他设计工具都有点不一样,所以,很多人对它很感兴趣,希望用它来实现自己心中如诗如画的构思;于是站在门口,不断的徘徊,却始终不得其门;
     我其实不是UI设计人员,我连Photoshop都不会,也从没有设计过像样的东西;并且对于Silverlight,我也不是很有经验;我想告诉你的是,它没有那么复杂,因为像我这样的人都能捣弄一下,除非你在后面留言说:“楼主,我比你想象的要笨! ” 
    示例说明:
     作为Blend操作的入门篇,我对Blend操作几个阶段的的定义是:
     1. 入门阶段-能得心应手的修改任何控件的样式;
     2. 中级阶段-能得心应手的设计动画,设计自定义控件的模板,状态,状态过渡,各种Behavior的运用和开发
     3. 高级阶段-只要你想得到的场景,你都知道怎么去设计
     这篇是针对第一阶段的朋友写的,因为我也只是刚过了第一阶段,其实开始不知道要做什么样的示例比较合适。昨天群里有一MM问怎么做一个好看一点的Tabcontrol控件,于是就想以修改Tabcontrol模板为示例,顺便也可以给她的帮助。以下是示例截图,先看一下效果,我不是设计人员,不要对我要求太高,我就做这个简单的修改,你会比我设计的好看得多:
     
    小话设计模式:
     有一个坏毛病,总是喜欢在文章中插一些不相关的东西,这里也来一个小话设计模式。
     记得,以前写Behavior的时候也是提到了设计模式,那么这里又提到设计模式。有的人说,既然对设计模式这么感兴趣,这个人为什么没有写过一篇设计模式的文章呢,也来讲一下比如单例模式的UML图是什么样子,实现代码是什么样子。
     等等,这时候你会说,那些设计模式书上不是已经这样写了么!
     是的,个人觉得,对于那些经典设计模式的描述,已经有很多书了,而且我认为没有必要再出那些设计模式的书了,除非你自己总结出新的设计模式实践,或者真正实用的应用。大部分的关于设计模式的书,都是去构造一个不同的故事,不同的场景,最后写出相同的标准的类,接口的定义,一个简单的差不多都相同代码的示例。
     我建议再也不要写这样的书了,很多人看了很多本这样的书,还是不会应用(我就是其中一个)。
     我建议的下一本关于设计模式的书是,用真实的经典的开发者比较熟悉的案例来分析设计模式。那么我们来看一下模板方法。
     不知道你现在对模板方法有一个什么样的概念,如果你现在还不是很清楚,那么我相信这一次你应该可以清楚了,不清楚的请在后面留言“我很笨!”
     可能在你的开发中很少用到模板方法,但是我告诉你你天天在运用模板方法,你相不相信呢?
     模板方法的概念是,定义一个框架,但是将具体的算法延迟到子类中,使得子类可以定义自己的算法实现。这样的定义像很多设计模式书描述的一样,你会很迷糊,下面举个例子:
     ASP.NET的核心机制就是模板方法,每一个Page是一个实现IHttpHandler的实现类,Page定义了一些框架,比如page_Load这些方法,这样当ASP.NET在执行页面请求的时候,首先找到对应的Page(也就是子类),然后依次调用框架中定义的方法,比如Init,Pre_Render等等。ASP.NET框架本身来控制对子类方法的调用,通过这样的方式来处理不同的请求,你可以定义不同的Page做不同的事情,用不同的算法,后台逻辑。
     所以,模板方法是非常重要的设计模式,它通常用于框架设计,你定义一个框架,不同的子类遵循这个框架实现规定的方法,然后框架主体来调用子类的方法,到达可扩展性,因为子类只要遵循这个框架定义,它可以有不同的实现。
     所以,模板方法不仅仅只是一种设计模式,它基本上可以算作一个架构风格了,你可以站在架构师的层面,去运用模板方法。
     那我们这里为什么讲模板方法呢?因为Silverlight实现控件模板自定义的样式的机制也是模板方法:控件定义一个ControlTemplate属性,这个对象你可以自己定义,如果你不定义,Silverlight去取默认的Xaml样式,实例化然后赋值给控件,如果你定义了自己的ControlTemplate对象,Silverlight就用你自定义的对象来初始化控件,这样控件就能使用你的样式。
     Silverlight中的ControlTemplate就相当于Page,其本身相当于定义了一个框架,具体的样式定义在ControlTemplate中。控件本身不知道样式,就像ASP.NET不知道怎么去处理每个页面。
     这,就是模板方法,有了这个提醒之后,相信你会发现更多模板方法的样子,相信现在你应该在对模板方法的概念感到模糊了。
     我相信,用我们熟悉的东西来讲设计模式,这样效果好得多,这样的例子很多很多,因为经典设计模式其实也应用在很多经典的软件设计中,而不仅仅只是一个概念。
     很希望,下一本设计模式的书是这样的。
    Blend入门很简单:
     除了修改颜色等基本属性,下面列举几点必须掌握的东西:
     1. 示例数据
     示例数据的管理位于Data面板,如下图:
     
     为什么它是很重要的,设计Silverlight UI很痛苦的是,数据都是从Service上获取的,这样在设计的时候一般看不到控件的外观。那么这个时候我们就要采用本地示例数据,它能够帮助我们在设计时很完整的看到运行时的效果,这样才能更好的设计。它的原理是,Blend或者VS的设计器在呈现Xaml的时候,其实和真实的运行时差不多,在运行时,Silverlight插件初始化SL控件,获取每个元素的布局信息,然后将其绘制到屏幕上正确的位置。那设计器想要呈现设计时的外观,也是要家里这样的过程,想要初始化控件树,绘制到屏幕,不同的是设计时只会初始化Xaml中定义的元素。如果这个时候示例数据是本地资源,并且示例数据的集合对象是放在Xaml中初始化的,那么设计器就能找到对应数据,将它作为控件的Context,而到该控件实例化的时候,它发现已经有了数据,自然不会放过,会将其绘制出来,所以我们在设计时能看到效果。尤其是某些复杂的集合控件,这非常重要。Blend中示例数据实际上是一个Xml文件,可以定义不同类型的数据,比如Image,String,你告诉它想要多少条数据,它就会生成随机的示例数据,达到以假乱真:
     示例数据的使用非常简单,拖动collection那一行,至一个控件比如ListBox即可:
     然后就会生成预览图:
     这个时候再来修改模板是不是方便多了,一般建议是建立和真实的数据结构一样的属性,在真实运行的时候,将数据源改为真实的数据源,这样绑定属性不用改。这是Blend入门最基本的东西。
     
     2. 怎样调出控件模板ControlTemplate
     在Silverlight中修改模板得首先把原来的模板样式取出来,然后在其基础之上去修改,Blend为我们提供了一种很方便的方式把这段模板取出来,如我们想调出ListBox的ControlTemplate,如下:
     
     右键选择Edit Template->Edit a Copy,Blend就会把控件的默认模板xaml定义片段复制到Xaml中,你可以在此基础之上修改,你也可以选中ListBox,在视图面板中进行该操作,效果是一样的,建议采用后一种方式,后面会看到为什么:
     这个时候你就可以看到该模板的定义,状态,可以进行修改。这里有一个细节就是,怎样退出控件模板,即怎样在页面也控件模板之间切换,如下我们看到的是ListBox的ControlTemplate视图,我们可以点击视图面板上端的控件名称切换为页面视图。Blend所有的模板之间的切换都是在这里,如果想进入模板视图,可以再选择:ListBox->Edit Template->Edit Current进入:
     
     3. 怎样调出集合控件中子元素的ControlTemplate,即集合控件的ItemTemplate
     按如上的操作就可以进入ItemTemplate视图,这个时候就可以编辑每个子项的模板。不过这里我们将看到,这里的只包含每个子项的组成元素,每个子项包含哪些元素。但是不能编辑不同的状态,比如ListBox在MouseOver的时候是什么样式,在Selected的时候是另外一种样式,这个时候需要选择下面的Edit Generated Item Container(ItemContainerStyle),就可以编辑控件的状态。
     
     4. 怎样编辑控件的不同状态下的外观
     我们看到,经典的Button控件,它不是只有一个样子的。它有很多状态,比如鼠标经过是一种样子,鼠标离开,点击都会呈现不同的样子,这都是通过定义不同的状态来实现的。
     一个控件可以定义多个状态,在每个状态下可以设定不同的外观,比如ComboBox控件,在点击下拉框的时候就显示下面的选项,鼠标离开就隐藏选项。当进入不同的状态,状态管理器就将改变控件外观至该状态定义的样子。关于状态可以参考我之前的文章。我们看下ListBoxItem的状态:
     状态位于States面板,我们看到ListBoxItem模板定义了很多状态,状态可以分成不同的状态组,每个状态组之间的状态是互斥的,选择其中一个状态,就可以设置该状态下控件的外观,比如如图我点击最后一个Selected状态,这个时候视图面板就会多出一个红色的框,表示开始录制动画,在运行时的如果切换至该状态,状态管理器就会触发动画使外观呈现定义的样子。比如我们将每个项的宽度拉长,这样在运行时选择的时候,被选中的项就会拉长。
     
    示例操作:
     上面介绍的几点是我觉得Blend入门应该知道的最基本的东西,当然不能写的太详细,需要你去操作体会实践,我也不可能写出所有细节,每一个操作都还有一些更细的东西和功能,可以自己去摸索,我们下面就来从头实现本文提到的示例:
     1. 打开Blend 4,新建MyTabcontrol项目,项目类型Silverlight Application + WebSite
     2. 添加一个TabControl到MainPage控件中,并设置MainPage背景为黑色
     3. 选中TabControl控件,在Properties面板设置如下,并这个时候TabControl呈现如图所示外观:
     4. 右键选择TabControl—》Edit Template—》Edit a Copy,这个时候会看到如下图,删掉其中的后面三项,只留下TemplateTop(后面三项是说Tab分别在下面,左边和右边的情形,我们这里Tabs在上边,其他三个用不着,你也可以选择其他的布局):
     
     5. 选择TemplateTop,可以看到其是一个Grid控件,共有2行,第一行是Tabs,第二行就是下面的现实面板。修改Properties面板中RowDefinitions ,将Grid的第一行高度为50px;
     6. 展开TemplateTop,选中TabPanelTop,这就是定义TabControl控件上面Tabs部分的背景的。在Properties面板中设置Background为:#FFBBD7FA 
     7. 选中下面的Border,将Background Reset: #FFB1CBEB;BorderBrush:#FFE4E4E4;BorderThickness:0,0,1,0;并加上圆角CornerRadius:0,0,10,108,. 选中ContentTop,加上Margin:10。这个时候看起来是这个样子:
     9. 下面我们来编辑TabItem的模板。
     回到MainPage的视图,展开TabControl,选择第一个TabItem,调出其面板(右键选择TabItem—>Edit Template—>Edit a Copy)。进入TabItem面板之后看到有许多部件,删掉其他的只留下三个:TemplateTopSelected,TemplTopUnSelected,FocusVisualElement,原因和前面的一样,我们的Tab只会在顶端,你可以自己去修改其他样式。
     接下来的,TemplateTopSelected就是定义选择状态的外观,TemplateTopUnSelected就是定义没有选择的外观。
     10. 在States面板,选中base,在Objects and TimeLine面板选中Root元素,在属性面板修改高度为50px;
     11. 选中Selected状态,开始录制
     12. 展开TemplateTopSelected,将b元素order的Margin,BorderThickness,CornerRadius都设置为0,Background设置为:#FFBBD7FA
     13. 选择border1元素,设置参数,BorderThickness:1,1,1,0;CornerRadius:10,10,10,0;Margin:2,2,5,0;Background和BorderBrush都设置为纯白色。
     14. 选择HeaderTopSelected元素:修改字体大小样式至以下的样子:这个时候看起来是这样:
     15. 接下来我们修改TemplateTopUnSelected的样子,它是处于UnSelected的状态,所以我们在States面板选中UnSelected状态,开始录制。注意这个时候找到TabControl控件中的两个TabItem控件,将第一个TabItem的Style="{StaticResource TabItemStyle1}"样式复制给第二个。
     16. 对着前面的步骤,修改TemplateTopUnSelected下面的各个元素,使之呈现如下样式:
     17. 大功告成,就是还有一点小小的瑕疵,就是在选中的时候有个讨厌的边框,这个时候将FocusVisualTop的BorderBrush设置为无色就可以了,看一下我们的成果:
     
    总结:
     本篇先总结了一点Blend入门的注意事项和要领,接着展示了一个示例。
     开始使用Blend,绝大部分是修改控件模板,本文抛砖引玉,演示了一般的修改控件模板的方法,包括ControlTemplate以及集合控件的ItemTemplate。掌握了这些方法,修改Silverlight控件外观就不难了,相信,杀掉Silverlight默认的丑陋的外观,你自己设计的外观一定如诗如画。你们公司的销售也该请你吃饭了,哈哈。
     当然,Blend还有很多东西,本文讲得很简单,还希望大家多多实践,多看看老外的文章,示例。
     后期有时间在写一篇高级一点的,包括动画和自定义状态,以及状态之间的转换动画等等。
     总之,Blend还是比较有趣的,只要你想得出来的,它基本上都可以做出来,希望大家多多实践。
     如有兴趣,可以加入QQ群讨论,本群热烈欢迎有志于高水平UI设计的朋友,当然我们也讨论Silverlight,Windows Azure,Ajax,WCF等等微软技术,欢迎你。
     QQ群:6183299
    View Code

    C# TabContorl选项卡

    C# TabContorl选项卡
    if(textBox1.Text=="")
    {
    tabControl1.SelectedIndex=0;//如果想打开的是第二个,则为1
    }
    这样就可以打开你想打开的任意一个对话框
    想随着选择combobox地具体内容地不同,TabControl顶出现地标签也不同,标签已经创建好地,如1共就为三个标签页,需要么就为三个全显示,需要么就为显示其中地两个,应该如何?
    我还没有找到好的办法,我想只想将其中不用的移除吧。
    tabControl1.TabPages.Remove(tbOneCinema);其中tbOneCinema为当中一个选项卡的(Name)
    添加新的项卡时,为
    TabPage MyTabPage=new TabPages(Title);
    tabControl1.TabPages.Add(MyTabPage);
    c#中字符串截取使用的方法
    String.Substring 方法
    名称 说明 
    String.Substring (Int32)         从此实例检索子字符串。子字符串从指定的字符位置开始。 
    String.Substring (Int32, Int32) 从此实例检索子字符串。子字符串从指定的字符位置开始且具有指定的长度。
    举例如下:
    using System;
    using System.Collections.Generic;
    using System.Text;
    namespace ConsoleApplication1
    {
         class Program
         {
             static void Main(string[] args)
             {
                 string s = "Hello C# World!";
                 //s1为从s中截取的位置为3的字符以后的字符子串,3表示子字符串的起始字符位置
                string s1=s.Substring(3);
                 Console.WriteLine(s1);
                 //s2为从s中截取的位置为6的字符开始长度为2的字符串,6表示子字符的起始字符位置,2表示子字符长度
                string s2 = s.Substring(6, 2);
                 Console.WriteLine(s2);
            }
         }
    }
    结果如下:
    lo C# World!
    C#
    随即在附上一个C#截取字符串函数
    public string getString(string RawString, Int32 Length)
         {
             if (RawString.Length <= Length)
             {
                 return RawString;
             }
             else
             {
                 for (Int32 i = RawString.Length - 1; i >= 0; i--)
                 {
                     if (System.Text.Encoding.GetEncoding("GB2312").GetByteCount(RawString.Substring(0, i)) < Length)
                     {
                         return RawString.Substring(0, i) + "...";
                     }
                 }
                 return "...";
             }
         }
    
    1/**//// <summary> 
    2         /// 截取字符串,不限制字符串长度 
    3         /// </summary> 
    4         /// <param name="str">待截取的字符串</param> 
    5         /// <param name="len">每行的长度,多于这个长度自动换行</param> 
    6         /// <returns></returns> 
    7         public string CutStr(string str,int len) 
    8         {     string s=""; 
    9             
    10             for(int i=0;i<str.Length ;i++) 
    11             { 
    12                 int r= i% len; 
    13                 int last =(str.Length/len)*len; 
    14                 if (i!=0 && i<=last) 
    15                 { 
    16                     
    17                     if( r==0) 
    18                     { 
    19                         s+=str.Substring(i-len,len)+"<br>"; 
    20                     } 
    21                         
    22                 } 
    23                 else if (i>last) 
    24                 { 
    25                     s+=str.Substring(i-1) ; 
    26                     break; 
    27                 } 
    28                 
    29             } 
    30             
    31             return s; 
    32             
    33         } 
    34 
    35 
    36         /**//// <summary> 
    37         /// 截取字符串并限制字符串长度,多于给定的长度+。。。 
    38         /// </summary> 
    39         /// <param name="str">待截取的字符串</param> 
    40         /// <param name="len">每行的长度,多于这个长度自动换行</param> 
    41         /// <param name="max">输出字符串最大的长度</param> 
    42         /// <returns></returns> 
    43         public string CutStr(string str,int len,int max) 
    44         { 
    45             string s=""; 
    46             string sheng=""; 
    47             if (str.Length >max) 
    48             { 
    49                 str=str.Substring(0,max) ; 
    50                 sheng=""; 
    51             } 
    52             for(int i=0;i<str.Length ;i++) 
    53             { 
    54                 int r= i% len; 
    55                 int last =(str.Length/len)*len; 
    56                 if (i!=0 && i<=last) 
    57                 { 
    58                     
    59                     if( r==0) 
    60                     { 
    61                         s+=str.Substring(i-len,len)+"<br>"; 
    62                     } 
    63                         
    64                 } 
    65                 else if (i>last) 
    66                 { 
    67                     s+=str.Substring(i-1) ; 
    68                     break; 
    69                 } 
    70                 
    71             } 
    72             
    73             return s+sheng; 
    74             
    75         } 
     
    View Code
  • 相关阅读:
    Shiro 登录、退出、校验是否登录涉及到的Session和Cookie
    Apache Tomcat 8.0 官方文档
    FastDFS分布式文件系统(主备Tracker、主备Storage)
    PHP文件系统
    PHP 文件包含
    PHP函数
    PHP 全局变量
    PHP7新增知识点
    PHP数据
    PHP常量
  • 原文地址:https://www.cnblogs.com/blogpro/p/11458485.html
Copyright © 2020-2023  润新知