C#面试题(转载) SQL Server 数据库基础笔记分享(下) SQL Server 数据库基础笔记分享(上) Asp.Net MVC4中的全局过滤器 C#语法——泛型的多种应用...

C#面试题(转载)

原文地址:100道C#面试题(.net开发人员必备)  https://blog.csdn.net/u013519551/article/details/51220841

 

1. .NET和C#有什么区别

答:.NET一般指 .NET FrameWork框架,它是一种平台,一种技术。

C#是一种编程语言,可以基于.NET平台的应用。

2.一列数的规则如下: 1、1、2、3、5、8、13、21、34...... 求第30位数是多少,用递归算法实现。
答:public class MainClass 
                 { 
                 public static void Main() 
                 { 
                 Console.WriteLine(Foo(30)); 
                 } 
                 public static int Foo(int i) 
                 { 
                 if (i <= 0) 
                 return 0; 
                 else if(i > 0 && i <= 2) 
                 return 1; 
                 else return Foo(i -1) + Foo(i - 2); 
                 } 
                 }

3. C#中的委托是什么?事件是不是一种委托?
答 : 委托可以把一个方法作为参数代入另一个方法。
                 委托可以理解为指向一个函数的引用。
                 是,是一种特殊的委托

4. 简述 private、 protected、 public、internal 修饰符的访问权限。

答 . private : 私有成员, 在类的内部才可以访问。

protected: 保护成员,该类内部和继承类中可以访问。

public: 公共成员,完全公开,没有访问限制。

internal:在同一命名空间内可以访问。

5. override与重载的区别

答 :override 与重载的区别。重载是方法的名称相同。参数或参数类型不同,进行多

次重载以适应不同的需要

Override是进行基类中函数的重写。为了适应需要。

6.如果在一个B/S结构的系统中需要传递变量值,但是又不能使用Session、Cookie、Application,您有几种方法进行处理?
答 :   this.Server.Transfer

7. 请编程遍历页面上所有TextBox控件并给它赋值为string.Empty?
答:
foreach (System.Windows.Forms.Control control in this.Controls)
{
if (control is System.Windows.Forms.TextBox)
{
System.Windows.Forms.TextBox tb = (System.Windows.Forms.TextBox)control ; 
tb.Text = String.Empty ;
}
}

8. 请编程实现一个冒泡排序算法?
答:
int [] array = new int [*] ;
int temp = 0 ;
for (int i = 0 ; i < array.Length - 1 ; i++)
{
for (int j = i + 1 ; j < array.Length ; j++)
{
if (array[j] < array[i])
{
temp = array[i] ;
array[i] = array[j] ;
array[j] = temp ;
}
}
}

9. 描述一下C#中索引器的实现过程,是否只能根据数字进行索引?
答:不是。可以用任意类型。

10. 求以下表达式的值,写出您想到的一种或几种实现方法:1-2+3-4+……+m
答:
int Num = this.TextBox1.Text.ToString() ;
int Sum = 0 ;
for (int i = 0 ; i < Num + 1 ; i++)
{
if((i%2) == 1)
{
Sum += i ;
}
else
{
Sum = Sum - I ;
}
}
System.Console.WriteLine(Sum.ToString());
System.Console.ReadLine() ;

11. 在下面的例子里
using System;
class A
{
public A()
{
PrintFields();
}
public virtual void PrintFields(){}
}
class B:A
{
int x=1;
int y;
public B()
{
y=-1;
}
public override void PrintFields()
{
Console.WriteLine("x={0},y={1}",x,y);
}
当使用new B()创建B的实例时,产生什么输出?
答:X=1,Y=0;x= 1 y = -1

12. CTS、CLS、CLR分别作何解释?
答:CTS:通用语言系统。CLS:通用语言规范。CLR:公共语言运行库。

13. 什么是装箱和拆箱?
答:从值类型接口转换到引用类型装箱。从引用类型转换到值类型拆箱。

14. 什么是受管制的代码?
答:unsafe:非托管代码。不经过CLR运行。

15.什么是强类型系统?

答:RTTI:类型识别系统。

16.net中读写数据库需要用到那些类?他们的作用?

答:DataSet:数据存储器。

DataCommand:执行语句命令。

DataAdapter:数据的集合,用语填充。

17.列举ASP.NET页面之间传递值的几种方式。 
答. 1).使用QueryString, 如....?id=1; response. Redirect().... 
                 2).使用Session变量 
                 3).使用Server.Transfer

18.什么是Code-Behind技术?

答:代码后植。

19.在.net中,配件的意思是?

答:程序集。(中间语言,源数据,资源,装配清单)

20.常用的调用WebService的方法有哪些?

答:1.使用WSDL.exe命令行工具。

2.使用VS.NET中的Add Web Reference菜单选项

21..netRemoting 的工作原理是什么?

答:服务器端向客户端发送一个进程编号,一个程序域编号,以确定对象的位置

22.在C#中,string str = null 与 string str = “” 请尽量使用文字或图

象说明其中的区别。

答:string str = null 是不给他分配内存空间,而string str = "" 给它分配

长度为空字符串的内存空间。

23.请详述在dotnet中类(class)与结构(struct)的异同?

答:Class可以被实例化,属于引用类型,是分配在内存的堆上的,Struct属于值类

型,是分配在内存的栈上的.

24.分析以下代码,完成填空

stringstrTmp = "abcdefg某某某";

inti= System.Text.Encoding.Default.GetBytes(strTmp).Length;

intj= strTmp.Length;

以上代码执行完后,i= j=

答:i=13,j=10

25.SQLSERVER服务器中,给定表table1 中有两个字段 ID、LastUpdateDate,

ID表示更新的事务号,LastUpdateDate表示更新时的服务器时间,请使用一句

SQL语句获得最后更新的事务号

答:Select ID FROM table1 Where LastUpdateDate = (Select MAX

(LastUpdateDate)FROM table1)

26.简要谈一下您对微软.NET构架下remoting和webservice两项技术的理解以及

实际中的应用。

答:WS主要是可利用HTTP,穿透防火墙。而Remoting可以利用TCP/IP,二进制传

送提高效率。

27.公司要求开发一个继承System.Windows.Forms.ListView类的组件,要求达到

以下的特殊功能:点击ListView各列列头时,能按照点击列的每行值进行重排视

图中的所有行 (排序的方式如DataGrid相似)。根据您的知识,请简要谈一下您的

思路

答:根据点击的列头,包该列的ID取出,按照该ID排序后,在给绑定到ListView中。

 

28.写出一条Sql语句:取出表A中第31到第40记录(SQLServer,以自动增长的ID

作为主键,注意:ID可能不是连续的。

答:解1: select top 10 * from A where id not in (select top 30 id

fromA)

解2: select top 10 * from A where id >(select max(id) from (select

top30 id from A )as A)

29.面向对象的语言具有________性、_________性、________性

答:封装、继承、多态。

30.能用foreach遍历访问的对象需要实现 ________________接口或声明

________________方法的类型。

答:IEnumerable 、 GetEnumerator。

31.GC是什么? 为什么要有GC?

答:GC是垃圾收集器。程序员不用担心内存管理,因为垃圾收集器会自动进行管

理。要请求垃圾收集,可以调用下面的方法之一:

System.gc()

Runtime.getRuntime().gc()

32.Strings = new String("xyz");创建了几个String Object?

答:两个对象,一个是“xyx”,一个是指向“xyx”的引用对象s。

 

33.启动一个线程是用run()还是start()?

答:启动一个线程是调用start()方法,使线程所代表的虚拟处理机处于可运行状

态,这意味着它可以由JVM调度并执行。这并不意味着线程就会立即运行。run()

方法可以产生必须退出的标志来停止一个线程。

34.接口是否可继承接口? 抽象类是否可实现(implements)接口? 抽象类是否可

继承实体类(concrete class)?

答:接口可以继承接口。抽象类可以实现(implements)接口,抽象类是否可继承

实体类,但前提是实体类必须有明确的构造函数。

35.构造器Constructor是否可被override?

答:构造器Constructor不能被继承,因此不能重写Overriding,但可以被重载

Overloading。

36.是否可以继承String类?

答:String类是final类故不可以继承。

37.try{}里有一个return语句,那么紧跟在这个try后的finally {}里的code会

不会被执行,什么时候被执行,在return前还是后?

答:会执行,在return前执行。

38.两个对象值相同(x.equals(y)== true),但却可有不同的hash code,这句

话对不对?

答:不对,有相同的hash code。

39.swtich是否能作用在byte上,是否能作用在long上,是否能作用在String上?

答:switch(expr1)中,expr1是一个整数表达式。因此传递给 switch 和

case语句的参数应该是 int、 short、 char 或者 byte。long,string 都不

能作用于swtich。

40.当一个线程进入一个对象的一个synchronized方法后,其它线程是否可进入此

对象的其它方法?

不能,一个对象的一个synchronized方法只能由一个线程访问。

41.abstract的method是否可同时是static,是否可同时是native,是否可同时是

synchronized?

答:都不能。

42.List,Set, Map是否继承自Collection接口?

答:List,Set是Map不是

43.Set里的元素是不能重复的,那么用什么方法来区分重复与否呢? 是用==还是

equals()?它们有何区别?

答:Set里的元素是不能重复的,那么用iterator()方法来区分重复与否。

equals()是判读两个Set是否相等。

equals()和==方法决定引用值是否指向同一对象equals()在类中被覆盖,为的是

当两个分离的对象的内容和类型相配的话,返回真值。

44.数组有没有length()这个方法? String有没有length()这个方法?

答:数组没有length()这个方法,有length的属性。String有有length()这个方

法。

45.sleep()和 wait() 有什么区别?

答:sleep()方法是使线程停止一段时间的方法。在sleep 时间间隔期满后,线程

不一定立即恢复执行。这是因为在那个时刻,其它线程可能正在运行而且没有被

调度为放弃执行,除非(a)“醒来”的线程具有更高的优先级

(b)正在运行的线程因为其它原因而阻塞。

wait()是线程交互时,如果线程对一个同步对象x 发出一个wait()调用,该线程

会暂停执行,被调对象进入等待状态,直到被唤醒或等待时间到。

46.shorts1 = 1; s1 = s1 + 1;有什么错? short s1 = 1; s1 += 1;有什么错

?

答:short s1 = 1; s1 = s1 + 1;有错,s1是short型,s1+1是int型,不能显式

转化为short型。可修改为s1 =(short)(s1 + 1) 。short s1 = 1; s1 += 1正

确。

47.谈谈final,finally, finalize的区别。

答:

final—修饰符(关键字)如果一个类被声明为final,意味着它不能再派生出新

的子类,不能作为父类被继承。因此一个类不能既被声明为 abstract的,又被

声明为final的。将变量或方法声明为final,可以保证它们在使用中 不被改变。

被声明为final的变量必须在声明时给定初值,而在以后的引用中只能读取,不可

修改。被声明为 final的方法也同样只能使用,不能重载

finally—再异常处理时提供finally 块来执行任何清除操作。如果抛出一个异

常,那么相匹配的 catch 子句就会 执行,然后控制就会进入 finally 块(如果

有的话)。

finalize—方法名。Java 技术允许使用 finalize() 方法在垃圾收集器将对象

从内存中清除出去之前做必要的清理工作。这个方法是由垃圾收集器在确定这个

对象没有被引用时对这个对象调用的。它是在 Object 类中定义的 ,因此所有的

类都继承了它。子类覆盖 finalize() 方法以整理系统资源或者执行其他清理工

作。finalize() 方法是在垃圾收集器删除对象之前对这个对象调用的。

48.如何处理几十万条并发数据?

答:用存储过程或事务。取得最大标识的时候同时更新..注意主键不是自增量方

式这种方法并发的时候是不会有重复主键的..取得最大标识要有一个存储过程来

获取.

49.Session有什么重大BUG,微软提出了什么方法加以解决?

答:是iis中由于有进程回收机制,系统繁忙的话Session会丢失,可以用Sate

server或SQL Server数据库的方式存储Session不过这种方式比较慢,而且无法

捕获Session的END事件。

50.进程和线程的区别?

答:进程是系统进行资源分配和调度的单位;线程是CPU调度和分派的单位,一个

进程可以有多个线程,这些线程共享这个进程的资源。

51.堆和栈的区别?

答: 栈:由编译器自动分配、释放。在函数体中定义的变量通常在栈上。

堆:一般由程序员分配释放。用new、malloc等分配内存函数分配得到的就是在堆

上。

52.成员变量和成员函数前加static的作用?

答:它们被称为常成员变量和常成员函数,又称为类成员变量和类成员函数。分

别用来反映类的状态。比如类成员变量可以用来统计类实例的数量,类成员函数

负责这种统计的动作。

53.ASP。NET与ASP相比,主要有哪些进步?

答:asp解释形,aspx编译型,性能提高,有利于保护源码。

54.请说明在.net中常用的几种页面间传递参数的方法,并说出他们的优缺点。

答:session(viewstate) 简单,但易丢失

application全局

cookie简单,但可能不支持,可能被伪造

inputttype="hidden" 简单,可能被伪造

url参数 简单,显示于地址栏,长度有限

数据库 稳定,安全,但性能相对弱

55.请指出GAC的含义?

答:全局程序集缓存。

56.向服务器发送请求有几种方式?

答:get,post。get一般为链接方式,post一般为按钮方式。

57.DataReader与Dataset有什么区别?

答:一个是只能向前的只读游标,一个是内存中虚拟的数据库。

58.软件开发过程一般有几个阶段?每个阶段的作用?

答:需求分析,架构设计,代码编写,QA,部署

59.在c#中using和new这两个关键字有什么意义,请写出你所知道的意义?using

指令 和语句 new 创建实例 new 隐藏基类中方法。

答:using 引入名称空间或者使用非托管资源

new新建实例或者隐藏父类方法

60.需要实现对一个字符串的处理,首先将该字符串首尾的空格去掉,如果字符串中

间还有连续空格的话,仅保留一个空格,即允许字符串中间有多个空格,但连续的空

格数不可超过一个.

答:string inputStr=" xx xx ";

inputStr=Regex.Replace(inputStr.Trim(),"*"," ");

 

61.什么叫做SQL注入,如何防止?请举例说明。

答:利用sql关键字对网站进行攻击。过滤关键字'等

62.什么是反射?

答:动态获取程序集信息

63.用Singleton如何写设计模式

答:static属性里面new ,构造函数private

64.什么是ApplicationPool?

答:Web应用,类似Thread Pool,提高并发性能。

65.什么是虚函数?什么是抽象函数?

答:虚函数:没有实现的,可由子类继承并重写的函数。抽象函数:规定其非虚

子类必须实现的函数,必须被重写。

66.什么是XML?

答:XML即可扩展标记语言。eXtensible Markup Language.标记是指计算机所能

理解的信息符号,通过此种标记,计算机之间可以处理包含各种信息的文章等。

如何定义这些标记,即可以选择国际通用的标记语言,比如HTML,也可以使用象

XML这样由相关人士自由决定的标记语言,这就是语言的可扩展性。XML是从SGML

中简化修改出来的。它主要用到的有XML、XSL和XPath等。

67.什么是WebService?UDDI?

答:Web Service便是基于网络的、分布式的模块化组件,它执行特定的任务,遵

守具体的技术规范,这些规范使得Web Service能与其他兼容的组件进行互操作。

UDDI的目的是为电子商务建立标准;UDDI是一套基于Web的、分布式的、为

WebService提供的、信息注册中心的实现标准规范,同时也包含一组使企业能将

自身提供的Web Service注册,以使别的企业能够发现的访问协议的实现标准。

68.什么是ASP.net中的用户控件?

答:用户控件一般用在内容多为静态,或者少许会改变的情况下..用的比较大..类

似ASP中的include..但是功能要强大的多。

69.列举一下你所了解的XML技术及其应用

答:xml用于配置,用于保存静态数据类型.接触XML最多的是web Services..和

config

70.ADO.net中常用的对象有哪些?分别描述一下。

答:Connection 数据库连接对象

Command数据库命令

DataReader数据读取器

DataSet数据集

71.什么是code-Behind技术。

答:ASPX,RESX和CS三个后缀的文件,这个就是代码分离.实现了HTML代码和服务

器代码分离.方便代码编写和整理.

72.什么是SOAP,有哪些应用。

答:simple object access protocal,简单对象接受协议.以xml为基本编码结构

,建立在已有通信协议上(如http,不过据说ms在搞最底层的架构在tcp/ip上的

soap)的一种规范WebService使用的协议..

73.C#中 property 与 attribute的区别,他们各有什么用处,这种机制的好处

在哪里?

答:一个是属性,用于存取类的字段,一个是特性,用来标识类,方法等的附加

性质

74.XML与 HTML 的主要区别

答:1. XML是区分大小写字母的,HTML不区分。

2.在HTML中,如果上下文清楚地显示出段落或者列表键在何处结尾,那么你可以

省略</p>或者</li>之类的结束标记。在XML中,绝对不能省略掉结束标记。

3.在XML中,拥有单个标记而没有匹配的结束标记的元素必须用一个 / 字符作为

结尾。这样分析器就知道不用查找结束标记了。

4.在XML中,属性值必须分装在引号中。在HTML中,引号是可用可不用的。

5.在HTML中,可以拥有不带值的属性名。在XML中,所有的属性都必须带有相应

的值。

75.c#中的三元运算符是?

答:?:。

76.当整数a赋值给一个object对象时,整数a将会被?

答:装箱。

77.类成员有_____种可访问形式?

答:this.;new Class().Method;

78.publicstatic const int A=1;这段代码有错误么?是什么?

答:const不能用static修饰。

79.floatf=-123.567F; int i=(int)f;i的值现在是_____?

答:-123。

80.委托声明的关键字是______?

答:delegate.

81.用sealed修饰的类有什么特点?

答:密封,不能继承。

82.在Asp.net中所有的自定义用户控件都必须继承自________?

答:Control。

83.在.Net中所有可序列化的类都被标记为_____?

答:[serializable]

84.在.Net托管代码中我们不用担心内存漏洞,这是因为有了______?

答:GC。

85.当类T只声明了私有实例构造函数时,则在T的程序文本外部,___可以___(可

以 or 不可以)从T派生出新的类,不可以____(可以 or 不可以)直接创建T的

任何实例。

答:不可以,不可以。

86.下面这段代码有错误么?

switch(i){

case():答://case()条件不能为空

CaseZero();

break;

case1:

CaseOne();

break;

case2:

dufault;答://wrong,格式不正确

CaseTwo();

break;

}

87.在.Net中,类System.Web.UI.Page 可以被继承么?

答:可以。

88..net的错误处理机制是什么?

答:.net错误处理机制采用try->catch->finally结构,发生错误时,层层上抛

,直到找到匹配的Catch为止。

89.利用operator声明且仅声明了==,有什么错误么?

答:要同时修改Equale和GetHash() ? 重载了"==" 就必须重载 "!="

90.在.net(C# or vb.net)中如何取消一个窗体的关闭。

答:private void Form1_Closing(object sender,

System.ComponentModel.CancelEventArgse)

{

e.Cancel=true;

}

91.在.net(C# or vb.net)中,Appplication.Exit 还是 Form.Close有什么

不同?

答:一个是退出整个应用程序,一个是关闭其中一个form。

 

92.某一密码仅使用K、L、M、N、O共5个字母,密码中的单词从左向右排列,密

码单词必须遵循如下规则:

(1) 密码单词的最小长度是两个字母,可以相同,也可以不同

(2) K不可能是单词的第一个字母

(3) 如果L出现,则出现次数不止一次

(4) M不能使最后一个也不能是倒数第二个字母

(5) K出现,则N就一定出现

(6) O如果是最后一个字母,则L一定出现

问题一:下列哪一个字母可以放在LO中的O后面,形成一个3个字母的密码单词?

A)K B)L C) M D) N

答案:B

问题二:如果能得到的字母是K、L、M,那么能够形成的两个字母长的密码单词的

总数是多少?

A)1个 B)3个 C)6个 D)9个

答案:A

问题三:下列哪一个是单词密码?

A)KLLN B) LOML C) MLLO D)NMKO

答案:C

93.62-63=1 等式不成立,请移动一个数字(不可以移动减号和等于号),使得等

式成立,如何移动?

答案:62移动成2的6次方

94.C#中 property 与 attribute的区别,他们各有什么用处,这种机制的好

处在哪里?

答:attribute:自定义属性的基类;property :类中的属性

95.在C#中,string str = null 与 string str = "" 请尽量使用文字或图

象说明其中的区别。

答:null是没有空间引用的;

"" 是空间为0的字符串;

96.abstract class和interface有什么区别?

答:声明方法的存在而不去实现它的类被叫做抽像类(abstract class),它用于要创建一个体现某些基本行为的类,并为该类声明方法,但不能在该类中实现该类的情况。不能创建abstract 类的实例。然而可以创建一个变量,其类型是一个抽像类,并让它指向具体子类的一个实例。不能有抽像构造函数或抽像静态方法。Abstract 类的子类为它们父类中的所有抽像方法提供实现,否则它们也是抽像类为。取而代之,在子类中实现该方法。知道其行为的其它类可以在类中实现这些方法。

接口(interface)是抽像类的变体。在接口中,所有方法都是抽像的。多继承性可通过实现这样的接口而获得。接口中的所有方法都是抽像的,没有一个有程序体。接口只可以定义static final成员变量。接口的实现与子类相似,除了该实现类不能从接口定义中继承行为。当类实现特殊接口时,它定义(即将程序体给予)所有这种接口的方法。然后,它可以在实现了该接口的类的任何对像上调用接口的方法。由于有抽像类,它允许使用接口名作为引用变量的类型。通常的动态联编将生效。引用可以转换到接口类型或从接口类型转换,instanceof 运算符可以用来决定某对象的类是否实现了接口。

97.<%# %> 和 <% %> 有什么区别?

答:<%# %>表示绑定的数据源

<%%>是服务器端代码块

98.重载与覆盖的区别?

答:1、方法的覆盖是子类和父类之间的关系,是垂直关系;方法的重载是同一个

类中方法之间的关系,是水平关系

2、覆盖只能由一个方法,或只能由一对方法产生关系;方法的重载是多个方法之

间的关系。

99.Overloaded的方法是否可以改变返回值的类型?

答:Overloaded的方法是可以改变返回值的类型。

100.C#可否对内存进行直接的操作?

答:在.net下,.net引用了垃圾回收(GC)功能,它替代了程序员不过在C#中。

 

 

SQL Server 数据库基础笔记分享(下)

 

前言

本文是个人学习SQL Server 数据库时的以往笔记的整理,内容主要是对数据库的基本增删改查的SQL语句操作约束,视图,存储过程,触发器的基本了解。

注:内容比较基础,适合入门者对SQL Server 数据库的了解!!!

正文

1.子查询

--把一个查询结果作为另外一个查询的查询源 
select * from (select * from Student where tbage between 3 and 5)
as ct where tbname=5 --ct是新创的表名

--把另外一个查询的结果作为当前查询的条件来使用。
--子查询中=、!= 、< 、> 、<= 、>=之后只能返回单个值,如果多个值就会报错
--解决办法 可以用in 代替
select * from Student
where tbage in(select tbage from Student where tbname=3)

select * from Student
where tbage=(select tbage from Student where tbname=3)

》》》》》》子查询分页《《《《《《

--1》显示第一页的数据
--分页查询的时候首先是将数据排序
select * from Student order by id desc

--2》第一页 显示5条数据
select Top 5 * from Student order by id desc

--3》第二页
select top 5 * from Student 
where id not in (select top 5 * from Student order by id desc)
order by id desc

--4》第三页
select top 5 * from Student 
where id not in (select top (2*5) * from Student order by id desc)
order by id desc

》》》开窗函数分页《《《

--第七页数据 每页5条数据
--over属于开窗函数

select * from
(
select * ,row_number() over( order by id desc) as paixu from Student
) as tbl
where tbl.paixu between 6*5+1 and 7*5

2.连表查询

--查询所有学生的姓名、年龄及所在班级 (班级在另一个表中)
--当多个列在不同的表中时,要跨表查询,所以一般可以使用inner join
--tc ts是对表名起的别名
select
ts.tsname,
ts.tsage,
tc.tclassname
from TblStudent as ts
inner join TblClass as tc on ts.tsclassid=tc.tclassid(只查询两个表中都有的数据)

--》》》full join 是查询所有的数据(没有的为空)

---子查询写法
select
tsname,
tsage,
班级名称=(select tclassname from TblClass where TblClass.tclassid=TblStudent.tsclassid)
from TblStudent

--查询学生姓名、年龄、班级及成绩(成绩属于第三张表)
select 
ts.tsname,
ts.tsage,
tc.tclasssname,
tscore.tenglish,
tscore.tmath
from TblStudent as ts
inner join TblClass as tc on ts.tsclassid=tc.tclassid 
inner join TblScore as tscore on tscore.tsid=ts.tsid

 

--》》》左外联接(左联接)

--查询没有参加考试的学生的姓名与编号
--把左表(left join 关键字左边的表)中的全部记录都显示出来,对于那些在右表中能找到匹配的记录,显示对应匹配数据,对于那些右表中找不到匹配的记录显示为null
select
ts.tsid,
ts.tsname,
TblScore.*
from TblStudent as ts
left outer join TblSore.tsid=ts.tsid   --outer可以不写

--》》》右外联接
--表示要将右表(right join 右边的表)中的所有数据都显示,左表中只显示那些匹配的数据。

select
ts.tsid,
ts.tsname,
TblScore.*
from TblStudent as ts
right outer join TblSore.tsid=ts.tsid

--右外联与左外联都是先将匹配的数据找到,然后再将那些没有匹配的数据添加进来,(注意:不是一起查询出来的,有先后顺序)

--》》》练习:查询所有学生(参加和未参加的考试)的学生姓名、年龄、成绩,如果没有参加考试显示缺考,如果小于english或者math 小于60分显示不及格
select
ts.tsname,
ts.tsage,
tscore.tsid,
case
when tscore.tenglish is null then '缺考'
else convert(varchar(10),tscore.tenglish)
end as 英语成绩,
case
when tscore.tmath id null then '缺考'
else convert (varchar(10),tscore.tmath)
end as 数学成绩,
是否报考=
case
when tscore.tscoreid is null then '是'
else '否'
end
from TblStudent as ts
left join TblScore as tscore on ts.tsid=tscore.tsid

3.视图

视图本身并不存储数据,只是存储的查询语句,如果把真实表中的数据修改后,则通过视图查询到的结果也变了。

视图的目的是方便查询,所以一般情况下不能对视图进行增删改查


--在视图中的查询语句,必须为每一列创建一个列名
create view vw2
as
select
tsname,
case
when tsage>13 and tsage<=16 then '少年'
when tsage>50 then '老年'
else '青壮年'
end as 称呼
from TblStudent

--在视图中不能使用order by语句。除非:另外还指定了top 或for xml
--错误
create view vw3
as 
select * from TblStudent order by tsage desc

--正确
create view vw3
as 
select top 3 * from TblStudent order by tsage desc

4.声明变量与使用

--》》》局部变量
--声明变量
declare @name varchar(10)
declare @age int

--赋值
set @name='yhz'
set @age=17

--输出值
print @name
print @age

--使用set与select为变量赋值的区别
declare @rcount int 
set @rcount=(select count(*) from TblStudent)
print @rcount

declare @rcount int 
select @rcount=count(*) from TblStudent
print @rcount


--》》》全局变量
print @@language
print @@version
print 'aaa'+100
--通过判断@@error变量中是否不为0,就可以判断上一条sql语句执行是否出错了
--如果@@error为0,表示上一条sql语句执行没出错,如果@@error不为0,则表示上一条sql语句出错了。
print@@error

--》》》通过while计算1-100之间所有奇数的和

--声明变量并初始化

declare @sum int=0
declare @i int =1
while @i<=100
begin
if @i%2<>0
begin
set @sum=@sum+@i
end
end
print @sum

5.事务

事务有四个属性:原子性 一致性 隔离性 持久性
原子性:对于数据修改,要么全都执行,要么全都不执行
一致性:当数据完成时,数据必须处于一致状态
隔离性:对数据进行修改的所有并发事务时彼此隔离的。这表明事务必须是独立的,它不应以任何方式依赖于或影响其他事务
永久性:事务完成后,他对数据库的修改被永久保持,事务日志能够保持事务的永久性

--打开事务
begin transaction

--提交事务
commit transaction

--回滚事务
rollback transaction


--账户A给账户B转账 当一方出问题时,两个语句都不执行
begin tran
declare @sum int=0
update bank set balance =balance-1000 where cid='0001'
set @sum=@sum+@@error
update banl set balance =balance+1000 where cid='0002'
set @sum=@sum+@@error

if @sum<>0
begin
rollback tran
print '回滚'
end
else 
begin
commit tran
print '提交了'
end

6.存储过程

--创建一个自定义的存储过程
create proc usp_HelloWorld
as 
begin
print 'hello world'
end

--输出存储过程
exec usp_HelloWorld

--创建一个存储过程计算两个数的和
create procedure usp_Add
@num1 int,
@num2 int
as 
begin
print @num1+@num2
end

--输出值
exec usp_Add 100,230


--存储过程中的参数的问题
--存储过程如果有参数,则调用的时候必须为参数赋值
exec usp_Add --不传参数则报错


--第二个参数如果用户不传,则有一个默认值
create procedure usp_Add
@num1 int,
@num2 int 1000 --为存储过程的参数设置默认值
as 
begin
print @num1+@num2
end

--创建分页存储过程
create proc usp_PageBum
@pageSize int, --每页显示的数量
@pageIndex int --第几页
as
begin
select * from (select *,row_number()over (order by CityID asc)as num from S_City )as s 
where s.num between (@pageIndex -1)*@pageSize +1 and @pageIndex *@pageSize 
end
--查询第5页内容每页显示10条数据
exec usp_PageBum 10,5

--删除一个存储过程
drop proc usp_Add

7.触发器

尽量避免在触发器中执行耗时操作,因为触发器会与sql语句认为在同一个事务中(事务不结束,就无法释放锁)

--创建插入数据触发器
create trigger tri_Teacher_insert_after
on Teacher after insert
as
begin
declare @id int
declare @name varchar(10)
declare @phone int
declare @mail varchar(50)
select @id=tcid,@name=tcname,@phone=tcphone,@mail=tcmail from inserted

print @id
print @name
print @phone
print @mail
end

--插入数据
insert into Teacher values('网名好','12352536','Wjifdfji@qq.com')


--创建删除数据触发器
--不能有主键
create trigger tri_Teacher_after
on Teacher after delete
as
begin
insert into TeacherBak
select * from deleted
end

--删除数据
--sql server中的触发器是表级触发器,无论删除多少行或者插入多少行,只触发一次
--是按语句来触发的,每次执行一次语句,触发一次触发器
delete from Teacher where tcid>18

8.游标

--1.定义游标
declare cur_Student cursor fast_forward for select * from Student

--2.打开游标
open cur_Student

--2.1 对游标的操作
--将每条数据读取并输出

--2.1.1将游标向后移动一条
fetch next from cur_Student

--将游标循环向后移动,直到末尾
while @@fetch_status=0
begin
fetch next from cur_Student
end


--3.关闭游标
close cur_Student

--4.释放资源
deallocate cur_Student

9.(补充)全局临时表,局部临时表

局部临时表:表名以#为开头。只在当前会话中有效,不能跨连接访问。如果直接在连接会话中创建,则当前连接断开后删除,如果是在存储过程中创建的,则存储过程执行完毕后删除

全局临时表:表名以##为开头。多个会话可共享全局临时表。当创建全局临时表的会话断开,并且没有用户正在访问全局临时表时删除

10.(补充)约束

--删除一列(EmpAddress列)
alter table Class drop column EmpAddress

--增加一列(增加一列EmpAddr varchar(1000))
alter table Class Add EmpAddr varchar(1000)

--修改一下Emp 的数据类型(varchar(200))
alter table Class alter column Emp varchar(200)

--为EmpId增加一个主键约束
alter table Class add constraint PK_Class_EmpId primary key(EmpId)

--为EmpName增加一个唯一约束
alter table Class add constraint UQ_Class_EmpName unique(EmpName)

--为性别增加一个默认约束,默认为男
alter table Class add constraint DF_Class_EmpGender default('男') for EmpGender

--为年龄增加一个检查约束,年龄必须在1—120岁之间(包含)
alter table Class add constraint CK_Class_EmpAge check(EmpAge>=0 and EmpAge<=120)

--增加外键约束,表Class中有一列EmpDeptId引用Student表中的DeptId
alter table Class add EmpDeptId int not null
alter table Student add constraint PK_Student_DeptId primary key(DeptId)

alter table Class add constraint FK_Class_Student foreign key(EmpDeptId)
references Student(DeptId)


--一条语句删除多个约束,约束名用 逗号 隔开
alter table Class drop constraint
PK_Student_DeptId,
FK_Class_Student,
CK_Class_EmpAge


--用一条语句为表增加多个约束
alter table Class add 
constraint PK_Student_DeptId primary key(DeptId),
constraint CK_Class_EmpAge check(EmpAge>=0 and EmpAge<=120),
add constraint DF_Class_EmpGender default('男') for EmpGender

后记

笔记不全,还请见谅!希望对你有所提高。

 

SQL Server 数据库基础笔记分享(上)

 

前言

本文是个人学习SQL Server 数据库时的以往笔记的整理,内容主要是对数据库的基本增删改查的SQL语句操作约束,视图,存储过程,触发器的基本了解。

注:内容比较基础,适合入门者对SQL Server 数据库的了解!!!

正文

1.主键:

主键的作用:保证表中的每条数据的唯一性
特点: 主键不能重复 不能为空
分类:
逻辑主键:选择为表中增加的那些“自动编号”列或者“GUID”列为主键(没有实际业务上的意义)的主键 (建议使用逻辑主键)
业务主键:选择表中那些在业务中有实际意义的列作为主键
》》》》》》》》》选择主键的策略,选什么样的列作为主键《《《《《《《《《
1》主键,建议选择那些一般不会被修改的列
2》选择单列,不选择多列(不用组合主键)
3》选择那些简单列(整数列(自动编号))

 

2.char(),nchar(),varchar()之间的区别

》》》》》》》》》char(10)与varchar(10)的区别《《《《《《《《《
char(10) 固定长度,表示在数据库中存储的时候占用10个字节的空间,如果超出10个则报错,如果不够10个则用空格补全。 
varchar(10) 可变长度,表示该列最多可以存储10个字节,如果实际存储不够10个字节,则会在存储的时候自动计算一下实际的存储个数,而动态的改变长度。【节省空间】

》》》》》》》》》char(10)与nchar(10)的区别《《《《《《《《《

char(10) 可以存储10个字母或者5个汉字。 用来存储数据的时候,英文站1个字节,中文站2个字节。

nchar(10) 表示可以存储10个字母或10个汉字,因为每个字符都是按照unicode方法来存储的。当使用nchar(10),来存储数据的时候无论存储的是中文还是英文都是每个字符占2个。

 

3. 创建数据库

--创建一个数据库
create database School

--删除数据库
drop database School

--创建数据库的时候,指定一些数据库的相关参数。
create database School
on primary --主数据文件
(
name='School',
size=10mb,
filename='c:school.mdf',
filegrowth=10%,
maxsize=100mb
)
log on --日志文件
(
name='School_log',
filename='c:school.ldf',
size=5mb,
filegrowth=5mb,
maxsize=50mb
)

--切换数据库
use school
go

4. 创建表

--创建表
create table Class
(
ClassId int identity(1,1) primary key,
ClassName varchar(50) not null,
ClassDesc varchar(50) not null
)

--删除表
drop table Class

--向Class表中插入数据
insert into Class(ClassName,ClsDesc)values('大三','三年');

--insert into...values.. 这种写法每次只能插入一条数据

--向Class表中插入多条数据
--重复数据不重复插入,union关键字本身就具有去掉重复的意思
--union | union all (重复插入)
insert into Class
select '大三','三年' union
select '三五','间谍' union
select '一一','多久' union
select '六七','得到'


--将Class表中的数据备份到Student表中
--这种写法会将Class表中的所有数据插入到Student表中
--前提是Student表不存在,如果这个表存在则报错。
select * into Student from Class


--向一个已经存在的表中插入数据,数据的来源是Class表
insert into Student(ClassName,ClsDesc) 
select ClassName,ClsDesc from Class

 

--查询表中数据
select * from Class

5.update 数据

--将所有年龄小于20岁的人的年龄都改成19(tage是Class表后加属性)
update Class set tage=19 where tage<20

--将年龄为19岁的并且性别为0的人的姓名两边★改为☆
update Class set ClassName =replace (tname,'★','☆') where tage=19 and tgender=0

6.删除数据

delete from Class --删除所有数据 自动编号没有恢复到默认值 可以根据条件来删除
truncate table Class --重新设置了自动编号 删除只能一次性都清空,不能根据条件来删除 清除速度(性能)比delete语句快的多


delete from Class where tage=19 or tage is null --删除19岁或者空值

》》》》》》》》》删除重复数据只保留一条(id最小的一条)《《《《《《《《《
》》》》》》》》》删除表中多余的重复记录,重复记录是根据单个字段(peopleId)来判断,只留有rowid最小的记录 《《《《《《《《《
delete from people 
where peopleName in (select peopleName from people group by peopleName having count(peopleName) > 1) 
and peopleId not in (select min(peopleId) from people group by peopleName having count(peopleName)>1)

 

7.条件查询,模糊查询

--查询数学没有及格的学生的学号
select 
fid as 学号,
fmath as 分数
from MyStudent where fmath<60

--查询年龄在20-30岁之间的男学生
select
fname as 姓名 from MyStudent where fage between 20 and 30 and fgender='男'

--查询班级id 1 2 3 的所有学生
select * from MyStudent where classid in (1,2,3)

--查询所有姓赵的同学 (通配符%表示:任意多个任意字符)
select * from MyStudent where fname like '赵%'

--查询出姓名中只要包含一个‘民’字即可。
select * from MyStudent where fname like '%民%'

--查询所有姓赵的同学,并且姓名字数是3个
--通配符 _ :表示任意的单个字符。
select * from MyStudent where fname like '赵__'
select * from MyStudent where fname like '赵%' and len(fname)=3

--查询出姓名中包含‘民’或‘用’的同学
--通配符[]:表示中括号中的任意个字符,只选一个匹配
--通配符 ^a :表示除了a这个字符都行。
select * from MyStudent where fname like '%[民用]%'

8.聚合函数

--查询数学成绩最高低分
select max(fMath) as 数学成绩最高分 from MyStudent
select min(fMath) as 数学成绩最低分 from MyStudent

--平均分(计算平均分的时候对空值不处理)
select avg(fMath) as 平均分 from MyStudent

--求数据记录中的总条数(总人数)
select count(*) as 班级总人数 from MyStudent


select
最高分=(select max(fMath) as 数学成绩最高分 from MyStudent),
最低分=(select min(fMath) as 数学成绩最低分 from MyStudent),
平均分=(select avg(fMath) as 平均分 from MyStudent)


--分数评级
--90以上 优秀
--80以上 良好
--70以上 中
--70以下 差
select chengji,
评级=
case
when shuxue>=90 then '优秀'
when shuxue>=80 then '良好'
when shuxue>=70 then '中'
else '差'
end
from Student

9.null 问题

--请查询出学生表中所有数学成绩为null的人的信息
--null在数据库中表示unknow(不知道),判断一个值是否为null,也就不能用=或者<>来判断
select * from MyStudent where fMath=null 错误(不返回任何数据)

正确 select * from MyStudent where fMath is null

--查询所有fmath为非null的值
select * from MyStudent where fMath is not null


--null值与任何数据运算后得到的还是null值。
update MyStudent set fage=fage+1 where fid=1

10.分组group by

--统计出mystudent表中,男女同学的个数

select 
fgender as 性别, --这时,count(*)统计的是每一组的记录条数, 不是总条数
count(*) as 人数
from MyStudent group by fgender --先执行group by语句分组,分完组在统计每 组个数。 分出来几个组,那么count(*)就统 计几次


--查询班级的男同学的人数大于2的信息

--having是group by的条件对分组后的数据进行筛选(与where类似,都是筛选,只不过having是用来筛选分组后的组的)
select 
classid as 班级号,
count(*) as 班级人数
from TblStudent
where fgender='男'
group by classid
having count(*)>2

》》》》》》》》》语句执行顺序《《《《《《《《《

select
--distinct / top 之类的关键字
fgender as 性别, --5》选择列
count(*) as 人数
from MyStudent --1》先从表中拿到数据
where fage>30 --2》从MyStudent的数据中筛选出所有年龄大于30岁的任的信息
group by fgender --3》按照性别分组,分完组得到一个新的结果集
having count(*)>500 --4》基于分组以后的结果集,然后再筛选,筛选出那些组中记录大于500的组
order by 人数 asc --6》最后把显示出来的结果排序


--语句执行顺序
from > where > group by > having > select > order by

11.日期函数

--请查询出所有入职一年以上的员工信息
select * from TblStudent
where dateadd(year,1,tsday)<getdate()


--计算两个时间差
--查询90年距今是多少年
select datediff(year,'1990-9-9',getdate())


--查询一个日期的特定部分
select year(getdate())
select datepart(year,getdate())

--输出所有数据中通话时间最长的5条记录。
select top 5 *,'通话时长(秒)'=datediff(second,Startdatetime,Enddatetime) from Calltecords order by datediff(second,Stardatetime,enddatetime) desc

后记

下篇分享视图、触发器等,分页查询、子查询、连表查询等

 

Asp.Net MVC4中的全局过滤器

 

可以对整个项目进行全局监控。

              新建一个MVC4项目,可以在global.asax文件中看到如下代码:  FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);

              表示注册全局过滤器. 

               GlobalFilters是全局过滤器的集合,可以通过add方法添加过滤器,默认情况下,HandleErrorAttribute过滤器被添加到集合中。

                接下来我们创建一个自定义过滤器,然后添加到全局过滤器集合中。

               1.创建自定义过滤器

                  创建自定义过滤器要继承ActionFilterAttribute类。我们创建一个名称为CustomerFilterAttribute的过滤器,在action里面记录下时间。

                    代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public  class  CustomerFilterAttribute : ActionFilterAttribute 
  
   
       public  override  void  OnActionExecuting(ActionExecutingContext filterContext) 
      
           base .OnActionExecuting(filterContext); 
           filterContext.HttpContext.Response.Write( "开始时间:" +DateTime.Now.ToString()+ "<br/>" ); 
      
   
       public  override  void  OnActionExecuted(ActionExecutedContext filterContext) 
      
           base .OnActionExecuted(filterContext); 
           var  controllerName = filterContext.RouteData.Values[ "controller" ].ToString(); 
           var  actionName = filterContext.RouteData.Values[ "action" ].ToString(); 
   
           filterContext.HttpContext.Response.Write( "结束时间:"  + DateTime.Now.ToString() +  "<br/>" ); 
           filterContext.HttpContext.Response.Write( "controller:"  +controllerName+ ",action:" +actionName); 
      
  

  

       2.注册全局过滤器

                     过滤器创建完成后,我们把这个过滤器添加到全局过滤器中,使用  filters.Add(new CustomerFilterAttribute());方法,

                       代码如下:

1
2
3
4
5
6
7
8
public  class  FilterConfig 
   
        public  static  void  RegisterGlobalFilters(GlobalFilterCollection filters) 
       
            filters.Add( new  HandleErrorAttribute()); 
            filters.Add( new  CustomerFilterAttribute()); 
       
   

    接下来我们运行项目中的每一个页面,都会看到页面中输出时间和controller名称,效果图如下:

                                

 

 

C#语法——泛型的多种应用

 

本篇文章主要介绍泛型的应用。

泛型是.NET Framework 2.0 版类库就已经提供的语法,主要用于提高代码的可重用性、类型安全性和效率。

泛型的定义

下面定义了一个普通类和一个泛型类,我们可以明确看到泛型类和普通类最大的区别就是多了一个<T>。

所以,这个<T>就标记了,这个类是泛型类。其中这个T,也可以写成A,B,C,D或其他字符。

1
2
3
4
public  class  Generic
{
     public  String Name;
}
1
2
3
4
public  class  Generic<T>
{
     public  T Name;
}

泛型,顾名思义,就是泛指的类型。好比男人,女人,白人,黑人,可以泛称为【人】。

但类型只能是一个类型。 那么泛型和类型之间是什么关系呢?

其实很简单,泛型在定义的时候,是泛指类型;在使用的时候,就需要被指定,到底使用哪个类型。

即,使用时,就不在是泛指类型,而是特定类型。

好比,定义时,定义了一个人。但在使用时,必须明确指定,到底是黑人还是白人。

泛型的使用

泛型类跟普通类的使用方式一样,都需要实例化对象,再由对象来调用内部的属性或方法。

下面代码实例化了泛型Generic,实例化时,还指定了该泛型Generic的指定类型为String。

所以要给泛型Generic的属性Name赋值,就需要赋值字符串类型的值。

1
2
3
4
5
public  static  void  Excute()
{
     Generic<String> gs =  new  Generic<String>();
     gs.Name =  "Kiba518" ;
}

下面代码定义了一个Int类型的泛型Generic。

1
2
3
4
5
public  static  void  Excute()
{
     Generic< int > gs =  new  Generic< int >();
     gs.Name = 518;
}

泛型的默认值

泛型的默认值,如下面代码所示。需要使用default(T)来赋值。

不管泛型到底是String,int,bool或者是一个Class类型,都可以被自动赋值。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public  static  void  Excute()
{
     Generic< int > gs =  new  Generic< int >();
     gs.Name = 518;
     Generic<Task> gsTask =  new  Generic<Task>();
     gsTask.Name =  new  Task(()=> {
         Console.WriteLine( "Kiba518" );
     });
}
 
public  class  Generic<T>
{
     public  T Name =  default (T);
}

泛型的约束

在泛型类中,有个特别的约束可供我们使用。

当我们不显示的声明时,这个约束不存在。但当我们显示的声明的时候,这个约束就会执行。

下面,我们来看看这个特别的约束。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public  static  void  Excute()
{
     Generic<FanXing> gFanXing =  new  Generic<FanXing>();
     Generic<Base> gFanXingBase =  new  Generic<Base>();
     //Generic<string> gs = new Generic<string>(); 这样定义会报错
}
public  class  Generic<T>  where  T : Base
{
     public  T Name =  default (T);
}
public  class  Base 
{
     public  string  Name {  get set ; }
}
public  class  FanXing : Base
{
     public  new  string  Name {  get set ; }
}

如上面代码所示,【where T : Base】就是这个特别的约束。

当显示声明这个约束的时候,定义会限制泛型的类型。

什么是限制泛型的类型呢?

很简单,泛型T,是泛指某一个类型。我们在定义泛型类时,还需显示的指定类型,此时我们显示指定的类型,要受这个限制。

这个限制就是指【where T : Base】。

它的限制是,要求我们指定的类型T必须是Base,或者该类型继承自Base,如FanXing类。

泛型的函数

在C#中,泛型不仅可以用于类,还可以直接用于函数。

具体使用方式如下:

1
2
3
4
5
6
7
8
9
10
11
12
public  static  void  Excute()
{
     GenericFunc gf =  new  GenericFunc();
     gf.FanXingFunc<FanXing>( new  FanXing() { Name= "Kiba518" });
}
public  class  GenericFunc
{
     public  void  FanXingFunc<T>(T obj)
     {
         Console.WriteLine(obj.GetType());
     }
}

很简单,调用泛型函数的时候,指定泛型函数的[指定类型]即可。

但是,这里我们发现一个问题,那就是,在泛型函数里,使用泛型对象的时候,我们发现对象都是object类型的。

那我们如果想使用泛型对象里的属性和方法时,要怎么办呢?

也很简单,反射就可以了。

下面我们添加一个反射函数GetPropertyValue,专门用来获取属性。

1
2
3
4
5
6
7
8
9
10
11
12
13
public  class  GenericFunc
{
     public  void  FanXingFunc<T>(T obj)
     {
         var  name = GetPropertyValue(obj,  "Name" );
         Console.WriteLine(name);
     }
     public  object  GetPropertyValue( object  obj,  string  name)
     {
         object  drv1 = obj.GetType().GetProperty(name).GetValue(obj,  null );
         return  drv1;
     }
}

输出结果如下:

这样我们就得到了我们想要的结果,如果想使用泛型类里的函数,道理也一样,只需要用反射来调用即可。

结语

看到这里,有些同学可能会觉得泛型很复杂,连使用其对象下的属性,都得反射,太繁琐了,还不如不用呢。

有这样想法的同学,心里想想就好了,如果对老司机这么说,他肯定会内心默默的微笑,然后对你说,你想的没错。

然后,你就没有然后了。

泛型的应用,开篇已经说了,主要用在提高代码的可重用性、类型安全性和效率上。

如果只是定义一个类,调用一个属性,那泛型的存在就是鸡肋。

但事实上,我们的系统永远只有更复杂,更复杂,更复杂。因此泛型才有了用武之地。

C#语法——委托,架构的血液

C#语法——元组类型

C#语法——await与async的正确打开方式

转载于:https://www.cnblogs.com/cjm123/p/9471751.html

  • 0
    点赞
  • 0
    评论
  • 2
    收藏
  • 一键三连
    一键三连
  • 扫一扫,分享海报

表情包
插入表情
评论将由博主筛选后显示,对所有人可见 | 还能输入1000个字符
©️2021 CSDN 皮肤主题: 编程工作室 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值