[?]|登录|免费注册®
高级...

浏览选项

  • 手机网站排名
  • 手机网用户圈
  • UA设置:(默认)
  • 设置浏览参数
  • 重新下载页面

[+](无标题)

?博客园_首页代码改变世界uuid:a43c6e56-9b52-42f9-a7d1-ee0600f885f7;id=53462014-08-22T21:05:55Zfeed.cnblogs.comhttp://www.cnblogs.com/edisonchou/p/3930414.html走向面试之经典的数据库基础:二、SQL进阶之case、子查询、分页、join与视图 - Edison Chou一、CASE的两种用法1.1等值判断->相当于switch case (1)具体用法模板: CASE expression WHEN value1 THEN returnvalue1 WHEN value2 THEN returnvalue2 WHEN valu...2014-08-22T18:00:00Z2014-08-22T18:00:00ZEdison Chouhttp://www.cnblogs.com/edisonchou/<p><strong>一、CASE的两种用法</strong></p><p><strong>1.1&nbsp;等值判断-&gt;相当于switch case</strong></p><p>  (1)具体用法模板:</p><p>    CASE <strong>expression</strong></p><p>&nbsp;      WHEN <strong>value1</strong> THEN returnvalue1</p><p>&nbsp;      WHEN <strong>value2</strong> THEN returnvalue2</p><p>&nbsp;      WHEN <strong>value3</strong> THEN returnvalue3</p><p>      &nbsp;ELSE defaultreturnvalue</p><p>    END&nbsp;</p><p>  (2)具体使用示例:</p><p>  假设我们有一个论坛网站,其中有一张User表{ UId,Name,Level },Level是一个int类型,代表了用户等级类型,例如:1代表骨灰,2代表大虾等;我们就可以用CASE来对其进行等值判断了:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> Name,Rank<span style="color: #808080;">=</span><span style="color: #000000;">(<br/> </span><span style="color: #ff00ff;">case</span> <span style="color: #0000ff;">Level</span><br/> <span style="color: #0000ff;">when</span> <span style="color: #800000; font-weight: bold;">1</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">骨灰</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">when</span> <span style="color: #800000; font-weight: bold;">2</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">大虾</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">when</span> <span style="color: #800000; font-weight: bold;">3</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">菜鸟</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">end</span><span style="color: #000000;"><br/>) <br/></span><span style="color: #0000ff;">from</span> <span style="color: #ff00ff;">User</span></div><p><strong>1.2 条件判断-&gt;相当于if else if else</strong></p><p>  (1)具体用法模板:</p><p>    CASE</p><p>      WHEN <strong>condition1</strong> THEN returnvalue1</p><p>      WHEN <strong>condition2</strong> THEN returnvalue2</p><p>      WHEN <strong>condition3</strong> THEN returnvalue3</p><p>      ELSE defaultreturnvalue</p><p>    END</p><blockquote><p><strong>注意:</strong>then后面返回的数据类型要一致, returnvalue1、 returnvalue2、 returnvalue3的数据类型必须一致。</p></blockquote><p>  (2)具体使用示例:</p><p>  假设我们有一张Score成绩表,里面记录有所有同学的成绩,此时我们想要对所有成绩进行一个评级,比如成绩如果&gt;=90那么则评为A级,&gt;=80且&lt;90则评为B级,这里我们怎来写呢?</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> studentId,rank<span style="color: #808080;">=</span><span style="color: #000000;">( <br/> </span><span style="color: #ff00ff;">case</span><br/> <span style="color: #0000ff;">when</span> english <span style="color: #808080;">between</span> <span style="color: #800000; font-weight: bold;">90</span> <span style="color: #808080;">and</span> <span style="color: #800000; font-weight: bold;">100</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">A</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">when</span> english <span style="color: #808080;">between</span> <span style="color: #800000; font-weight: bold;">80</span> <span style="color: #808080;">and</span> <span style="color: #800000; font-weight: bold;">89</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">B</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">when</span> english <span style="color: #808080;">between</span> <span style="color: #800000; font-weight: bold;">70</span> <span style="color: #808080;">and</span> <span style="color: #800000; font-weight: bold;">79</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">C</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">when</span> english <span style="color: #808080;">between</span> <span style="color: #800000; font-weight: bold;">60</span> <span style="color: #808080;">and</span> <span style="color: #800000; font-weight: bold;">69</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">D</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">when</span> english <span style="color: #808080;">&lt;</span> <span style="color: #800000; font-weight: bold;">60</span> <span style="color: #0000ff;">then</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">E</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">else</span> <span style="color: #ff0000;">'</span><span style="color: #ff0000;">缺考</span><span style="color: #ff0000;">'</span><br/> <span style="color: #0000ff;">end</span><span style="color: #000000;"><br/>)<br/></span><span style="color: #0000ff;">from</span> Score</div><p><strong>二、子查询的用法</strong></p><p><strong>2.1 子查询初步</strong></p><p>  就像使用普通的表一样,被当作结果集的查询语句被称为子查询。所有可以使用表的地方几乎都可以使用子查询来代替。例如:我们如果要找到所有计科一班的同学信息,可以首先通过T_Class表找到计科一班的Id,然后再在T_Student表中找到所有ClassId为计科一班Id的行即可。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span> T_Student <span style="color: #0000ff;">where</span> ClassId<span style="color: #808080;">=</span><span style="color: #000000;"><br/>(<br/> </span><span style="color: #0000ff;">select</span> Id <span style="color: #0000ff;">from</span> T_Class <span style="color: #0000ff;">where</span> Name<span style="color: #808080;">=</span><span style="color: #ff0000;">'</span><span style="color: #ff0000;">计科一班</span><span style="color: #ff0000;">'</span><span style="color: #000000;"><br/>)</span></div><p><strong>2.2 单值子查询</strong></p><p>  只有返回且仅返回一行、一列数据的子查询才能当成单值子查询。例如我们上面提到的例子,子查询中只返回了一个ClassId,这就是单值子查询。当子查询跟随在=、!=、&lt;、&lt;=、&gt;、&gt;=,&lt;&gt; 之后,或子查询用作表达式,只能使用单值子查询。</p><p><strong>2.3 多值子查询</strong></p><p>  如果子查询是多行单列的子查询,这样的子查询的结果集其实是一个集合,那么可以使用in关键字代替=号。例如:我们如果想快速地在T_Student表中删除计科一班和计科二班的所有学生记录,我们可以使用in关键字:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">delete</span> <span style="color: #0000ff;">from</span> T_Student <span style="color: #0000ff;">where</span> ClassId <span style="color: #808080;">in</span><span style="color: #000000;"><br/>(<br/> </span><span style="color: #0000ff;">select</span> Id <span style="color: #0000ff;">from</span> T_Class <span style="color: #0000ff;">where</span> Name<span style="color: #808080;">=</span><span style="color: #ff0000;">'</span><span style="color: #ff0000;">计科一班</span><span style="color: #ff0000;">'</span> <span style="color: #808080;">or</span> Name<span style="color: #808080;">=</span><span style="color: #ff0000;">'</span><span style="color: #ff0000;">计科二班</span><span style="color: #ff0000;">'</span><span style="color: #000000;"><br/>)</span></div><p><strong>2.4 Exists&mdash;你存在我深深的脑海里</strong></p><p>  exists是用来判断是否存在的,当exists查询中的查询存在结果时则返回真,否则返回假。not exists则相反。</p><p>  exists做为where 条件时,是<strong>先对where 前的主查询询进行查询,然后用主查询的结果一个一个的代入exists的查询进行判断</strong>,如果为真则输出当前这一条主查询的结果,否则不输出。</p><p>  exists后面的查询称为相关子查询,即<strong>子查询的查询条件依赖于外层父查询中的某个属性值</strong>,其处理过程一般为:先取外层查询中的第一个元组,根据它与内层查询中的相关属性值处理内层查询,若where子句返回true,则将此元组放入结果表中,然后取外层查询中的下一个元组,重复这个过程直到全部检查完毕为止。</p><p>  例如:我们有一张人员信息表,里边有一个人员类型Id字段(pTypeId),它是一个外键,对应着人员类型表的主键ptId。如果我们有以下的SQL语句,使用Exists关键字则可以有如下的理解:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span> Employee e <span style="color: #0000ff;">where</span> <span style="color: #808080;">exists</span><span style="color: #000000;"> <br/>(</span><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span> EmployeeType et <span style="color: #0000ff;">where</span> e.pTypeId<span style="color: #808080;">=</span>et.ptId)</div><p>  那么,在这句SQL的执行过程中,我们可以将其理解为一个双重的for循环,外边是主表的循环遍历,然后将其放到一个temp变量中,再进入从表的for循环,并与从表的项进行一个一个的按照匹配规则(这里是e.pTypeId=et.ptId)进行匹配,如果有匹配成功则返回true,否则返回false。</p><p><img style="display: block; margin-left: auto; margin-right: auto;" src="http://images.cnitblog.com/blog/381412/201408/230055453937284.jpg" alt="" width="861" height="170" /></p><p><strong>三、手写分页SQL代码</strong></p><p>  这里假设每页的页大小为10条记录</p><p><strong>3.1 利用Top N进行简单分页</strong></p><p>  (1)如果我们要获取第一页的数据,也就是前10个:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #0000ff;">top</span> <span style="color: #800000; font-weight: bold;">10</span> * <span style="color: #0000ff;">from</span> Account</div><p>  (2)现在我们要获取第一页之后的数据,也就是第20个~最后一个:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span> Account <span style="color: #0000ff;">where</span> Id <span style="color: #808080;">not</span> <span style="color: #808080;">in</span> (<span style="color: #0000ff;">select</span> <span style="color: #0000ff;">top</span> <span style="color: #800000; font-weight: bold;">10</span> Id <span style="color: #0000ff;">from</span> Account) </div><p>  (3)现在我们对第20个~最后一个的数据集中取前10个也就成为第二页的数据了:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #0000ff;">top</span> <span style="color: #800000; font-weight: bold;">10</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span> Account <span style="color: #0000ff;">where</span> Id <span style="color: #808080;">not</span> <span style="color: #808080;">in</span> (<span style="color: #0000ff;">select</span> <span style="color: #0000ff;">top</span> <span style="color: #800000; font-weight: bold;">10</span> Id <span style="color: #0000ff;">from</span> Account) </div><p>  (4)将上述代码总结为分页代码,设页大小为pageSize,请求页号为pageIndex:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #0000ff;">top</span> @pageSize <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span> Account <span style="color: #0000ff;">where</span> Id <span style="color: #808080;">not</span> <span style="color: #808080;">in</span> (<span style="color: #0000ff;">select</span> <span style="color: #0000ff;">top</span> ((@pageIndex<span style="color: #808080;">-</span><span style="color: #800000; font-weight: bold;">1</span>)<span style="color: #808080;">*@</span>pageSize) Id <span style="color: #0000ff;">from</span> Account) </div><blockquote><p><strong>PS:</strong>这种分页方式的缺点是如果要取很多页之后的数据,那么就要取出前面很多页的ID,查询开销较大,执行效率也就低下。</p><p><img style="display: block; margin-left: auto; margin-right: auto;" src="http://images.cnitblog.com/blog/381412/201408/230114102373020.jpg" alt="" /></p><p>  从图中可以看出,在小数据量的对比下,Top N的查询开销较小。但是在大数据量的对比下,Row_Number的方式会取得更高的查询效率以及较小的开销。</p></blockquote><p><strong>3.2 利用Row_Number()进行高效分页</strong></p><p>  (1)SQL Server 2005后增加了Row_Number函数,可以简化分页代码的实现。首先,Row_Number()是一个排序函数,它可以生成一个有序的行号(如果单靠ID来排序,中间存在断层,例如某一个ID行已经被删除了)。根据MSDN的定义:返回结果集分区内行的序列号,每个分区的第一行从 1 开始。而排序的标准是什么呢?这个就要靠紧跟其后的OVER()语句来定义了。这里我们可以通过一个示例来看看,其生成的行号如何。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span>,ROW_NUMBER() <span style="color: #0000ff;">OVER</span>(<span style="color: #0000ff;">order</span> <span style="color: #0000ff;">by</span> Id) <span style="color: #0000ff;">as</span> RowNum <span style="color: #0000ff;">from</span> Account</div><p><img style="display: block; margin-left: auto; margin-right: auto;" src="http://images.cnitblog.com/blog/381412/201408/230128245182842.jpg" alt="" /></p><p>  (2)根据ROW_NUMBER()的使用,我们可以将其应用到分页上,于是我们可以写出以下的代码来实现获取第二页的数据集:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span><span style="color: #000000;"> (<br/> </span><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span>,ROW_NUMBER() <span style="color: #0000ff;">OVER</span>(<span style="color: #0000ff;">Order</span> <span style="color: #0000ff;">by</span> Id) <span style="color: #0000ff;">as</span> rownum <span style="color: #0000ff;">from</span><span style="color: #000000;"> Account<br/>) </span><span style="color: #0000ff;">as</span><span style="color: #000000;"> t<br/></span><span style="color: #0000ff;">where</span> t.rownum <span style="color: #808080;">between</span> <span style="color: #800000; font-weight: bold;">11</span> <span style="color: #808080;">and</span> <span style="color: #800000; font-weight: bold;">20</span><br/><span style="color: #0000ff;">order</span> <span style="color: #0000ff;">by</span> t.Id <span style="color: #0000ff;">asc</span></div><p>  (3)将上述代码总结为分页代码,设页大小为pageSize,请求页号为pageIndex:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span><span style="color: #000000;"> (<br/> </span><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span>,ROW_NUMBER() <span style="color: #0000ff;">OVER</span>(<span style="color: #0000ff;">Order</span> <span style="color: #0000ff;">by</span> Id) <span style="color: #0000ff;">as</span> rownum <span style="color: #0000ff;">from</span><span style="color: #000000;"> Account<br/>) </span><span style="color: #0000ff;">as</span><span style="color: #000000;"> t<br/></span><span style="color: #0000ff;">where</span> t.rownum <span style="color: #808080;">between</span> (<span style="color: #008000;">@pageIndex</span><span style="color: #808080;">-</span><span style="color: #800000; font-weight: bold;">1</span>)<span style="color: #808080;">*</span>pageSize<span style="color: #808080;">+</span><span style="color: #800000; font-weight: bold;">1</span> <span style="color: #808080;">and</span> <span style="color: #008000;">@pageSize</span><span style="color: #808080;">*</span><span style="color: #000000;">pageIndex<br/></span><span style="color: #0000ff;">order</span> <span style="color: #0000ff;">by</span> t.Id <span style="color: #0000ff;">asc</span></div><p><strong>四、各种连接&mdash;JOIN</strong></p><p><strong>4.1 Join==Inner Join</strong></p><p>  默认情况下,使用Join则代表Inner Join内连接,表示两个表根据某种等值规则进行连接。例如下面示例:查询所有学生的学号、姓名及所在班级</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> p.Id,p.Name,c.Name <span style="color: #0000ff;">from</span> T_Person p <span style="color: #808080;">join</span> T_Class c <span style="color: #0000ff;">on</span> p.ClassId<span style="color: #808080;">=</span>c.Id</div><p><strong>4.2 Left Join</strong></p><p>  例如:查询所有学生(参加及未参加考试的都算)及成绩,这里涉及到学生表及成绩表,题目要求参加及未参加考试的都要列出来,于是以学生表为基准,对成绩表进行左连接:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span><span style="color: #000000;"> Student s<br/></span><span style="color: #808080;">left</span> <span style="color: #808080;">join</span> SC sc <span style="color: #0000ff;">on</span> s.S#<span style="color: #808080;">=</span>sc.S#</div><p><img style="display: block; margin-left: auto; margin-right: auto;" src="http://images.cnitblog.com/blog/381412/201408/230140581283372.jpg" alt="" /></p><p>  可以通过运行结果图,看到王二小这个童鞋没有参加考试,也就没有成绩。</p><p><strong>4.3 Right Join</strong></p><p>  例如:要查询出所有没有参加考试(在成绩表中不存在的学生)的学生的姓名。于是还是可以以学生表为基准,但是要对成绩表进行右连接:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span><span style="color: #000000;"> Student s<br/></span><span style="color: #808080;">right</span> <span style="color: #808080;">join</span> SC sc <span style="color: #0000ff;">on</span> s.S#<span style="color: #808080;">=</span>sc.S#</div><p><strong>4.4 Cross Join</strong></p><p>  此种连接在实际应用中不算常见的,但却是理论基础,因为它代表了笛卡尔积。其实,所有连接方式都会先生成临时笛卡尔积表,笛卡尔积是关系代数里的一个概念,表示第一个表的行数乘以第二个表的行数等于笛卡尔积结果集的大小。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">select</span> <span style="color: #808080;">*</span> <span style="color: #0000ff;">from</span><span style="color: #000000;"> Student s<br/></span><span style="color: #0000ff;">cross</span> <span style="color: #808080;">join</span> SC sc</div><p>  这里假如Student表中有10行记录,SC表中有20行记录,那么两表进行交叉连接笛卡尔积运算会得到10*20=200行记录的结果集。</p><p><strong>五、视图</strong></p><p><strong>5.1 三个角度看视图</strong></p><p>  从用户角度来看,一个视图是从一个特定的角度来查看数据库中的数据。</p><p>  从数据库系统内部来看,一个视图是由SELECT语句组成的查询定义的<strong>虚拟表</strong>。</p><p>  从数据库系统内部来看,视图是由一张或多张表中的数据组成的;从数据库系统外部来看,视图就如同一张表一样,对表能够进行的一般操作都可以应用于视图,例如查询,插入,修改,删除操作等。</p><p><strong>5.2 创建视图</strong></p><p>  例如,我们可以创建一个学生成绩详细信息视图,对一个需要进行三表连接的查询进行封装:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">create</span> <span style="color: #0000ff;">view</span><span style="color: #000000;"> vw_sc<br/></span><span style="color: #0000ff;">as</span><br/><span style="color: #0000ff;">select</span> s.S#,s.Sname,c.Cname,sc.Score <span style="color: #0000ff;">from</span><span style="color: #000000;"> Student s<br/></span><span style="color: #808080;">join</span> SC sc <span style="color: #0000ff;">on</span> s.S#<span style="color: #808080;">=</span><span style="color: #000000;">sc.S#<br/></span><span style="color: #808080;">join</span> Course c <span style="color: #0000ff;">on</span> sc.C#<span style="color: #808080;">=</span>c.C#</div><p>  然后,我们对vw_sc进行select查询:</p><p><img style="display: block; margin-left: auto; margin-right: auto;" src="http://images.cnitblog.com/blog/381412/201408/230154025963900.jpg" alt="" /></p><p><strong>5.3 视图的注意事项</strong></p><p>  (1)视图在操作上和数据表没有什么区别,但两者的差异是其本质是不同:<strong>数据表是实际存储记录的地方,然而视图并不保存任何记录</strong>。</p><p>  (2)相同的数据表,<strong>根据不同用户的不同需求,可以创建不同的视图</strong>(不同的查询语句)。</p><div>  (3)视图的目的是方便查询,所以<strong>一般情况下不对视图进行增改,不能删</strong>。</div><p>&nbsp;</p><div id="Copyright"><p><span style="font-size: 14px;">作者:<a href="http://edisonchou.cnblogs.com">周旭龙</a></span></p><p><span style="font-size: 14px;">出处:<a href="http://edisonchou.cnblogs.com" target="_blank">http://edisonchou.cnblogs.com</a></span></p><p><span style="font-size: 14px;">本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。</span></p></div><img src="http://counter.cnblogs.com/blog/rss/3930414" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/edisonchou/p/3930414.html" target="_blank">走向面试之经典的数据库基础:二、SQL进阶之case、子查询、分页、join与视图</a>,转载请注明。</p>http://www.cnblogs.com/wxb0328/p/3930520.html关于ASCII、GB231、GBK、UTF-8/UTF8、ANSI、unicode的学习笔记 - 趴在巨人肩上的矮子继续上次的学习内容,写一些自己学习的笔记吧!总是觉得没有笔记的学习总是不那么踏实,我承认自己是个记忆力很差的人,特别羡慕那些可以把自己学过的东西记得很牢靠的人。哎!可惜我不是,那只能做出来点东西,就算以后忘了,回过头来可以看一看,有东西查;毕竟是自己亲手打出来的一个一个字啊 已经一万六千字了!...2014-08-22T17:07:00Z2014-08-22T17:07:00Z趴在巨人肩上的矮子http://www.cnblogs.com/wxb0328/<p>  继续上次的学习内容,写一些自己学习的笔记吧!总是觉得没有笔记的学习总是不那么踏实,我承认自己是个记忆力很差的人,特别羡慕那些可以把自己学过的东西记得很牢靠的人。哎!可惜我不是,那只能做出来点东西,就算以后忘了,回过头来可以看一看,有东西查;毕竟是自己亲手打出来的一个一个字啊<img src="http://images.cnitblog.com/blog/537027/201408/230039008622094.jpg" alt="" /></p><p>  已经一万六千字了!嘿嘿。。。继续加油!</p><p>  今天公司网络有问题,说是让在家里上班,skype在线就行了。结果我这自觉性不够,就打酱油了!还是继续学习我的mysql吧!我是刚毕业的大学生,学的和写的都是一些最基础的,如果巧遇大牛,还请轻喷!马上就要看完这部视频了,每次看到最后又着急又兴奋的,就像赶紧看完。这里我谈一点对于看视频自学和看书自学的感悟:</p><p>  看视频适合入门、基础比较薄的需要提升基础知识的人。看视频形象并且有很多的老师讲的例子,我们可以一边看视频一边练习。但是,看视频实在是很耗费时间。而且对于一个有网还是一个人坐在屋子里自学的人来说,自制力有要求啊!不过凡事慢慢看,一点一点积累,只要自己有计划,心里别着急,技术这东西还是记不得的。</p><p>  看书适合有了一定的基础,对于书上的大部分只是已经有一定的了解和掌握,作为提高和工作中遇到难题来弄懂这一块还是很好地,看书快啊。但是看过去很快就有许多就忘记了,因为书中的每一个知识点,不是我们天天都会用的,不用当让时间一长就忘记了 。所以,看书适合有一点基础,看的过程中不用每一页都是:一行一行的划重点,试想,一本书看完了,这么多重点我都不知道,那我划过去的重点我就都能记住吗?</p><p>  还记得四月份的时候给我对象处理毕业设计时遇到的乱码问题,我整整弄了三天算是解决了问题,但是自己更本没有弄懂其中的原因,所以借这次刚好好好学习一下,了了自己一个心愿。</p><p>  我看的是:燕十八老师的mysql从入门到精通 在此感谢十八哥!网络真是伟大,让我们可以免费学的很多东西。大家有想学习的可以去网上找,要是实在是找不到,就留言,我百度网盘分享,1T的资源,估计这辈子都看不完!好了计入正题</p><p>  </p><p><strong>  <span style="font-size: 14pt;">ascii</span></strong></p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 最早的编码集,0-127个,一个字节存储</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-size: 14pt;">&nbsp;<strong>GB2312</strong></span></p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 到了中国,常用汉字3000个,生僻字不用说了!</p><p>    一个字节不够用了?</p><p>    思考:用两个字节来表示</p><p>      [][]</p><p>      0000 0000 0000 0000</p><p>      1111 1111 1111 1111</p><p>      0-&gt;65535,六万多种组合,够用了</p><p>    GB2312字符集</p><p>      [202 197] 假如代表 中</p><p>      [69 197] 代表什么?69 和197是整个理解,还是单个理解?此时就产生了歧义:就是因为单字节的小于127的值,正好是ascii的值了</p><p>      如果就严格的2字节绑定,理解成中文</p><p>      侧GB2312就不能兼容小于127的英文字符了</p><p>    问:如何兼容ascii,又能两个字节表示中文?</p><p>      ascii 0-127</p><p>      0xxx xxxx</p><p>      干脆 ,gb2312完全不占用0-127</p><p>      gb2312占用的两个字节的来自[129-255][129-255]</p><p>    但是这样,中文的组合数也就减少了,大约有一万多</p><p> <span style="font-size: 14pt;"> GBK</span></p><p>    完全兼容GB2312,只是比GB2312的容量大了,也就是说可以编码更多的字符</p><p>    GBK和GB2312一样都是两个字节,那么如何扩充容量?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 答:GBK的第二位,低位,不在局限于129-255了,&lt;127的也能用了</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 例如 148 35 65 179 82</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 问:几个中文,几个英文?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 中文 英文 中文</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 我们可以看出,GBK是当遇到的这个一个字节大于127时,就知道这是中文,后面还有一个字节,当小于127时,就表明这是英文编码,后面没有跟带的字节了。也就是遇到大于127的根据规则,后面还有一个字节,需要在读取;小于时就知道,这是英文,后面没有了</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 容量:收录汉字21003个、符号883个、并提供1894个造字位</p><p><span style="font-size: 14pt;">ANSI、Unicode、utf-8</span></p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 中国&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; GBK 两个字节</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; [137][134] &ndash;&gt;可能对应-&gt; 中</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 那么到了日本呢?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; [137][134] 对应什么?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 日本使用的是 jis 字符集</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <span style="font-size: 18px;">ANSI&nbsp;</span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 代表本地字符集</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 这个更你本机的语言设置有关,比如中文的ANSI就是GBK,日文的ANSI就是jis</p><p>解决了多字节以后,又有一个问题:如何解决各国的字符集兼容问题?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 答案:<span style="font-size: 18px;">Unicode(</span>统一码、万国码)&nbsp;&nbsp; 终极大招</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 解决国际化问题,unicode是什么?就是一个全世界通用的码表,全世界范围的字符,统一分配一个标号,这样就不会乱了</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 容量:2^32 40多亿,天文数字,足够用了</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 但是我们常用的,集中在前65535个标号里</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 因此,两个字节就够了,</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 请注意:unicode只是负责分配编号,而且都用4个字节来编号,而且和ascii不兼容</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; unicode只负责分配编号,那么就有其他在不改变unicode的基础上,简化字节</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 0000 0000 0000 0000 0000 0000 0000 0041 &ndash;&gt; A</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 简化</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 0000 0041 -&gt; A</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 把高位浪费的0值,用一定的规则舍弃,这种是形成具体的编码格式(猜想:简化后的编码,可以更加节省带宽,网络传输更快,原因:原来都需要4个字节,现在简化以后可能只需要一个字节)</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 这样简化以后的编码才能在网络上传输,unicode只是负责分配编号,不进行实际的编码工作,一定需要注意区别!!!!也就是说unicode是字符集,utf-8是字符编码,这里我找了一票博客,推荐给大家</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <em>关于字符集和编码的一些基础知识:</em></p><p><em>&nbsp;&nbsp;&nbsp;     1.</em><em>基础知识</em></p><p><em>      算机中储存的信息都是用二进制数表示的;而我们在屏幕上看到的英文、汉字等字符是二进制数转换之后的结果。通俗的说,按照何种规则将字符存储在计算机中,如'a'用什么表示,称为"编码";反之,将存储在计算机中的二进制数解析显示出来,称为"解码",如同密码学中的加密和解密。在解码过程中,如果使用了错误的解码规则,则导致'a'解析成'b'或者乱码。</em></p><p><em>      字符集(Charset):是一个系统支持的所有抽象字符的集合。字符是各种文字和符号的总称,包括各国家文字、标点符号、图形符号、数字等。</em></p><p><em>      字符编码(Character Encoding):是一套法则,使用该法则能够对自然语言的字符的一个集合(如字母表或音节表),与其他东西的一个集合(如号码或电脉冲)进行配对。即在符号集合与数字系统之间建立对应关系,它是信息处理的一项基本技术。通常人们用符号集合(一般情况下就是文字)来表达信息。而以计算机为基础的信息处理系统则是利用元件(硬件)不同状态的组合来存储和处理信息的。元件不同状态的组合能代表数字系统的数字,因此字符编码就是将符号转换为计算机可以接受的数字系统的数,称为数字代码。</em></p><p><em>    2.</em><em>常用字符集和字符编码</em></p><p><em>  常见字符集名称:ASCII</em><em>字符集、GB2312</em><em>字符集、BIG5</em><em>字符集、GB18030</em><em>字符集、Unicode</em><em>字符集等。计算机要准确的处理各种字符集文字,需要进行字符编码,以便计算机能够识别和存储各种文字。</em></p><p><em>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </em><em>详细参考博客:<a title="字符集和字符编码(Charset &amp; Encoding)" href="http://www.cnblogs.com/skynet/archive/2011/05/03/2035105.html" target="_blank">http://www.cnblogs.com/skynet/archive/2011/05/03/2035105.html</a></em></p><p>继续上面探讨</p><p>  <span style="font-size: 14pt;">UTF8/UTF-8</span></p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 那么怎么简化传输?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 这时就出现了utf-8(unicode transformation format)这就是其中最出名的一个</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 这种简化,就像原文件和压缩文件的关系</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 问:给定unicode的字符&mdash;&gt;utf-8的二进制值?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 知道unicode可以推出utf-8,同样utf-8也能推出unicode</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 问:utf8占几个字节?</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 首先不可能是定长,否则压缩有什么意义!</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 我们看看对应表</p><p>      <img src="http://images.cnitblog.com/blog/537027/201408/230057318152364.jpg" alt="" /></p><p><strong>    utf8</strong><strong>是一个变长:1-6</strong><strong>个字节</strong></p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 变长!那么如何确定字符的边界,即计算机去几个字节,才能知道这个字符正确的取出来了</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 我们仔细观察上表:</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 当最高位字节有一个0时(0xxxxxxx)取一个字节,110(110xxxxx 10xxxxxx)取两个字节,1110取三个字节,11110取四个字节,111110取五个字节,1111110x取六个字节</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 中文utf8占三个字节</p><p><strong>   推论</strong><strong>: </strong><strong>什么时将会乱码</strong><strong>?</strong></p><p>      1: client声明与事实不符</p><p>      2:results与客户端页面不符的时候.</p><p>&nbsp;</p><p><strong>   什么时间将会丢失数据?</strong></p><p>    Connetion和服务器的字符集比client小时.<br />    <img src="http://images.cnitblog.com/blog/537027/201408/230104172683165.jpg" alt="" /></p><p><strong>   校对集</strong><strong>: </strong><strong>指字符集的排序规则</strong></p><p>      一种字符集可以有一个或多个排序规则.</p><p>      以Utf8为例, 我们默认使的utf8_general_ci 规则,也可以按二进制来排, utf8_bin</p><p>&nbsp;    怎么样声明校对集?</p><p>      Create table ()... Charset utf8 collateutf8_general_ci;</p><p><strong>     <span style="color: #ff0000;"> 注意:</span></strong><span style="color: #ff0000;"><strong>声明的校对集必须是字符集合法的校对集.</strong></span></p><p>  好了,就先这么多吧,待会再发一篇关于mysql字符集的学习。太长了看着人太累,写短一点,一篇记录一个问题</p><p>&nbsp;</p><img src="http://counter.cnblogs.com/blog/rss/3930520" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/wxb0328/p/3930520.html" target="_blank">关于ASCII、GB231、GBK、UTF-8/UTF8、ANSI、unicode的学习笔记</a>,转载请注明。</p>http://www.cnblogs.com/Hackerman/p/3930505.html20个linux命令行工具监视性能 - Hackerman对于每一个系统管理员或网络管理员每天监视或调试linux系统的性能问题是一件非常困难的事,在it行业作为一个linux管理员五年之后,我开始知道监视和保持系统启动和运行有多么的困难。由于这个原因,我们编写了20条常用的命令行工具,希望对每一个linux系统管理员有用。这些命令是可利用的在所有类型的l...2014-08-22T16:30:00Z2014-08-22T16:30:00ZHackermanhttp://www.cnblogs.com/Hackerman/<p>对于每一个系统管理员或网络管理员每天监视或调试linux系统的性能问题是一件非常困难的事,在it行业作为一个linux管理员五年之后,我开始知道监视和保持系统启动和运行有多么的困难。由于这个原因,我们编写了20条常用的命令行工具,希望对每一个linux系统管理员有用。这些命令是可利用的在所有类型的linux之下并且对监视和发现性能的实际原因是很有用的。这些列表的命令是足够的对于你选择适合你的任何一个监控场景。</p><p><img src="" alt="" /></p><p>1,top&nbsp; ---linux process monitoring</p><p>&nbsp;&nbsp; top 命令是一个性能监视程序,通常被许多系统管理员频繁使用监视linux性能并且能够使用在不同的类型的linux之下。top命令用于显示所有运行和活跃的实时进程并定期更新它。它显示了cpu使用率,内存使用率,交换内存,缓存大小,缓冲区大小,进程pid,用户,命令等等。它也展示了运行进程的高cpu利用率和内存利用率。top命令对系统管理员监视和在需要的时候采取正确的行动是非常的重要。下面让我们看看top命令。</p><h3 style="background: none repeat scroll 0px 0px #555; border-radius: 4px; box-shadow: 0px 0px 15px rgba(0, 0, 0, 0.25) inset; clear: both; color: #fff; font: 1.1em/1.2;">#top</strong></p><p><img src="" alt="" />&nbsp;</p><p>&nbsp;了解top的更多带参数的命令在shell中输入&ldquo;man top&rdquo;。</p><p>2,VmStat -- Virtual Memory Statistic</p><p>&nbsp;&nbsp; VmStat命令用来显示<strong>虚拟内存</strong>, <strong>内核线程</strong>, <strong>磁盘</strong>, <strong>系统进程</strong>, <strong>输入输出块</strong>, <strong>中断</strong>, <strong>活动的cpu</strong> 等等.默认的VmStat命令是不能直接被直接使用的,在linux系统下你必须安装一个名叫<strong>sysstat</strong>的包,这个包包含Vmstat命令。这个命令的使用如下。</p>procs -----------memory---------- ---swap-- -----io---- --system-- -----cpu-----<br/> r b swpd free inact active si so bi bo in cs us sy id wa st<br/> 1 0 0 810420 97380 70628 0 0 115 4 89 79 1 6 90 3 0<br/><br/><p>&nbsp;&nbsp;&nbsp; 了解VmStat的更多命令请在shell中输入&ldquo;man vmstat&rdquo;。</p><p>3, Lsof &ndash; List Open Files</p><p>&nbsp;&nbsp;&nbsp; Lsof命令用来显示所有打开的文件和进程的一个列表。打开的文件包括<strong>磁盘文件</strong>, <strong>网络套接字</strong>, <strong>管道</strong>, <strong>设备</strong> 和 <strong>进程</strong>。使用这个命令的主要原因之一是当文件系统不能被卸载或者文件出现一个错误是文件正在被使用或打开。使用这个命令你能够很容易的找到那个文件正在被使用。这个命令的使用格式如下。</p><strong style="font-weight: bold;"><br/>COMMAND PID USER FD TYPE DEVICE SIZE NODE NAME<br/>init 1 root cwd DIR 104,2 4096 2 /<br/>init 1 root rtd DIR 104,2 4096 2 /<br/>init 1 root txt REG 104,2 38652 17710339 /sbin/init<br/>init 1 root mem REG 104,2 129900 196453 /lib/ld-2.5.so<br/>init 1 root mem REG 104,2 1693812 196454 /lib/libc-2.5.so<br/>init 1 root mem REG 104,2 20668 196479 /lib/libdl-2.5.so<br/>init 1 root mem REG 104,2 245376 196419 /lib/libsepol.so.1<br/>init 1 root mem REG 104,2 93508 196431 /lib/libselinux.so.1<br/>init 1 root 10u FIFO 0,17 953 /dev/initctl<br/><br />了解更多的Lsof命令请直接打开shell输入&ldquo;man lsof&rdquo;<br /></strong><br/><p>4,Tcpdump -- Network Packet Analyzer</p><p><strong>&nbsp; Tcpdump</strong> 是一个被广泛使用的命令行工具,<strong>network packet analyzer(网络包分析)</strong> 或<strong>packets sniffer(包嗅探)</strong> 命令 被用来 capture(捕获) or filter(过滤) 接受或传输在连接网络上的接口上的&nbsp; <strong>TCP/IP</strong> 包,它也提供一个选项来保存捕获的包在一个文件中之后进行分析。&nbsp; tcpdump 也是有用的在分布的不同类型的linux中。tcpdump的使用如下。</p><h3 style="background: none repeat scroll 0px 0px #555; border-radius: 4px; box-shadow: 0px 0px 15px rgba(0, 0, 0, 0.25) inset; clear: both; color: #fff; font: 1.1em/1.2;"># tcpdump -i eth0</strong></p><strong><br/>tcpdump: verbose output suppressed, use -v or -vv for full protocol decode<br/>listening on eth0, link-type EN10MB (Ethernet), capture size 96 bytes<br/>22:08:59.617628 IP tecmint.com.ssh &gt; 115.113.134.3.static-mumbai.vsnl.net.in.28472: P 2532133365:2532133481(116) ack 3561562349 win 9648<br/>22:09:07.653466 IP tecmint.com.ssh &gt; 115.113.134.3.static-mumbai.vsnl.net.in.28472: P 116:232(116) ack 1 win 9648<br/>22:08:59.617916 IP 115.113.134.3.static-mumbai.vsnl.net.in.28472 &gt; tecmint.com.ssh: . ack 116 win 64347<br/></strong><br/><p>5,Netstat -- Network Statistics</p><p>&nbsp;&nbsp; <strong>Netstat</strong> 是一个监视输入和输出网络包统计的命令行工具,它是非常的有用对于每一个系统管理员监视网络性能和 网络连接故障问题。使用方法如下。</p><p>&nbsp;&nbsp;</p><h3 style="background: none repeat scroll 0px 0px #555; border-radius: 4px; box-shadow: 0px 0px 15px rgba(0, 0, 0, 0.25) inset; clear: both; color: #fff; font: 1.1em/1.2;"># netstat -a | more</strong></p><strong><br/>Active Internet connections (servers and established)<br/>Proto Recv-Q Send-Q Local Address Foreign Address State<br/>tcp 0 0 *:mysql *:* LISTEN<br/>tcp 0 0 *:sunrpc *:* LISTEN<br/>tcp 0 0 *:realm-rusd *:* LISTEN<br/>tcp 0 0 *:ftp *:* LISTEN<br/>tcp 0 0 localhost.localdomain:ipp *:* LISTEN<br/>tcp 0 0 localhost.localdomain:smtp *:* LISTEN<br/>tcp 0 0 localhost.localdomain:smtp localhost.localdomain:42709 TIME_WAIT<br/>tcp 0 0 localhost.localdomain:smtp localhost.localdomain:42710 TIME_WAIT<br/>tcp 0 0 *:http *:* LISTEN<br/>tcp 0 0 *:ssh *:* LISTEN<br/>tcp 0 0 *:https *:* LISTEN<br/></strong><br/><p>&nbsp; netstat 的更多使用方法见&ldquo;man netstat&rdquo;。</p><p>6,Htop &ndash; Linux Process Monitoring</p><p>&nbsp;&nbsp; <strong>Htop</strong> 是一个更先进的交互和实时进程监控工具。这和linux的top命令更相似但是Htop有更丰富的界面就像用户友好的界面管理进程,快捷键,水平和垂直的视图浏览进程等等。Htop是一个三部分的工具默认情况下没有包含。你必须安装它使用YUM包管理工具。</p><p>&nbsp;</p><h3 style="background: none repeat scroll 0px 0px #555; border-radius: 4px; box-shadow: 0px 0px 15px rgba(0, 0, 0, 0.25) inset; clear: both; color: #fff; font: 1.1em/1.2;"># htop</strong></p><p><img style="display: inline;" title="Htop Command Example" src="http://www.tecmint.com/wp-content/uploads/2012/08/htop-3.jpg" alt="Htop Command Example" width="600" height="363" data-lazy-loaded="true" /></p><p>&nbsp;<a title="安装htop" href="http://www.tecmint.com/install-htop-linux-process-monitoring-for-rhel-centos-fedora/" target="_blank">安装htop。</a></p><p>&nbsp;7,Iotop -- Monitor Linux Disk I/O</p><p><strong>&nbsp;&nbsp; Iotop</strong> 也和top,htop命令相似, 但是它能监视和显示实时的磁盘I/O信息和进程信息。这个工具更有用对于查找而外的进程和高使用磁盘读写进程。</p><p>&nbsp;</p><p><strong># iotop</strong></p><p><img style="display: inline;" title="Iotop Command Example" src="http://www.tecmint.com/wp-content/uploads/2012/07/iotop-Screen.jpg" alt="Iotop Command Example" width="630" height="420" data-lazy-loaded="true" /></p><p><a title="安装iotop" href="http://www.tecmint.com/install-iotop-monitor-linux-disk-io-in-rhel-centos-and-fedora/" target="_blank">安装iotop。</a></p><p>8,Iostat &ndash; Input/Output Statistics</p><p>&nbsp;&nbsp; <strong>IoStat</strong> 是一个简单的收集和显示输入输出存储设备的统计信息的工具。这个工具被用来追This to踪存储设备性能问题包括 <strong>devices(设备)</strong>, <strong>local disks(本地磁盘)</strong>, <strong>remote disks(远程磁盘)</strong> such as <strong>NFS(网络存储文件)</strong>。</p><p>&nbsp;</p><h3 style="background: none repeat scroll 0px 0px #555; border-radius: 4px; box-shadow: 0px 0px 15px rgba(0, 0, 0, 0.25) inset; clear: both; color: #fff; font: 1.1em/1.2;"># iostat</strong></p><strong><br/>Linux 2.6.18-238.9.1.el5 09/13/2012<br/></strong><br/>avg-cpu: %user %nice %system %iowait %steal %idle<br/> 2.60 3.65 1.04 4.29 0.00 88.42<br/><br/>Device: tps Blk_read/s Blk_wrtn/s Blk_read Blk_wrtn<br/>cciss/c0d0 17.79 545.80 256.52 855159769 401914750<br/>cciss/c0d0p1 0.00 0.00 0.00 5459 3518<br/>cciss/c0d0p2 16.45 533.97 245.18 836631746 384153384<br/>cciss/c0d0p3 0.63 5.58 3.97 8737650 6215544<br/>cciss/c0d0p4 0.00 0.00 0.00 8 0<br/>cciss/c0d0p5 0.63 3.79 5.03 5936778 7882528<br/>cciss/c0d0p6 0.08 2.46 2.34 3847771 3659776<br/><p><br />9. IPTraf &ndash; Real Time IP LAN Monitoring</p><p>&nbsp;&nbsp; <strong>IPTraf&nbsp;</strong>&nbsp; 是一个开源的基于console的实时网络监视工具,它收集了不同种类的信息例如通过网络的ip交通监视。包括tcp flag信息,icmp信息,tcp/udp交通故障,tcp连接包和字节统计。它还收集接口的统计TCP、UDP、IP、ICMP、非IP,IP校验和错误,界面活动的信息等。</p><p><img class="size-full wp-image-2167" style="display: inline;" src="http://www.tecmint.com/wp-content/uploads/2013/02/IPTraf2.png" alt="IP Traffic Monitor" width="600" height="377" data-lazy-loaded="true" /></p><p>&nbsp;</p><p>10,Psacct or Acct -- Monitor User Activity</p><p>&nbsp;&nbsp;&nbsp; psacct or acct tools 是非常的有用对于监视每一个用户的活动, 这两个守护进程在后台运行并保持密切关注系统上的每个用户的整体活动有什么资源被消耗。</p><p>&nbsp;&nbsp;&nbsp; 这些工具是非常有用的对系统管理员追踪每个用户的活动例如用户正在做什么,他们发布了什么命令,他们在使用多少资源,他们使用系统多久了等等。</p><p>&nbsp;&nbsp;&nbsp; <a title="怎么使用psacct" href="http://www.tecmint.com/how-to-monitor-user-activity-with-psacct-or-acct-tools/" target="_blank">How to Monitor User Activity with psacct or acct Tools </a></p><p>今天就翻译到这里,后面的10个下次在翻译。初次翻译博客希望大家多多指教。欢迎大家指点。</p><p>最后注明原文出处:<a href="http://www.tecmint.com/command-line-tools-to-monitor-linux-performance/" target="_blank">20 Command Line Tools to Monitor Linux Performance</a></p><img src="http://counter.cnblogs.com/blog/rss/3930505" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/Hackerman/p/3930505.html" target="_blank">20个linux命令行工具监视性能</a>,转载请注明。</p>http://www.cnblogs.com/zxoling/p/3930497.htmlCocos_Code_Ide学习(一):理解Cocos Lua Project下的frameworks的proj.win32 - 侏小儒第一次写,不知道有没有用,有不对的地方,接受大家的批评。勿喷,谢谢。1.首先,创建工程-----------------------------------------------------------------------------------------------------------...2014-08-22T16:28:00Z2014-08-22T16:28:00Z侏小儒http://www.cnblogs.com/zxoling/<p><span style="color: #ff0000;"><strong>第一次写,不知道有没有用,有不对的地方,接受大家的批评。勿喷,谢谢。</strong></span></p><p>&nbsp;</p><p><span style="color: #ff0000;"><strong>1.首先,创建工程</strong></span></p><p><span style="color: #ff0000;"><strong>---------------------------------------------------------------------------</strong></span></p><p>&nbsp;</p><p><span style="color: #ff0000;"><strong><img src="http://images.cnitblog.com/blog/528088/201408/222324573938289.png" alt="" /></strong></span></p><p><span style="color: #ff0000;"><strong>---------------------------------------------------------------------------</strong></span></p><p><span style="color: #ff0000;"><strong><img src="http://images.cnitblog.com/blog/528088/201408/222326188789849.png" alt="" /></strong></span></p><p><span style="color: #ff0000;"><strong>---------------------------------------------------------------------------</strong></span></p><p><span style="color: #ff0000;"><strong>2. 用 VS 打开 proj.win32 下的 .sln 文件,编译一遍</strong></span></p><p><span style="color: #ff0000;"><strong>成功后,会出现</strong></span></p><p><span style="color: #ff0000;"><strong><img src="http://images.cnitblog.com/blog/528088/201408/222340190817003.png" alt="" /></strong></span></p><p><span style="color: #ff0000;"><strong><strong><strong>上面场景的代码,是 main.lua 实现的。</strong></strong></strong></span></p><p><span style="color: #ff0000;"><strong><strong>---------------------------------------------------------------------------</strong></strong></span></p><p>&nbsp;</p><p><span style="color: #ff0000;"><strong>3.</strong></span><span style="color: #ff0000;"><strong>接下来,要实现 Cocos Code Ide 与 VS 互通</strong></span></p><p><span style="color: #ff0000;"><strong> 首先设置 Build Win32 Runtime, 在Cocos Code Ide里面,右键项目名称---选择 Cocos Tools ---选择 Build Custom Runtimes</strong></span></p><p><span style="color: #ff0000;"><strong> 勾选 &ldquo;<strong>Build Win32 Runtime&rdquo;,然后等待完成。</strong></strong></span></p><p><span style="color: #ff0000;"><strong><strong><img src="http://images.cnitblog.com/blog/528088/201408/222355436905677.png" alt="" /></strong></strong></span></p><p>&nbsp;</p><p><img src="http://images.cnitblog.com/blog/528088/201408/222356007069374.png" alt="" /></p><p><span style="color: #ff0000;">&nbsp; &nbsp;<strong>完成后,在你项目目录下的 runtime -- win32 里面,会出现 (你的项目名称).exe</strong></span></p><p><span style="color: #ff0000;"><strong><img src="http://images.cnitblog.com/blog/528088/201408/222357536903059.png" alt="" /></strong></span></p><p>&nbsp;</p><p> <strong><span style="color: #ff0000;">然后点击 Debug Configurations</span></strong></p><p><strong><span style="color: #ff0000;"><img src="http://images.cnitblog.com/blog/528088/201408/222359453463771.png" alt="" /></span></strong></p><p><img src="http://images.cnitblog.com/blog/528088/201408/230002526901539.png" alt="" /></p><p><strong><span style="color: #ff0000;">  然后 Apply -- Debug,会出现刚才的农场画面</span></strong></p><p><strong><span style="color: #ff0000;"><strong><strong>---------------------------------------------------------------------------</strong></strong></span></strong></p><p>&nbsp;</p><p><span style="color: #ff0000;"><strong>4.但是,我想使用自己的场景。</strong></span></p><p><span style="color: #ff0000;"><strong> 在 VS 中,添加再熟悉不过的HelloWorld类(自己实现吧)</strong></span></p><p><span style="color: #ff0000;"><strong> 我将 main.lua 里面文件的代码删减一些,main 函数里面只剩下</strong></span></p><p><span style="color: #ff0000;"><strong><img src="http://images.cnitblog.com/blog/528088/201408/230006397687879.png" alt="" /></strong></span></p><p><span style="color: #ff0000;"><strong> 此时,在 VS 的&nbsp;AppDelegate::applicationDidFinishLaunching() 中,添加一句</strong></span></p><p><span style="color: #ff0000;"><strong>&nbsp; CCLOG("test");</strong></span></p><p><span style="color: #ff0000;"><strong> 然后保存,在 Cocos Code Ide 中,右键项目 --- Easy Shell --- Open... &nbsp;输入 cocos compile -p win32,运行</strong></span></p><p><span style="color: #ff0000;"><strong> <img src="http://images.cnitblog.com/blog/528088/201408/230014576741593.png" alt="" /></strong></span></p><p><span style="color: #ff0000;"><strong> 会出现刚才添加那句的测试。</strong></span></p><p><span style="color: #ff0000;"><strong>&nbsp; &nbsp;注:<strong>cocos compile -p win32 命令 是在你频繁的修改 C++ 代码后,不再需要在 Cocos Code Ide中,右键设置 Build Win32 Runtime</strong></strong></span></p><p><span style="color: #ff0000;"><strong><strong> 记住在修改 C++代码后,一定要保存。</strong></strong></span></p><p><span style="color: #ff0000;"><strong><strong><strong>---------------------------------------------------------------------------</strong></strong></strong></span></p><p><span style="color: #ff0000;"><strong> 5.接下来替换自己的场景,一般在AppDelegate的applicationDidFinishLaunching()里面创建并runwithscene(),</strong></span><span style="color: #ff0000;"><strong>试了试,发现不好使,后来发现,场景的入口,在 Runtime.cpp 文件,里面有个startRuntime() 函数</strong></span></p><p><span style="color: #ff0000;"><strong> <img src="http://images.cnitblog.com/blog/528088/201408/230021165492575.png" alt="" /></strong></span></p><p><span style="color: #ff0000;"><strong> 将里面的场景,替换成 HelloWorld 的场景,将下面的 2行 注释,ConnectWaitLayer 是上面实现的一个 Layer,就是刚开始看到的带有IP地址的一个层,</strong></span></p><p><span style="color: #ff0000;"><strong> 然后保存,输入 cocos compile -p win32,运行,会出现你想要的场景。</strong></span></p><p><span style="color: #ff0000;"><strong> 若你在HelloWorld.cpp 的 init() 函数里面,只添加一个背景层,设置位置,运行后,会出现先正常,然后放大的现象,只要把 main.lua 里面的</strong></span></p><p><span style="color: #ff0000;"><strong>&nbsp; &nbsp;cc.Director:getInstance():getOpenGLView():setDesignResolutionSize(480, 320, 0) 注释掉,就好了。</strong></span></p><p>&nbsp;</p><img src="http://counter.cnblogs.com/blog/rss/3930497" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/zxoling/p/3930497.html" target="_blank">Cocos_Code_Ide学习(一):理解Cocos Lua Project下的frameworks的proj.win32</a>,转载请注明。</p>http://www.cnblogs.com/carlos-vic/p/YueCi_2.html和我一起开发Android应用(二)——“悦词-i背单词”项目功能分析 - Carlos.V好几天没有更新了,给关心这个系列的朋友们说声抱歉。今天我们开始第二节,项目功能分析。因为这个背单词软件虽说功能比较简单,但要真正实现起来也挺麻烦的。所以今天我们首先分析一下这个应用的功能,然后逐条慢慢实现。 PS:这款应用已经上线91助手,百度移动应用和应用宝,有兴趣下来研究的可以百度搜索“悦...2014-08-22T15:40:00Z2014-08-22T15:40:00ZCarlos.Vhttp://www.cnblogs.com/carlos-vic/<p>  好几天没有更新了,给关心这个系列的朋友们说声抱歉。今天我们开始第二节,项目功能分析。因为这个背单词软件虽说功能比较简单,但要真正实现起来也挺麻烦的。所以今天我们首先分析一下这个应用的功能,然后逐条慢慢实现。</p><p>&nbsp; &nbsp; &nbsp;PS:这款应用已经上线91助手,百度移动应用和应用宝,有兴趣下来研究的可以百度搜索&ldquo;悦词i背单词91&rdquo;就可找到,我想真正用一下这个应用再看这个教程会有比较直观的理解。好废话不多讲,进入正题。</p><p>  功能分析:</p><p>功能1、查单词。</p><p>  实现方法:金山词霸开放平台提供了一个开放API,通过Http访问金山词霸API提供的URL,可以获得一个XML文档,这个文档就包含了要查询的单词的释义、例句、音标、声音的地址。通过解析该XML就可以获得单词的所有信息。</p><p>  所用到的技术:</p><p>    1)Http访问网络,并下载网络文件</p><p>    2)对SD卡进行操作,从SD卡(这里的SD卡是指手机默认的存储卡,因为有些手机既  能能插SD卡又有内部存储,这里不涉及这个问题)中读取文件,和把从网络下载的文件存进  SD卡。</p><p>    3)解析XML文件</p><p>    4)播放音乐(这个我后来封装成了一个类,专门从网络上查询某个单词,解析XML文   件,并且将下载的Mp3文件存在SD卡中,然后播放该Mp3文件)</p><p>    5)数据库,这里涉及到第一个数据库,每查找一个单词之后,就会将该单词和释义存储  到一个SQLite数据库中。这样一来,下一次查找这个单词时,先访问数据库,看看数据库中  有没有这个单词,若有,就不用访问网络了。</p><p>&nbsp;</p><p>功能2、背单词。</p><p>  实现方法:这里要用到第二个数据库,背单词的词库。我们需要一个存放单词的TXT文件,通过解析这个TXT文件,将要背的单词解析并存进数据库中,然后根据一定的规律弹出单词。</p><p>  所用到的技术:</p><p>    1)数据库,同前面的数据库技术相似;</p><p>    2)对TXT文件中的单词进行解析,字符串解析函数;</p><p>    3)单词状态机,设计一定的算法,按照一定的规律弹出单词,并进行背词操作。(这个确实挺麻烦)</p><p>    4)文件浏览,做一个简易的文件浏览器,用于浏览SD卡中的单词源文件txt,然后导入词库。这个属于比较单独的一个功能。</p><p>功能3、设置界面</p><p>  用于对背词软件的一些参数进行设置,比如播放英音还是美音?前后两个的单词的背词间隔是多少?导入词库设置?计划完成日期,以及当前课程的名称设置。</p><p>功能4、金山词霸每日一词</p><p>  实现方法:这里要用到第三个数据库(就是数据库的一个表table),用于记录使用者的信息,如当前日期,当日期有更新时,应用会自动访问网络,获取最新的额每日一句,并且呈现在主界面上;当前背单词的完成进度,今天已经背的单词,待完成的单词,以及根据计划完成日期算出今天应该的单词的任务,(关于这一点大家可以参考拓词的实现效果)</p><p>  所用到的技术:</p><p>    1)数据库技术;</p><p>    2)Http访问网络,从网络下载图片并显示。</p><p>&nbsp; &nbsp; &nbsp;注意这一个功能虽然看似简单,但是,实现这个功能之后这个应用才显得活起来^^</p><p>&nbsp;</p><p>好了以上就是这款应用的主要框架。今天我们就来开始实现第一个功能:查单词功能。</p><p>  首先讲一下金山词霸API:浏览器输入http://open.iciba.com/就会出现啊如下界面:</p><p><img src="http://images.cnitblog.com/blog/552704/201408/222232417213158.png" alt="" width="656" height="413" /></p><p>点击词霸查词、并选择文档选项,http://open.iciba.com/?c=wiki,就可以看到如下界面:</p><p><img src="http://images.cnitblog.com/blog/552704/201408/222234265035234.png" alt="" width="653" height="434" /></p><p>再点击查词接口,就到了final page:</p><p><img src="http://images.cnitblog.com/blog/552704/201408/222235431438149.png" alt="" width="750" height="527" /></p><p>大家会注意到我们需要的东西:<a href="http://dict-co.iciba.com/api/dictionary.php?w=go" target="_blank">http://dict-co.iciba.com/api/dictionary.php?w=go&amp;key=********</a>&nbsp; 这里的key是你自己申请的金山词霸开放平台的API key,申请界面在这里:</p><p><img src="http://images.cnitblog.com/blog/552704/201408/222238379712994.png" alt="" /></p><p>网址随便填即可</p><p>当你申请到金山API key之后,就可在浏览器输出上面的地址:<a href="http://dict-co.iciba.com/api/dictionary.php?w=go" target="_blank">http://dict-co.iciba.com/api/dictionary.php?w=go&amp;key=</a>这里换成你的API key.&nbsp;</p><p>   这里多说一句,为什么选金山词霸API呢,其实有道词典也有开放API,但它提供的数据远不如金山词霸,最重要的一点:金山提供的单词的发音(金山真是够仗义的)</p><p>例如我们查一个hello,就可以在浏览器输入<a href="http://dict-co.iciba.com/api/dictionary.php?w=go" target="_blank">http://dict-co.iciba.com/api/dictionary.php?w=hello&amp;key</a><span style="text-decoration: underline;">=这里换成你的API key</span>. &nbsp; &nbsp;就是把w=后面写成hello,注意首字母必须小写!</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">dict </span><span style="color: #ff0000;">num</span><span style="color: #0000ff;">="219"</span><span style="color: #ff0000;"> id</span><span style="color: #0000ff;">="219"</span><span style="color: #ff0000;"> name</span><span style="color: #0000ff;">="219"</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">key</span><span style="color: #0000ff;">&gt;</span>hello<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">key</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">ps</span><span style="color: #0000ff;">&gt;</span>h?'l??<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">ps</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">pron</span><span style="color: #0000ff;">&gt;</span><span style="color: #000000;"><br/>http://res-tts.iciba.com/5/d/4/5d41402abc4b2a76b9719d911017c592.mp3<br/></span><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">pron</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">ps</span><span style="color: #0000ff;">&gt;</span>h??lo, h?-<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">ps</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">pron</span><span style="color: #0000ff;">&gt;</span><span style="color: #000000;"><br/>http://res.iciba.com/resource/amp3/1/0/5d/41/5d41402abc4b2a76b9719d911017c592.mp3<br/></span><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">pron</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">pos</span><span style="color: #0000ff;">&gt;</span>int.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">pos</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">acceptation</span><span style="color: #0000ff;">&gt;</span>哈喽,喂;你好,您好;表示问候;打招呼;<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">acceptation</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">pos</span><span style="color: #0000ff;">&gt;</span>n.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">pos</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">acceptation</span><span style="color: #0000ff;">&gt;</span>&ldquo;喂&rdquo;的招呼声或问候声;<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">acceptation</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">pos</span><span style="color: #0000ff;">&gt;</span>vi.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">pos</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">acceptation</span><span style="color: #0000ff;">&gt;</span>喊&ldquo;喂&rdquo;;<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">acceptation</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><span style="color: #000000;"><br/>This document contains Hello application components of each document summary of the contents.<br/></span><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>此文件包含组成Hello应用程序的每个文件的内容摘要.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><span style="color: #000000;"><br/>In the following example, CL produces a combined source and machine - code listing called HELLO. COD.<br/></span><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>在下面的例子中, CL将产生一个命名为HELLO. COD的源代码与机器代码组合的清单文件.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span>Hello! Hello! Hello! Hello! Hel - lo!<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>你好! 你好! 你好! 你好! 你好!<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span>Hello! Hello! Hello! Hello ! I'm glad to meet you.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>你好! 你好! 你好! 你好! 见到你很高兴.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span>Hello Marie. Hello Berlioz. Hello Toulouse.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>你好玛丽, 你好柏里欧, 你好图鲁兹.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">dict</span><span style="color: #0000ff;">&gt;</span></div><p>这个XML中含有几个元素:key:单词本身; ps:第一个是英音音标,第二个是美音音标; pron第一个是英音的MP3地址,第二个是美音的;pos 词性; acception 词义;sent &nbsp;例句; orig例句英语;trans例句中文翻译。我们要做的就是根据XML文件把这几个元素解析出来。</p><p>  另外这里就涉及到了另一个问题:能不能查中文呢?一开始我也没搞出来,后来才发现了秘密查中文(或日文)需要在待查的词前面加上一个下划线 _ 即如 :_你好</p><p>搜索你好:http://dict-co.iciba.com/api/dictionary.php?w=_你好&amp;key=这里换成你的API key&nbsp; ,得到结果</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">dict </span><span style="color: #ff0000;">num</span><span style="color: #0000ff;">="219"</span><span style="color: #ff0000;"> id</span><span style="color: #0000ff;">="219"</span><span style="color: #ff0000;"> name</span><span style="color: #0000ff;">="219"</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">key</span><span style="color: #0000ff;">&gt;</span>你好<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">key</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">fy</span><span style="color: #0000ff;">&gt;</span>Hello<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">fy</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span>Hello! Hello! Hello! Hello! Hel - lo!<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>你好! 你好! 你好! 你好! 你好!<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span>Hello! Hello! Hello! Hello ! I'm glad to meet you.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>你好! 你好! 你好! 你好! 见到你很高兴.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span>Hello Marie. Hello Berlioz. Hello Toulouse.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>你好玛丽, 你好柏里欧, 你好图鲁兹.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><span style="color: #000000;"><br/>B Hi Gao. How are you doing? It's good to meet you.<br/></span><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>B你好,高. 你好 吗 ?很高兴认识你.<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><span style="color: #000000;"><br/>Grant: Hi , Tess. Hi , Jenna. Are you doing your homework?<br/></span><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">orig</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span>格兰特: 你好! 苔丝. 你好! 詹娜. 你们在做家庭作业 吗 ?<span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">trans</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">sent</span><span style="color: #0000ff;">&gt;</span><br/><span style="color: #0000ff;">&lt;/</span><span style="color: #800000;">dict</span><span style="color: #0000ff;">&gt;</span></div><p>注意fy元素就是查询的意思,在解析XML文件时要考虑到这一点。</p><p>&nbsp;</p><p>  根据以上分析,我们首先需要访问网络,将这个xml文件下载下来并进行解析,下面我给出几个工具类:</p><p>访问网络类,注意对应的要在AndroidManifest.xml文件中添加访问网络权限。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">package</span><span style="color: #000000;"> com.carlos.internet;<br/><br/></span><span style="color: #0000ff;">import</span><span style="color: #000000;"> java.io.InputStream;<br/></span><span style="color: #0000ff;">import</span><span style="color: #000000;"> java.net.HttpURLConnection;<br/></span><span style="color: #0000ff;">import</span><span style="color: #000000;"> java.net.URL;<br/><br/></span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">class</span><span style="color: #000000;"> NetOperator {<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">final</span> <span style="color: #0000ff;">static</span> String iCiBaURL1="http://dict-co.iciba.com/api/dictionary.php?w="<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">final</span> <span style="color: #0000ff;">static</span> String iCiBaURL2="&amp;key=你申请的APIkey,不要忘记了替换!!"<span style="color: #000000;">; 注意!<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span><span style="color: #000000;"> InputStream getInputStreamByUrl(String urlStr){<br/> InputStream tempInput</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> URL url</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> HttpURLConnection connection</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">; <br/> </span><span style="color: #008000;">//</span><span style="color: #008000;">设置超时时间</span><br/> <br/> <span style="color: #0000ff;">try</span><span style="color: #000000;">{<br/> url</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> URL(urlStr);<br/> connection</span>=(HttpURLConnection)url.openConnection(); <span style="color: #008000;">//</span><span style="color: #008000;">别忘了强制类型转换</span><br/> connection.setConnectTimeout(8000<span style="color: #000000;">);<br/> connection.setReadTimeout(</span>10000<span style="color: #000000;">);<br/> tempInput</span>=<span style="color: #000000;">connection.getInputStream();<br/> }</span><span style="color: #0000ff;">catch</span><span style="color: #000000;">(Exception e){<br/> e.printStackTrace();<br/> }<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> tempInput;<br/> }<br/><br/>}</span></div><p>&nbsp;</p><p> 这个类的功能就是根据给出的URL,从网络获得输入流,iCiBaURL1 和iCiBaURL2是用于构成查单词的URL的。iCiBaURL1+要查的单词+iCiBaURL2 &nbsp;就构成了金山查单词的URL</p><p>这里首先给出一个对象WordValue,该对象用来存放一个单词的信息:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">class</span><span style="color: #000000;"> WordValue {<br/> </span><span style="color: #0000ff;">public</span> String word=<span style="color: #0000ff;">null</span>,psE=<span style="color: #0000ff;">null</span>,pronE=<span style="color: #0000ff;">null</span>,psA=<span style="color: #0000ff;">null</span>,pronA=<span style="color: #0000ff;">null</span><span style="color: #000000;">,<br/> interpret</span>=<span style="color: #0000ff;">null</span>,sentOrig=<span style="color: #0000ff;">null</span>,sentTrans=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> <br/> <br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> WordValue(String word, String psE, String pronE, String psA,<br/> String pronA, String interpret, String sentOrig, String sentTrans) {<br/> </span><span style="color: #0000ff;">super</span><span style="color: #000000;">();<br/> </span><span style="color: #0000ff;">this</span>.word = ""+<span style="color: #000000;">word;<br/> </span><span style="color: #0000ff;">this</span>.psE = ""+<span style="color: #000000;">psE;<br/> </span><span style="color: #0000ff;">this</span>.pronE = ""+<span style="color: #000000;">pronE;<br/> </span><span style="color: #0000ff;">this</span>.psA = ""+<span style="color: #000000;">psA;<br/> </span><span style="color: #0000ff;">this</span>.pronA = ""+<span style="color: #000000;">pronA;<br/> </span><span style="color: #0000ff;">this</span>.interpret = ""+<span style="color: #000000;">interpret;<br/> </span><span style="color: #0000ff;">this</span>.sentOrig = ""+<span style="color: #000000;">sentOrig;<br/> </span><span style="color: #0000ff;">this</span>.sentTrans = ""+<span style="color: #000000;">sentTrans;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> WordValue() {<br/> </span><span style="color: #0000ff;">super</span><span style="color: #000000;">();<br/> </span><span style="color: #0000ff;">this</span>.word = ""; <span style="color: #008000;">//</span><span style="color: #008000;">防止空指针异常</span><br/> <span style="color: #0000ff;">this</span>.psE = ""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">this</span>.pronE = ""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">this</span>.psA = ""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">this</span>.pronA = ""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">this</span>.interpret = ""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">this</span>.sentOrig = ""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">this</span>.sentTrans = ""<span style="color: #000000;">;<br/> <br/> <br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> ArrayList&lt;String&gt;<span style="color: #000000;"> getOrigList(){<br/> ArrayList</span>&lt;String&gt; list=<span style="color: #0000ff;">new</span> ArrayList&lt;String&gt;<span style="color: #000000;">();<br/> BufferedReader br</span>=<span style="color: #0000ff;">new</span> BufferedReader(<span style="color: #0000ff;">new</span> StringReader(<span style="color: #0000ff;">this</span><span style="color: #000000;">.sentOrig));<br/> String str</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">try</span><span style="color: #000000;">{<br/> </span><span style="color: #0000ff;">while</span>((str=br.readLine())!=<span style="color: #0000ff;">null</span><span style="color: #000000;">){<br/> list.add(str);<br/> }<br/> }</span><span style="color: #0000ff;">catch</span><span style="color: #000000;">(Exception e){<br/> e.printStackTrace();<br/> }<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> list;<br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> ArrayList&lt;String&gt;<span style="color: #000000;"> getTransList(){<br/> ArrayList</span>&lt;String&gt; list=<span style="color: #0000ff;">new</span> ArrayList&lt;String&gt;<span style="color: #000000;">();<br/> BufferedReader br</span>=<span style="color: #0000ff;">new</span> BufferedReader(<span style="color: #0000ff;">new</span> StringReader(<span style="color: #0000ff;">this</span><span style="color: #000000;">.sentTrans));<br/> String str</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">try</span><span style="color: #000000;">{<br/> </span><span style="color: #0000ff;">while</span>((str=br.readLine())!=<span style="color: #0000ff;">null</span><span style="color: #000000;">){<br/> list.add(str);<br/> }<br/> }</span><span style="color: #0000ff;">catch</span><span style="color: #000000;">(Exception e){<br/> e.printStackTrace();<br/> }<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> list;<br/> }<br/> <br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getWord() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> word;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setWord(String word) {<br/> </span><span style="color: #0000ff;">this</span>.word =<span style="color: #000000;"> word;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getPsE() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> psE;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setPsE(String psE) {<br/> </span><span style="color: #0000ff;">this</span>.psE =<span style="color: #000000;"> psE;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getPronE() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> pronE;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setPronE(String pronE) {<br/> </span><span style="color: #0000ff;">this</span>.pronE =<span style="color: #000000;"> pronE;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getPsA() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> psA;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setPsA(String psA) {<br/> </span><span style="color: #0000ff;">this</span>.psA =<span style="color: #000000;"> psA;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getPronA() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> pronA;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setPronA(String pronA) {<br/> </span><span style="color: #0000ff;">this</span>.pronA =<span style="color: #000000;"> pronA;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getInterpret() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> interpret;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setInterpret(String interpret) {<br/> </span><span style="color: #0000ff;">this</span>.interpret =<span style="color: #000000;"> interpret;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getSentOrig() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> sentOrig;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setSentOrig(String sentOrig) {<br/> </span><span style="color: #0000ff;">this</span>.sentOrig =<span style="color: #000000;"> sentOrig;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> String getSentTrans() {<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> sentTrans;<br/> }<br/><br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> setSentTrans(String sentTrans) {<br/> </span><span style="color: #0000ff;">this</span>.sentTrans =<span style="color: #000000;"> sentTrans;<br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> printInfo(){<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.word);<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.psE);<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.pronE);<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.psA);<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.pronA);<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.interpret);<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.sentOrig);<br/> System.out.println(</span><span style="color: #0000ff;">this</span><span style="color: #000000;">.sentTrans);<br/><br/> }<br/> <br/><br/>}</span></div><p>大家从成员变量的名字就可以看出,这个对象中的成员就对应从XML文件中解析出来的各个元素,大家可以在上面XML的介绍中找对应。</p><p>&nbsp;</p><p>接下来是一个ContentHandler对象,用于对XML的解析:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">class</span> JinShanContentHandler <span style="color: #0000ff;">extends</span><span style="color: #000000;"> DefaultHandler{<br/><br/> </span><span style="color: #0000ff;">public</span> WordValue wordValue=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">private</span> String tagName=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">private</span> String interpret=""; <span style="color: #008000;">//</span><span style="color: #008000;">防止空指针异常</span><br/> <span style="color: #0000ff;">private</span> String orig=""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">private</span> String trans=""<span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">private</span> <span style="color: #0000ff;">boolean</span> isChinese=<span style="color: #0000ff;">false</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> JinShanContentHandler(){<br/> wordValue</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> WordValue();<br/> isChinese</span>=<span style="color: #0000ff;">false</span><span style="color: #000000;">;<br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> WordValue getWordValue(){<br/> <br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> wordValue;<br/> }<br/> <br/> @Override<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span> characters(<span style="color: #0000ff;">char</span>[] ch, <span style="color: #0000ff;">int</span> start, <span style="color: #0000ff;">int</span><span style="color: #000000;"> length)<br/> </span><span style="color: #0000ff;">throws</span><span style="color: #000000;"> SAXException {<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> TODO Auto-generated method stub</span><br/> <span style="color: #0000ff;">super</span><span style="color: #000000;">.characters(ch, start, length);<br/> </span><span style="color: #0000ff;">if</span>(length&lt;=0<span style="color: #000000;">)<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=start; i&lt;start+length; i++<span style="color: #000000;">){<br/> </span><span style="color: #0000ff;">if</span>(ch[i]=='\n'<span style="color: #000000;">)<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;">;<br/> }<br/> <br/> </span><span style="color: #008000;">//</span><span style="color: #008000;">去除莫名其妙的换行!</span><br/><span style="color: #000000;"> <br/> String str</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> String(ch,start,length);<br/> </span><span style="color: #0000ff;">if</span>(tagName=="key"<span style="color: #000000;">){<br/> wordValue.setWord(str);<br/> }</span><span style="color: #0000ff;">else</span> <span style="color: #0000ff;">if</span>(tagName=="ps"<span style="color: #000000;">){<br/> </span><span style="color: #0000ff;">if</span>(wordValue.getPsE().length()&lt;=0<span style="color: #000000;">){<br/> wordValue.setPsE(str);<br/> }</span><span style="color: #0000ff;">else</span><span style="color: #000000;">{<br/> wordValue.setPsA(str);<br/> }<br/> }</span><span style="color: #0000ff;">else</span> <span style="color: #0000ff;">if</span>(tagName=="pron"<span style="color: #000000;">){<br/> </span><span style="color: #0000ff;">if</span>(wordValue.getPronE().length()&lt;=0<span style="color: #000000;">){<br/> wordValue.setPronE(str);<br/> }</span><span style="color: #0000ff;">else</span><span style="color: #000000;">{<br/> wordValue.setPronA(str);<br/> }<br/> }</span><span style="color: #0000ff;">else</span> <span style="color: #0000ff;">if</span>(tagName=="pos"<span style="color: #000000;">){<br/> isChinese</span>=<span style="color: #0000ff;">false</span><span style="color: #000000;">;<br/> interpret</span>=interpret+str+" "<span style="color: #000000;">;<br/> }</span><span style="color: #0000ff;">else</span> <span style="color: #0000ff;">if</span>(tagName=="acceptation"<span style="color: #000000;">){<br/> interpret</span>=interpret+str+"\n"<span style="color: #000000;">;<br/> interpret</span>=wordValue.getInterpret()+<span style="color: #000000;">interpret;<br/> wordValue.setInterpret(interpret);<br/> interpret</span>=""; <span style="color: #008000;">//</span><span style="color: #008000;">初始化操作,预防有多个释义</span><br/> }<span style="color: #0000ff;">else</span> <span style="color: #0000ff;">if</span>(tagName=="orig"<span style="color: #000000;">){<br/> <br/> <br/> orig</span>=<span style="color: #000000;">wordValue.getSentOrig();<br/> wordValue.setSentOrig(orig</span>+str+"\n"<span style="color: #000000;">);<br/> <br/> <br/> }</span><span style="color: #0000ff;">else</span> <span style="color: #0000ff;">if</span>(tagName=="trans"<span style="color: #000000;">){<br/> String temp</span>=wordValue.getSentTrans()+str+"\n"<span style="color: #000000;">;<br/> wordValue.setSentTrans(temp);<br/> <br/> }</span><span style="color: #0000ff;">else</span> <span style="color: #0000ff;">if</span>(tagName=="fy"<span style="color: #000000;">){<br/> isChinese</span>=<span style="color: #0000ff;">true</span><span style="color: #000000;">;<br/> wordValue.setInterpret(str);<br/> }<br/><br/><br/> }<br/><br/><br/><br/> @Override<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> endElement(String uri, String localName, String qName)<br/> </span><span style="color: #0000ff;">throws</span><span style="color: #000000;"> SAXException {<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> TODO Auto-generated method stub</span><br/> <span style="color: #0000ff;">super</span><span style="color: #000000;">.endElement(uri, localName, qName);<br/> tagName</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> <br/><br/> }<br/><br/><br/> @Override<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> startElement(String uri, String localName, String qName,<br/> Attributes attributes) </span><span style="color: #0000ff;">throws</span><span style="color: #000000;"> SAXException {<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> TODO Auto-generated method stub</span><br/> <span style="color: #0000ff;">super</span><span style="color: #000000;">.startElement(uri, localName, qName, attributes);<br/> tagName</span>=<span style="color: #000000;">localName;<br/> }<br/><br/> @Override<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span> endDocument() <span style="color: #0000ff;">throws</span><span style="color: #000000;"> SAXException {<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> TODO Auto-generated method stub</span><br/> <span style="color: #0000ff;">super</span><span style="color: #000000;">.endDocument();<br/> </span><span style="color: #0000ff;">if</span><span style="color: #000000;">(isChinese)<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;">;<br/> String interpret</span>=<span style="color: #000000;">wordValue.getInterpret();<br/> </span><span style="color: #0000ff;">if</span>(interpret!=<span style="color: #0000ff;">null</span> &amp;&amp; interpret.length()&gt;0<span style="color: #000000;">){<br/> </span><span style="color: #0000ff;">char</span>[] strArray=<span style="color: #000000;">interpret.toCharArray();<br/> wordValue.setInterpret(</span><span style="color: #0000ff;">new</span> String(strArray,0,interpret.length()-1<span style="color: #000000;">));<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;">去掉解释的最后一个换行符</span><br/><span style="color: #000000;"> }<br/> <br/> }<br/> <br/> <br/> <br/><br/>}</span></div><p>这里要注意的是:关于XML解析的基本知识我不想讲了,因为要讲这样细的话我三个月也完成不了这个系列。大家若有不懂的可以参考Mars 陈川老师的安卓开发视频教程,非常基础,入门必备。</p><p>关于XML解析基本就是用的他的的思路,但我想补充几点细节的东西:</p><p>  1)不仅在startElement()之后会调用character()方法,在endElement()之后也会调用character90方法;</p><p>&nbsp; &nbsp; &nbsp;2)在具有多层的元素如上面的sent元素里面有嵌套了orig 和trans元素,此时character()方法并不会严格地在startElement之后就立即调用;</p><p>以上两点就会导致一个问题:会把多余的空行(换行符)也读取进来,所以我在程序中添加了清除换行的代码。请看注释。</p><p>&nbsp; &nbsp; &nbsp;3)这里也考虑了查英文的翻译结果是pos acception &nbsp;而查中文的翻译结果是 fy ,可以看看上面ContentHandler中character()方法,这个方法是核心。</p><p>&nbsp;</p><p>接下来就是一个XMLParser对象,该对象把XML解析用的SAXParserFactory等获取实例的工作封装起来,有了这个对象,解析XML时只需创建一个XMLParser对象,调用的该对象的parseJinShanXml()方法即可。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">class</span><span style="color: #000000;"> XMLParser {<br/> </span><span style="color: #0000ff;">public</span> SAXParserFactory factory=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">public</span> XMLReader reader=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> <br/> </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> XMLParser(){<br/> <br/> </span><span style="color: #0000ff;">try</span><span style="color: #000000;"> {<br/> factory</span>=<span style="color: #000000;">SAXParserFactory.newInstance();<br/> reader</span>=<span style="color: #000000;">factory.newSAXParser().getXMLReader();<br/> } </span><span style="color: #0000ff;">catch</span><span style="color: #000000;"> (Exception e) {<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> TODO Auto-generated catch block</span><br/><span style="color: #000000;"> e.printStackTrace();<br/> }<br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> parseJinShanXml(DefaultHandler content, InputSource inSource){<br/> </span><span style="color: #0000ff;">if</span>(inSource==<span style="color: #0000ff;">null</span><span style="color: #000000;">)<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">try</span><span style="color: #000000;"> {<br/> reader.setContentHandler(content);<br/> reader.parse(inSource);<br/> } </span><span style="color: #0000ff;">catch</span><span style="color: #000000;"> (Exception e) {<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> TODO Auto-generated catch block</span><br/><span style="color: #000000;"> e.printStackTrace(); <br/> } <br/> <br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> parseDailySentenceXml(DailySentContentHandler contentHandler, InputSource inSource){<br/> </span><span style="color: #0000ff;">if</span>(inSource==<span style="color: #0000ff;">null</span><span style="color: #000000;">)<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">try</span><span style="color: #000000;"> {<br/> reader.setContentHandler(contentHandler);<br/> reader.parse(inSource);<br/> } </span><span style="color: #0000ff;">catch</span><span style="color: #000000;"> (Exception e) {<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> TODO Auto-generated catch block</span><br/><span style="color: #000000;"> e.printStackTrace(); <br/> } <br/> }<br/> <br/><br/>}</span></div><p>大家可以看到还有一个&nbsp;parseDailySentenceXml()方法,这是解析每日一句的,暂时不用管它。</p><p>&nbsp;</p><p>那么如何根据一个单词来获取它的XML并进行解析呢?即如何进行调用?方法如下:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">public</span><span style="color: #000000;"> WordValue getWordFromInternet(String searchedWord){<br/> WordValue wordValue</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> String tempWord</span>=<span style="color: #000000;">searchedWord;<br/> </span><span style="color: #0000ff;">if</span>(tempWord==<span style="color: #0000ff;">null</span>&amp;&amp; tempWord.equals(""<span style="color: #000000;">))<br/> </span><span style="color: #0000ff;">return</span> <span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">char</span>[] array=<span style="color: #000000;">tempWord.toCharArray();<br/> </span><span style="color: #0000ff;">if</span>(array[0]&gt;256) <span style="color: #008000;">//</span><span style="color: #008000;">是中文,或其他语言的的简略判断</span><br/> tempWord="_"+<span style="color: #000000;">URLEncoder.encode(tempWord);<br/> InputStream in</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> String str</span>=<span style="color: #0000ff;">null</span><span style="color: #000000;">;<br/> </span><span style="color: #0000ff;">try</span><span style="color: #000000;">{<br/> String tempUrl</span>=NetOperator.iCiBaURL1+tempWord+<span style="color: #000000;">NetOperator.iCiBaURL2;<br/> in</span>=<span style="color: #000000;">NetOperator.getInputStreamByUrl(tempUrl); //从网络获得输入流<br/> </span><span style="color: #0000ff;">if</span>(in!=<span style="color: #0000ff;">null</span><span style="color: #000000;">){<br/> <span style="color: #00ff00;">//</span></span><span style="color: #00ff00;">new FileUtils().saveInputStreamToFile(in, "", "gfdgf.txt"</span><span style="color: #000000;"><span style="color: #00ff00;">); </span> <br/> XMLParser xmlParser</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> XMLParser();<br/> InputStreamReader reader</span>=<span style="color: #0000ff;">new</span> InputStreamReader(in,"utf-8"<span style="color: #000000;">); //最终目的获得一个InputSource对象用于传入形参<br/> JinShanContentHandler contentHandler</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> JinShanContentHandler();<br/> xmlParser.parseJinShanXml(contentHandler, </span><span style="color: #0000ff;">new</span><span style="color: #000000;"> InputSource(reader));<br/> wordValue</span>=<span style="color: #000000;">contentHandler.getWordValue();<br/> wordValue.setWord(searchedWord);<br/> }<br/> }</span><span style="color: #0000ff;">catch</span><span style="color: #000000;">(Exception e){<br/> e.printStackTrace();<br/> }<br/> </span><span style="color: #0000ff;">return</span><span style="color: #000000;"> wordValue;<br/> }</span></div><p>&nbsp;</p>这是我从Dict类中截取的一个方法,注意这一个:tempWord="_"+<span>URLEncoder.encode(tempWord); HttpURL中存在中文的话,会因为编码的问题产生乱码,所以先要对中文调用URLEncoder.encode()方法进行一下编码,这样才能得到正常的XML文件,这个问题当时困扰了我好久!<br />另外注意InputStream是二进制字节流,必须先经过InputStreamReader包装成字符流在创建InputSource对象,否则会出现编码异常,这个是我的一个经验。<br />调用这个方法就可以从网上获得要查询的单词的信息,并返回一个WordValue对象,然后我们再进行其它操作。<br /><br />  另外有一点必须强调:<span style="font-size: 16px; color: #ff0000;">如果某个方法要访问网络,必须开辟一个子线程,在子线程里调用该方法!!!!!<br /><br /></span>今天就介绍到这里吧,这个项目要讲完还得不少时间,毕竟我写了三个星期。另外在在整个程序大体讲完之前我不打算共享源代码,希望大家能够体谅。写这个Blog的目的主要是分享一下经验和思路,而不是分享代码。</span><p>&nbsp;</p><img src="http://counter.cnblogs.com/blog/rss/3930424" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/carlos-vic/p/YueCi_2.html" target="_blank">和我一起开发Android应用(二)——“悦词-i背单词”项目功能分析</a>,转载请注明。</p>http://www.cnblogs.com/lisr/p/3930381.htmlExt JS学习第六天 Ext_window组件(三) - code&monkey此文用来记录学习笔记;今天再来一个例子巩固一下学习的window;–例2: 在window中添加子组件,并讲解常用查找组件的方式:|重点分析:该实例主要针对于组件的查找进行详细的讲解,在以后的应用开发中,同学们应该学会各种方式去查找所需要的组件,不要拘泥于某种特定形式,这样会给开发思路带来很多好处。...2014-08-22T15:07:00Z2014-08-22T15:07:00Zcode&amp;monkeyhttp://www.cnblogs.com/lisr/<p>&nbsp;</p><p>此文用来记录学习笔记;</p><p>今天再来一个例子巩固一下学习的window;</p><div class="O1">&ndash;例2: 在window中添加子组件,并讲解常用查找组件的方式:</div><div class="O2">&bull;重点分析:该实例主要针对于组件的查找进行详细的讲解,在以后的应用开发中,同学们应该学会各种方式去查找所需要的组件,不要拘泥于某种特定形式,这样会给开发思路带来很多好处。</div><div class="O3">&ndash;ownerCt</div><div class="O3">&ndash;up/down方法</div><div class="O3">&ndash;Ext.getCmp方法</div><div class="O3">附上栗子代码1</div><div class="O3"><div class="O3"><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">Ext.onReady(<span style="color: #0000ff;">function<span style="color: #000000;">(){<br/><br/> <span style="color: #008000;">//<span style="color: #008000;">ex002 : 在组件中添加子组件 ,并进行一系列针对于组件的操作<br/> <br/> <span style="color: #008000;">//<span style="color: #008000;">在组件中添加子组件:<br/> <span style="color: #0000ff;">var win = <span style="color: #0000ff;">new<span style="color: #000000;"> Ext.window.Window({<br/> title:"添加子组件实例"<span style="color: #000000;"> , <br/> width:'40%'<span style="color: #000000;"> ,<br/> height:400<span style="color: #000000;"> , <br/> renderTo:Ext.getBody() ,<br/> draggable:<span style="color: #0000ff;">false , <span style="color: #008000;">//<span style="color: #008000;">不允许拖拽<br/> resizable:<span style="color: #0000ff;">false , <span style="color: #008000;">//<span style="color: #008000;">不允许改变窗口大小<br/> closable:<span style="color: #0000ff;">false, <span style="color: #008000;">//<span style="color: #008000;">不显示关闭按钮<br/> collapsible:<span style="color: #0000ff;">true , <span style="color: #008000;">//<span style="color: #008000;">显示折叠按钮<br/> bodyStyle: 'background:#ffc; padding:10px;' , <span style="color: #008000;">//<span style="color: #008000;"> 设置样式<br/> html:'我是window的内容!!'<span style="color: #000000;"> ,<br/> <span style="color: #008000;">//<span style="color: #008000;">Ext items(array) 配置子组件的配置项<br/><span style="color: #000000;"> items:[{<br/> <span style="color: #008000;">//<span style="color: #008000;">Ext的组件 给我们提供了一个简单的写法 xtype属性去创建组件<br/> xtype:'panel'<span style="color: #000000;">,<br/> width:'50%'<span style="color: #000000;">,<br/> height:100<span style="color: #000000;"> ,<br/> html:'我是面板'<span style="color: #000000;"><br/> },<br/> <span style="color: #008000;">//<span style="color: #008000;">第一种写法<br/> <span style="color: #0000ff;">new<span style="color: #000000;"> Ext.button.Button({<br/> text:'我是按钮'<span style="color: #000000;"> , <br/> handler:<span style="color: #0000ff;">function<span style="color: #000000;">(){<br/> alert('执行!!'<span style="color: #000000;">);<br/> }<br/> })<br/> <span style="color: #008000;">//<span style="color: #008000;">第二种写法<span style="color: #008000;"><br/>//<span style="color: #008000;"> {<span style="color: #008000;"><br/>//<span style="color: #008000;"> xtype:'button' , <span style="color: #008000;"><br/>//<span style="color: #008000;"> text:'我是按钮',<span style="color: #008000;"><br/>//<span style="color: #008000;"> handler:function(btn){<span style="color: #008000;"><br/>//<span style="color: #008000;"> alert('我被点击了');<span style="color: #008000;"><br/>//<span style="color: #008000;"> alert(btn.text);<span style="color: #008000;"><br/>//<span style="color: #008000;"> }<span style="color: #008000;"><br/>//<span style="color: #008000;"> }<br/><span style="color: #000000;"> ]<br/> <br/> });<br/> win.show(); <br/><br/> <br/> <br/>});</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></div></div></div><div class="O3">栗子代码2</div><div class="O3"><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;"> 1 Ext.onReady(<span style="color: #0000ff;">function<span style="color: #000000;">(){<br/><span style="color: #008080;"> 2 <br/><span style="color: #008080;"> 3 <span style="color: #008000;">//<span style="color: #008000;">ex002 : 在组件中添加子组件 ,并进行一系列针对于组件的操作<br/><span style="color: #008080;"> 4 <br/><span style="color: #008080;"> 5 <br/><span style="color: #008080;"> 6 <span style="color: #0000ff;">var win = <span style="color: #0000ff;">new<span style="color: #000000;"> Ext.Window({<br/><span style="color: #008080;"> 7 id:'mywin'<span style="color: #000000;"> ,<br/><span style="color: #008080;"> 8 title:'操作组件的形式'<span style="color: #000000;"> ,<br/><span style="color: #008080;"> 9 width:500<span style="color: #000000;"> , <br/><span style="color: #008080;">10 height:300<span style="color: #000000;"> , <br/><span style="color: #008080;">11 <span style="color: #000000;"> renderTo:Ext.getBody() , <br/><span style="color: #008080;">12 <span style="color: #008000;">//<span style="color: #008000;">表示在当前组件的top位置添加一个工具条<br/><span style="color: #008080;">13 tbar:[{ <span style="color: #008000;">//<span style="color: #008000;">bbar(bottom) lbar(leftbar) rbar(rightbar) fbar(footbar)<br/><span style="color: #008080;">14 text:'按钮1'<span style="color: #000000;"> ,<br/><span style="color: #008080;">15 handler:<span style="color: #0000ff;">function<span style="color: #000000;">(btn){<br/><span style="color: #008080;">16 <span style="color: #008000;">//<span style="color: #008000;">组件都会有 up、down 这两个方法(表示向上、或者向下查找) 需要的参数是组件的xtype或者是选择器<br/><span style="color: #008080;">17 alert(btn.up('window'<span style="color: #000000;">).title);<br/><span style="color: #008080;">18 <span style="color: #000000;"> }<br/><span style="color: #008080;">19 <span style="color: #000000;"> },{<br/><span style="color: #008080;">20 text:'按钮2'<span style="color: #000000;"> , <br/><span style="color: #008080;">21 handler:<span style="color: #0000ff;">function<span style="color: #000000;">(btn){<br/><span style="color: #008080;">22 <span style="color: #008000;">//<span style="color: #008000;">最常用的方式<br/><span style="color: #008080;">23 alert(Ext.getCmp('mywin'<span style="color: #000000;">).title);<br/><span style="color: #008080;">24 <span style="color: #000000;"> }<br/><span style="color: #008080;">25 <span style="color: #000000;"> },{<br/><span style="color: #008080;">26 text:'按钮3'<span style="color: #000000;"> ,<br/><span style="color: #008080;">27 handler:<span style="color: #0000ff;">function<span style="color: #000000;">(btn){<br/><span style="color: #008080;">28 <span style="color: #008000;">//<span style="color: #008000;">以上一级组件的形式去查找 OwnerCt<br/><span style="color: #008080;">29 <span style="color: #008000;">//<span style="color: #008000;">console.info(btn.ownerCt);<br/><span style="color: #008080;">30 <span style="color: #000000;"> alert(btn.ownerCt.ownerCt.title);<br/><span style="color: #008080;">31 <span style="color: #000000;"> } <br/><span style="color: #008080;">32 <span style="color: #000000;"> }] <br/><span style="color: #008080;">33 <span style="color: #000000;"> });<br/><span style="color: #008080;">34 <span style="color: #000000;"> win.show();<br/><span style="color: #008080;">35 <br/><span style="color: #008080;">36 <br/><span style="color: #008080;">37 });</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></div></div><div class="O3">学习之余给大家推荐一个美文网站<a href="http://www.fishcmonkey.com">www.fishcmonkey.com</a>,提高文学修养也是好的</div><img src="http://counter.cnblogs.com/blog/rss/3930381" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/lisr/p/3930381.html" target="_blank">Ext JS学习第六天 Ext_window组件(三)</a>,转载请注明。</p>http://www.cnblogs.com/popsuper1982/p/3930350.html别以为真懂Openstack: 虚拟机创建的50个步骤和100个知识点(2) - popsuper1982二、nova-api步骤3:nova-api接收请求nova-api接收请求,也不是随便怎么来都接收的,而是需要设定rate limits,默认的实现是在ratelimit的middleware里面实现的。然而有时候,我们希望实现distributed rate-limiting,从而Turnsti...2014-08-22T14:29:00Z2014-08-22T14:29:00Zpopsuper1982http://www.cnblogs.com/popsuper1982/<p><strong>二、nova-api</strong></p><p><img style="display: inline; border-width: 0px;" title="image" src="http://images.cnitblog.com/blog/635909/201408/222228515182030.png" alt="image" width="734" height="768" border="0" /></p><p><strong>步骤3:nova-api接收请求</strong></p><p>nova-api接收请求,也不是随便怎么来都接收的,而是需要设定rate limits,默认的实现是在ratelimit的middleware里面实现的。</p><p>然而有时候,我们希望实现distributed rate-limiting,从而Turnstile是一个不错的选择。</p><p><a href="https://github.com/klmitch/turnstile">https://github.com/klmitch/turnstile</a><br /><a href="http://pypi.python.org/pypi/turnstile">http://pypi.python.org/pypi/turnstile</a></p><p><strong>步骤4:对Token的验证</strong></p><p><strong>步骤5:查看Policy</strong></p><p>这两步已经在keystone的时候研究过</p><p><strong>步骤6:检查quota</strong></p><p>nova, neutron, Cinder各有各的quota,并且可以从命令行进行管理</p><p># nova -h | grep quota<br />&nbsp;&nbsp;&nbsp; quota-class-show&nbsp;&nbsp;&nbsp; List the quotas for a quota class.<br />&nbsp;&nbsp;&nbsp; quota-class-update&nbsp; Update the quotas for a quota class.<br />&nbsp;&nbsp;&nbsp; quota-defaults&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List the default quotas for a tenant.<br />&nbsp;&nbsp;&nbsp; quota-delete&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Delete quota for a tenant/user so their quota will<br />&nbsp;&nbsp;&nbsp; quota-show&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List the quotas for a tenant/user.<br />&nbsp;&nbsp;&nbsp; quota-update&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Update the quotas for a tenant/user.</p><p># nova quota-show<br />+-----------------------------+-------+<br />| Quota&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | Limit |<br />+-----------------------------+-------+<br />| instances&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 10&nbsp;&nbsp;&nbsp; |<br />| cores&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 20&nbsp;&nbsp;&nbsp; |<br />| ram&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 51200 |<br />| floating_ips&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 10&nbsp;&nbsp;&nbsp; |<br />| fixed_ips&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | -1&nbsp;&nbsp;&nbsp; |<br />| metadata_items&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 128&nbsp;&nbsp; |<br />| injected_files&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 5&nbsp;&nbsp;&nbsp;&nbsp; |<br />| injected_file_content_bytes | 10240 |<br />| injected_file_path_bytes&nbsp;&nbsp;&nbsp; | 255&nbsp;&nbsp; |<br />| key_pairs&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 100&nbsp;&nbsp; |<br />| security_groups&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 10&nbsp;&nbsp;&nbsp; |<br />| security_group_rules&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 20&nbsp;&nbsp;&nbsp; |<br />+-----------------------------+-------+</p><p># cinder -h | grep quota<br />&nbsp;&nbsp;&nbsp; quota-class-show&nbsp;&nbsp;&nbsp; List the quotas for a quota class.<br />&nbsp;&nbsp;&nbsp; quota-class-update&nbsp; Update the quotas for a quota class.<br />&nbsp;&nbsp;&nbsp; quota-defaults&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List the default quotas for a tenant.<br />&nbsp;&nbsp;&nbsp; quota-show&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List the quotas for a tenant.<br />&nbsp;&nbsp;&nbsp; quota-update&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Update the quotas for a tenant.<br />&nbsp;&nbsp;&nbsp; quota-usage&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List the quota usage for a tenant.</p><p># cinder quota-show 1779b3bc725b44b98726fb0cbdc617b1<br />+-----------+-------+<br />|&nbsp; Property | Value |<br />+-----------+-------+<br />| gigabytes |&nbsp; 1000 |<br />| snapshots |&nbsp;&nbsp; 10&nbsp; |<br />|&nbsp; volumes&nbsp; |&nbsp;&nbsp; 10&nbsp; |<br />+-----------+-------+</p><p># neutron -h | grep quota<br />&nbsp; quota-delete&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Delete defined quotas of a given tenant.<br />&nbsp; quota-list&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List quotas of all tenants who have non-default quota values.<br />&nbsp; quota-show&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Show quotas of a given tenant<br />&nbsp; quota-update&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Define tenant's quotas not to use defaults.</p><p># neutron quota-show 1779b3bc725b44b98726fb0cbdc617b1<br />+---------------------+-------+<br />| Field&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | Value |<br />+---------------------+-------+<br />| floatingip&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 50&nbsp;&nbsp;&nbsp; |<br />| network&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 10&nbsp;&nbsp;&nbsp; |<br />| port&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 50&nbsp;&nbsp;&nbsp; |<br />| router&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 10&nbsp;&nbsp;&nbsp; |<br />| security_group&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 10&nbsp;&nbsp;&nbsp; |<br />| security_group_rule | 100&nbsp;&nbsp; |<br />| subnet&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | 10&nbsp;&nbsp;&nbsp; |<br />+---------------------+-------+</p><p>推荐下面的文章</p><p><span style="font-size: 15px;"><a href="http://blog.csdn.net/hackerain/article/details/8223125"><span>openstack nova 基础知识&mdash;&mdash;Quota(配额管理)</span> </a></span></p><p><a href="http://www.sebastien-han.fr/blog/2012/09/19/openstack-play-with-quota/">http://www.sebastien-han.fr/blog/2012/09/19/openstack-play-with-quota/</a></p><p><strong>步骤7:在数据库中创建Instance实例</strong></p><p>有关nova的database schema参考下面的文章</p><p><a title="http://www.prestonlee.com/2012/05/03/openstack-nova-essex-mysql-database-schema-diagram-and-sql/" href="http://www.prestonlee.com/2012/05/03/openstack-nova-essex-mysql-database-schema-diagram-and-sql/">http://www.prestonlee.com/2012/05/03/openstack-nova-essex-mysql-database-schema-diagram-and-sql/</a></p><p>MySQL是Openstack中最重要的组件之一,所以在生产环境中High Availability是必须的。</p><p>MySQL的HA有下面几种方式:</p><p><a title="http://dev.mysql.com/doc/mysql-ha-scalability/en/index.html" href="http://dev.mysql.com/doc/mysql-ha-scalability/en/index.html">http://dev.mysql.com/doc/mysql-ha-scalability/en/index.html</a></p><table style="width: 1034px;" border="2" cellspacing="0" cellpadding="0"><tbody><tr><td width="315">Requirement</td><td width="207">MySQL Replication</td><td width="246">MySQL with DRBD with Corosync and Pacemaker</td><td width="262">MySQL Cluster</td></tr><tr><td width="315">Availability</td><td width="207">&nbsp;</td><td width="246">&nbsp;</td><td width="262">&nbsp;</td></tr><tr><td width="315">Platform Support</td><td width="207">All Supported by MySQL Server</td><td width="246">Linux</td><td width="262">All Supported by MySQL Cluster</td></tr><tr><td width="315">Automated IP Failover</td><td width="207">No</td><td width="246">Yes</td><td width="262">Depends on Connector and Configuration</td></tr><tr><td width="315">Automated Database Failover</td><td width="207">No</td><td width="246">Yes</td><td width="262">Yes</td></tr><tr><td width="315">Automatic Data Resynchronization</td><td width="207">No</td><td width="246">Yes</td><td width="262">Yes</td></tr><tr><td width="315">Typical Failover Time</td><td width="207">User / Script Dependent</td><td width="246">Configuration Dependent, 60 seconds and Above</td><td width="262">1 Second and Less</td></tr><tr><td width="315">Synchronous Replication</td><td width="207">No, Asynchronous and Semisynchronous</td><td width="246">Yes</td><td width="262">Yes</td></tr><tr><td width="315">Shared Storage</td><td width="207">No, Distributed</td><td width="246">No, Distributed</td><td width="262">No, Distributed</td></tr><tr><td width="315">Geographic redundancy support</td><td width="207">Yes</td><td width="246">Yes, via MySQL Replication</td><td width="262">Yes, via MySQL Replication</td></tr><tr><td width="315">Update Schema On-Line</td><td width="207">No</td><td width="246">No</td><td width="262">Yes</td></tr><tr><td width="315">Scalability</td><td width="207">&nbsp;</td><td width="246">&nbsp;</td><td width="262">&nbsp;</td></tr><tr><td width="315">Number of Nodes</td><td width="207">One Master, Multiple Slaves</td><td width="246">One Active (primary), one Passive (secondary) Node</td><td width="262">255</td></tr><tr><td width="315">Built-in Load Balancing</td><td width="207">Reads, via MySQL Replication</td><td width="246">Reads, via MySQL Replication</td><td width="262">Yes, Reads and Writes</td></tr><tr><td width="315">Supports Read-Intensive Workloads</td><td width="207">Yes</td><td width="246">Yes</td><td width="262">Yes</td></tr><tr><td width="315">Supports Write-Intensive Workloads</td><td width="207">Yes, via Application-Level Sharding</td><td width="246">Yes, via Application-Level Sharding to Multiple Active/Passive Pairs</td><td width="262">Yes, via Auto-Sharding</td></tr><tr><td width="315">Scale On-Line (add nodes, repartition, etc.)</td><td width="207">No</td><td width="246">No</td><td width="262">Yes</td></tr></tbody></table><p>要想系统的学习Mysql replication,推荐下面的这本书</p><p>《MySQL High Availability Tools for Building Robust Data Centers》</p><p>还有一种方式是Mysql + galera,可以搭建Active + Active的Mysql应用</p><p><img title="MySQL分支版本 MySQL/Galera 1.0发布" src="http://yp.oss.org.cn/images/resource_images/20111101-163003//1320136203.png" alt="MySQL分支版本 MySQL/Galera 1.0发布" width="551" height="384" border="0" /></p><p> 参考下面的两篇文章</p><p><a title="http://www.sebastien-han.fr/blog/2012/04/08/mysql-galera-cluster-with-haproxy/" href="http://www.sebastien-han.fr/blog/2012/04/08/mysql-galera-cluster-with-haproxy/">http://www.sebastien-han.fr/blog/2012/04/08/mysql-galera-cluster-with-haproxy/</a></p><p><a title="http://www.sebastien-han.fr/blog/2012/04/01/mysql-multi-master-replication-with-galera/" href="http://www.sebastien-han.fr/blog/2012/04/01/mysql-multi-master-replication-with-galera/">http://www.sebastien-han.fr/blog/2012/04/01/mysql-multi-master-replication-with-galera/</a></p><p>还有一种常见的HA的技术,就是pacemaker</p><p><img src="http://images.cnitblog.com/blog/635909/201407/291500410243506.png" alt="image" />&nbsp;</p><p>最底层是通信层corosync/openais</p><p>负责cluster中node之间的通信</p><p>上一层是Resource Allocation Layer,包含下面的组件:</p><p>CRM Cluster Resouce Manager</p><p>是总管,对于resource做的任何操作都是通过它。每个机器上都有一个CRM。</p><p>CIB Cluster Information Base</p><p>CIB由CRM管理,是在内存中的XML数据库,保存了cluster的配置和状态。我们查询出来的configuration都是保存在CIB里面的。nodes, resources, constraints, relationship.</p><p>DC Designated Coordinator</p><p>每个node都有CRM,会有一个被选为DC,是整个Cluster的大脑,这个DC控制的CIB是master CIB,其他的CIB都是副本。</p><p>PE Policy Engine</p><p>当DC需要进行一些全局配置的时候,首先由PE根据当前的状态和配置,计算出将来的状态,并生成一系列的action,使得cluster从初始状态变为结果状态。PE仅仅在DC上运行。</p><p>LRM Local Resource Manager</p><p>本地的resource管理,调用resource agent完成操作,启停resource,将结果返回给CRM</p><p>再上一层是Resource Layer</p><p>包含多个resource agent。resource agent往往是一些shell script,用来启动,停止,监控resource的状态。</p><p>要弄懂Pacemaker,推荐读《SUSE high availability guide》</p><p><a title="https://www.suse.com/documentation/sle_ha/singlehtml/book_sleha/book_sleha.html" href="https://www.suse.com/documentation/sle_ha/singlehtml/book_sleha/book_sleha.html">https://www.suse.com/documentation/sle_ha/singlehtml/book_sleha/book_sleha.html</a></p><p>本人做了一些笔记和实验,请参考</p><p><a href="http://www.cnblogs.com/popsuper1982/p/3875403.html">High Availability手册(1): 环境</a></p><p><a href="http://www.cnblogs.com/popsuper1982/p/3875536.html">High Availability手册(2): 架构</a></p><p><a href="http://www.cnblogs.com/popsuper1982/p/3887611.html">High Availability手册(3): 配置</a></p><p><strong>步骤8:创建filter_properties,用于nova scheduler</strong></p><p><strong>步骤9:发送RPC给nova-conductor</strong></p><p>有关nova-conductor的文章</p><p><a title="http://cloudystuffhappens.blogspot.com/2013/04/understanding-nova-conductor-in.html" href="http://cloudystuffhappens.blogspot.com/2013/04/understanding-nova-conductor-in.html">http://cloudystuffhappens.blogspot.com/2013/04/understanding-nova-conductor-in.html</a></p><p>在Openstack中,RPC的发送是通过RabbitMQ</p><p>RabbitMQ可以通过Pacemaker进行HA,当然也可以搭建自己的RabbitMQ Cluster</p><p>学习RabbitMQ当然首推《RabbitMQ in Action》</p><p>本人也做了一些笔记</p><p><strong>&nbsp;</strong></p><p><a href="http://www.cnblogs.com/popsuper1982/p/3800237.html">RabbitMQ in Action (1): Understanding messaging</a></p><p><a href="http://www.cnblogs.com/popsuper1982/p/3800416.html">RabbitMQ in Action (2): Running and administering Rabbit</a></p><p><a href="http://www.cnblogs.com/popsuper1982/p/3800419.html">RabbitMQ in Action(5): Clustering and dealing with failure</a></p><p>还没完全读完,敬请谅解</p><p>当然Openstack中对于RabbitMQ的使用,一篇很好的文章是</p><p><a href="http://blog.csdn.net/gaoxingnengjisuan/article/details/9623529">NOVA源码分析&mdash;&mdash;NOVA中的RabbitMQ解析</a></p><p>本人也对RPC的调用过程进行了代码分析</p><p><a href="http://www.cnblogs.com/popsuper1982/p/3800396.html">Openstack中RabbitMQ RPC代码分析</a>&nbsp;</p><p><strong>步骤10:nova-condutor创建request_spec,用于scheduler</strong></p><p><strong>步骤11:nova-conductor发送RPC给nova-scheduler</strong></p><p><strong>三、nova-scheduler</strong></p><p>&nbsp;</p><p><img style="display: inline; border: 0px;" title="image" src="http://images.cnitblog.com/blog/635909/201408/222228558935532.png" alt="image" width="1024" height="467" border="0" /> </p><p>选择一个物理机来创建虚拟机,我们称为schedule的过程</p><p>nova scheduler的一个经典的图如下</p><p><img src="http://docs.openstack.org/developer/nova/_images/filteringWorkflow1.png" alt="../_images/filteringWorkflow1.png" /></p><p>就是先Filter再Weighting,其实scheduler的过程在很早就参与了。</p><p><strong>步骤13:对Host进行Filtering</strong></p><p>Filtering主要通过两个变量进行,request_spec和filter_properties,而这些变量在前面的步骤,都已经准备好了。</p><p>而不同的Filter只是利用这些信息,然后再根据从HostManager统计上来的HostState信息,选出匹配的Host。</p><p>request_spec中的第一个信息就是image的properties信息,尤其是当你想支持多种Hypervisor的时候,Xen的image, KVM的image, Hyper-V的image各不相同,如何保证image跑在正确的Hypervisor上?在image里面这种hypervisor_type property就很必要。</p><p>请阅读下面的文章</p><p><a title="http://www.cloudbase.it/filtering-glance-images-for-hyper-v/" href="http://www.cloudbase.it/filtering-glance-images-for-hyper-v/">http://www.cloudbase.it/filtering-glance-images-for-hyper-v/</a></p><p>image properties还会有min_ram, min_disk,只有内存和硬盘够大才可以。</p><p>Flavor里面可以设置extra_specs,这是一系列key-value值,在数据结构中,以instance_type变量实现,可以在里面输入这个Flavor除了资源需求的其他参数,从而在Filter的时候,可以使用。</p><p>host aggregates将所有的Host分成多个Group,当然不同的Group可以根据不同的属性Metadata划分,一种是高性能和低性能。</p><p>在Openstack文档中,这个例子很好的展示了host aggregates和Flavor extra_specs的配合使用</p><p><a title="http://docs.openstack.org/trunk/config-reference/content/section_compute-scheduler.html" href="http://docs.openstack.org/trunk/config-reference/content/section_compute-scheduler.html">http://docs.openstack.org/trunk/config-reference/content/section_compute-scheduler.html</a></p><p>Example: Specify compute hosts with SSDs</p><p>This example configures the Compute service to enable users to request nodes that have solid-state drives (SSDs). You create a <code>fast-io</code> host aggregate in the <code>nova</code> availability zone and you add the <code>ssd=true</code> key-value pair to the aggregate. Then, you add the <code>node1</code>, and <code>node2</code> compute nodes to it.</p>$ nova aggregate-create fast-io nova<br/>+----+---------+-------------------+-------+----------+<br/>| Id | Name | Availability Zone | Hosts | Metadata |<br/>+----+---------+-------------------+-------+----------+<br/>| 1 | fast-io | nova | | |<br/>+----+---------+-------------------+-------+----------+<br/><br/>$ nova aggregate-set-metadata 1 ssd=true<br/>+----+---------+-------------------+-------+-------------------+<br/>| Id | Name | Availability Zone | Hosts | Metadata |<br/>+----+---------+-------------------+-------+-------------------+<br/>| 1 | fast-io | nova | [] | {u'ssd': u'true'} |<br/>+----+---------+-------------------+-------+-------------------+<br/><br/>$ nova aggregate-add-host 1 node1<br/>+----+---------+-------------------+-----------+-------------------+<br/>| Id | Name | Availability Zone | Hosts | Metadata |<br/>+----+---------+-------------------+------------+-------------------+<br/>| 1 | fast-io | nova | [u'node1'] | {u'ssd': u'true'} |<br/>+----+---------+-------------------+------------+-------------------+<br/><br/>$ nova aggregate-add-host 1 node2<br/>+----+---------+-------------------+---------------------+-------------------+<br/>| Id | Name | Availability Zone | Hosts | Metadata |<br/>+----+---------+-------------------+----------------------+-------------------+<br/>| 1 | fast-io | nova | [u'node1', u'node2'] | {u'ssd': u'true'} |<br/>+----+---------+-------------------+----------------------+-------------------+<p>Use the <strong>nova flavor-create</strong> command to create the <em><code>ssd.large</code></em> flavor called with an ID of 6, 8 GB of RAM, 80 GB root disk, and four vCPUs.</p>$ nova flavor-create <em><code>ssd.large</code></em> <em><code>6</code></em> <em><code>8192</code></em> <em><code>80</code></em> <em><code>4</code></em><br/>+----+-----------+-----------+------+-----------+------+-------+-------------+-----------+-------------+<br/>| ID | Name | Memory_MB | Disk | Ephemeral | Swap | VCPUs | RXTX_Factor | Is_Public | extra_specs |<br/>+----+-----------+-----------+------+-----------+------+-------+-------------+-----------+-------------+<br/>| 6 | ssd.large | 8192 | 80 | 0 | | 4 | 1 | True | {} |<br/>+----+-----------+-----------+------+-----------+------+-------+-------------+-----------+-------------+<p>Once the flavor is created, specify one or more key-value pairs that match the key-value pairs on the host aggregates. In this case, that is the <em><code>ssd=true</code></em> key-value pair. Setting a key-value pair on a flavor is done using the <strong>nova flavor-key</strong> command.</p><p>$ nova flavor-key <em><code>ssd.large</code></em> set <em><code>ssd=true</code></em></p><p>Once it is set, you should see the <code>extra_specs</code> property of the <code>ssd.large</code> flavor populated with a key of <code>ssd</code> and a corresponding value of <code>true</code>.</p>$ nova flavor-show ssd.large<br/>+----------------------------+-------------------+<br/>| Property | Value |<br/>+----------------------------+-------------------+<br/>| OS-FLV-DISABLED:disabled | False |<br/>| OS-FLV-EXT-DATA:ephemeral | 0 |<br/>| disk | 80 |<br/>| extra_specs | {u'ssd': u'true'} |<br/>| id | 6 |<br/>| name | ssd.large |<br/>| os-flavor-access:is_public | True |<br/>| ram | 8192 |<br/>| rxtx_factor | 1.0 |<br/>| swap | |<br/>| vcpus | 4 |<br/>+----------------------------+-------------------+<p>Now, when a user requests an instance with the <code>ssd.large</code> flavor, the scheduler only considers hosts with the <code>ssd=true</code> key-value pair. In this example, these are <code>node1</code> and <code>node2</code>.</p><p>另一个作用是Xen和KVM的POOL分开,有利于XEN进行Live Migration</p><p>另一个作用是Windows和Linux的POOL分开,因为Windows是需要收费的,而Linux大多不需要,Windows的收费是按照物理机,而非虚拟机来收费的,所有需要尽量的将windows的虚拟机放到一个物理机上。</p><p>Filter_properties的里面scheduler_hints是一个json,里面可以设置任何值,用于Filter的时候使用。</p><p>例如JsonFilter</p><p>The JsonFilter allows a user to construct a custom filter by passing a scheduler hint in JSON format. The following operators are supported:</p><ul><li>=</li><li>&lt;</li><li>&gt;</li><li>in</li><li>&lt;=</li><li>&gt;=</li><li>not</li><li>or</li><li>and</li></ul><p>The filter supports the following variables:</p><ul><li>$free_ram_mb</li><li>$free_disk_mb</li><li>$total_usable_ram_mb</li><li>$vcpus_total</li><li>$vcpus_used</li></ul><p>Using the <strong>nova</strong> command-line tool, use the <code>--hint</code> flag:</p>$ nova boot --image 827d564a-e636-4fc4-a376-d36f7ebe1747 --flavor 1 --hint query='["&gt;=","$free_ram_mb",1024]' server1<p>With the API, use the <code>os:scheduler_hints</code> key:</p><p><img title="Select Text" src="http://docs.openstack.org/trunk/config-reference/content/images/icon_clipboard.png" alt="Select Text" width="20" height="20" align="right" /></p><p><code>{</code></p><p><code></code><code>"server":</code> <code>{</code></p><p><code></code><code>"name":</code> <code>"</code><code>s</code><code>e</code><code>r</code><code>v</code><code>e</code><code>r</code><code>-</code><code>1</code><code>"</code><code>,</code></p><p><code></code><code>"imageRef":</code> <code>"</code><code>c</code><code>e</code><code>d</code><code>e</code><code>f</code><code>4</code><code>0</code><code>a</code><code>-</code><code>e</code><code>d</code><code>6</code><code>7</code><code>-</code><code>4</code><code>d</code><code>1</code><code>0</code><code>-</code><code>8</code><code>0</code><code>0</code><code>e</code><code>-</code><code>1</code><code>7</code><code>4</code><code>5</code><code>5</code><code>e</code><code>d</code><code>c</code><code>e</code><code>1</code><code>7</code><code>5</code><code>"</code><code>,</code></p><p><code></code><code>"flavorRef":</code> <code>"</code><code>1</code><code>"</code></p><p><code></code><code>}</code><code>,</code></p><p><code></code><code>"os:scheduler_hints":</code> <code>{</code></p><p><code></code><code>"query":</code> <code>"</code><code>[</code><code>&amp;</code><code>g</code><code>t</code><code>;</code><code>=</code><code>,</code><code>$</code><code>f</code><code>r</code><code>e</code><code>e</code><code>_</code><code>r</code><code>a</code><code>m</code><code>_</code><code>m</code><code>b</code><code>,</code><code>1</code><code>0</code><code>2</code><code>4</code><code>]</code><code>"</code></p><p><code></code><code>}</code></p><p><code>}</code></p><p><code>我们可以指定某个物理机,用下面的命令--availability-zone &lt;zone-name&gt;:&lt;host-name&gt; </code></p><p><strong><span style="font-family: 宋体;"><code>步骤14:对合适的Hosts进行weighting并且排序</code></span></strong></p><p><code>选出了Hosts,接下来就是进行Weighting的操作</code></p><p><code>Weighting可以根据很多变量来,一般来说Memory和disk是最先需要满足的,CPU和network io则需要次要考虑,一般来说,对于付钱较少的Flavor,能满足memory和disk就可以了,对于付钱较多的Flavor,则需要保证其CPU和network io.</code></p><p><strong>步骤15:nova-scheduler想选出的Host发送RPC</strong></p><img src="http://counter.cnblogs.com/blog/rss/3930350" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/popsuper1982/p/3930350.html" target="_blank">别以为真懂Openstack: 虚拟机创建的50个步骤和100个知识点(2)</a>,转载请注明。</p>http://www.cnblogs.com/lzkwin/p/xdomain-proxy.html使用Web代理实现Ajax跨域 - 李中凯目前的工作项目分为前端和后台,双方事先约定接口,之后独立开发。后台每天开发完后在测试服务器上部署,前端连接测试服务器进行数据交互。前端和后台分开的好处是代码不用混在一个工程里一起build,互不干涉。但由此也引发出一个问题,那就是Ajax跨域。目前的项目是一个Single Page App, 基本上...2014-08-22T14:13:00Z2014-08-22T14:13:00Z李中凯http://www.cnblogs.com/lzkwin/<p>目前的工作项目分为前端和后台,双方事先约定接口,之后独立开发。后台每天开发完后在测试服务器上部署,前端连接测试服务器进行数据交互。前端和后台分开的好处是代码不用混在一个工程里一起build,互不干涉。但由此也引发出一个问题,那就是Ajax跨域。目前的项目是一个Single Page App, 基本上所有数据交互都是通过Ajax请求来完成的。为了方便平时前端开发,必须解决跨域问题。</p><p>跨域方案有多种,我认为基本上可分为两大类,一类是需要目标Server配合的,另一类则不需要。前者限制稍多,必须由服务器显式允许跨域才行,比如返回HTTP头信息,修改服务器配置,返回JavaScript等。可以用JSONP,iframe等方式实现。后者主动权就掌握在跨域客户端,服务器不用为此做任何配置。这就是本文要说的Web代理。</p><p>作为前端开发,自然希望主动权在自己手里,不用劳烦Sever配合。所以我选择了使用Web代理方案。该方案的原理其实很简单,就是将跨域请求转变为同源请求。具体来说,就是在本地搭建一个Web站点(代理),该站点可以向目标服务器发送HTTP请求,接收响应。它的行为跟浏览器类似,因此目标服务器是不用区分对待的。然后将本地的前端站点也部署到这个Web站点中,这样它们就属于同一个域了。所有针对目标服务器的Ajax请求,都发送到这个代理,然后由代理负责转发和接收响应。这样就避开了跨域之名,却有跨域之实。</p><!--break--><p><img src="http://www.zoneky.com/blog/image/Cross-domain-proxy.png" alt="setting" /></p><p>图有点丑,欢迎拍砖。</p><p>剩下的就是实现细节了。由于对Asp.Net比较熟悉,就用它创建一个Web站点。在实现过程中,我觉得有几点细节需要关注一下。</p><h4 id="http-request">HTTP Request拦截</strong></p><p>由于事先不知道会有哪些请求(就算知道,请求的URL可能也会太多),不可能针对每个URL写一个转发规则。因此需要获取所有Ajax请求,经过统一的处理再转发到目标服务器。众所周知,Asp.Net的IHttpHandler接口定义了针对某个具体的HTTP请求的处理方法,如前所述,不可能为每个请求URL写一个Handler,有没有一种办法可以获取任何请求的信息?答案是HttpModule。HttpModule是先于HttpHandler处理的,所以在这里可以做些手脚。定义一个HttpModule也很简单,只要实现IHttpModule接口,监听Request事件就可以了。当然,需要在<code>Web.config</code>文件里注册这个HttpModule才能使用。</p><p><code>XDomainProxy.cs</code>定义:</p><code>public class XDomainProxy : IHttpModule<br/>{<br/> public void Dispose() { }<br/> public void Init(HttpApplication context)<br/> {<br/> context.BeginRequest += new EventHandler(Application_BeginRequest);<br/> context.EndRequest += new EventHandler(Application_EndRequest);<br/> }<br/> public void Application_BeginRequest(object sender, EventArgs e)<br/> {<br/><br/> }<br/> public void Application_EndRequest(object sender, EventArgs e)<br/> {<br/> HttpApplication application = sender as HttpApplication;<br/> HttpContext context = application.Context;<br/> HttpResponse response = context.Response;<br/> response.StatusCode = 200;<br/> }<br/>}</code><p>在<code>Web.config</code>中注册<code>HttpModule</code>:</p><code>&lt;configuration&gt;<br/>&lt;system.web&gt;<br/>&lt;httpHandlers&gt;<br/> ... <br/>&lt;/httpHandlers&gt;<br/> &lt;httpModules&gt;<br/> &lt;add name=&quot;ProxyModule&quot; type=&quot;AAProxy.Proxy&quot;/&gt;<br/> &lt;/httpModules&gt;<br/>&lt;/system.web&gt;<br/>&lt;/configuration&gt;</code><h4 id="http">模拟HTTP请求</strong></p><p>.Net 已经封装好了<code>HttpWebRequest</code>和<code>HttpWebResponse</code>两个关键的类,非常方便。</p><code>//实例化web访问类<br/>HttpWebRequest request = (HttpWebRequest)WebRequest.Create(requestUrl);<br/>request.Method = context.Request.HttpMethod;<br/>request.ContentType = context.Request.ContentType;<br/><br/>string postData = context.Request.Form.ToString();<br/>byte[] postdatabytes = Encoding.UTF8.GetBytes(postData);<br/>request.ContentLength = postdatabytes.Length;<br/>request.AllowAutoRedirect = false;<br/>request.CookieContainer = loginCookie;<br/>request.KeepAlive = true;<br/>request.UserAgent = context.Request.UserAgent;<br/><br/>if (context.Request.HttpMethod == &quot;POST&quot;)<br/>{<br/> Stream stream;<br/> stream = request.GetRequestStream();<br/> stream.Write(postdatabytes, 0, postdatabytes.Length);<br/> stream.Close();<br/>}<br/><br/>//接收响应<br/>response = (HttpWebResponse)request.GetResponse();<br/> using (StreamReader reader = new StreamReader(response.GetResponseStream(), Encoding.UTF8))<br/>{<br/> string content = reader.ReadToEnd();<br/>}</code><p>这样就完成了一个简单的HTTP请求。注意POST请求需要另外发参数。</p><h4 id="cookie">Cookie</strong></p><p>由于所有的Ajax请求都需要用户登录才能进行,所以代理程序也必须模拟用户登录目标服务器站点。用户的登录信息是保存在cookie里的,所以在模拟请求的时候还需要保存cookie。<code>HttpWebRequest</code>有个<code>CookieContainer</code>属性,就是用来装cookie的。保存好cookie后,之后的每个请求都必须带上它,这样才能维持登录状态。另外,还需要把这些cookie写到浏览器中。这里需要注意,<code>.Net</code>封装了两个关于cookie的类,<code>HttpCookie</code>和<code>Cookie</code>。前者是Asp.Net程序写回给浏览器用的,而后者是向别的服务器发起HTTP请求时用的。所以代理程序收到目标Server返回的<code>Cookie</code>对象时,要转换成<code>HttpCookie</code>对象再返回给浏览器。</p><code>//response是目标服务器的响应对象,context是返回给浏览器的上下文对象<br/>void SetCookie(HttpWebResponse response, HttpContext context)<br/>{<br/> foreach (Cookie cookie in response.Cookies)<br/> {<br/> HttpCookie httpCookie = new HttpCookie(cookie.Name, cookie.Value);<br/> httpCookie.Domain = cookie.Domain;<br/> httpCookie.Expires = cookie.Expires;<br/> httpCookie.Path = cookie.Path;<br/> httpCookie.HttpOnly = cookie.HttpOnly;<br/> httpCookie.Secure = cookie.Secure;<br/> context.Response.SetCookie(httpCookie);<br/> }<br/>}</code><h4 id="https">Https连接</strong></p><p>在使用过程中发现,如果目标服务器的数字证书是不受信任的,连接将被拒绝。这在连接Server端开发同事的PC调试时不方便,怎么破?.Net对Http请求有个证书验证机制,只要让这个验证总是通过就好了。(一切为了开发方便)</p><code>ServicePointManager.ServerCertificateValidationCallback = new System.Net.Security.RemoteCertificateValidationCallback(ValidateRemoteCertificate);<br/><br/>private bool ValidateRemoteCertificate(object sender, System.Security.Cryptography.X509Certificates.X509Certificate certificate, System.Security.Cryptography.X509Certificates.X509Chain chain, System.Net.Security.SslPolicyErrors sslPolicyErrors)<br/>{<br/> return true;<br/>}</code><h4 id="web">部署Web代理</strong></p><p>部署就很简单了,新建一个IIS站点,根目录指向前端项目的路径,设定一个端口号。将Web代理发布到本地的某个目录下,然后作为一个应用程序添加到之前的IIS站点中即可。</p><p><strong>总结</strong></p><p>经过以上几个关键步骤,基本上搭建好了Web代理。当然,这个过程中还有一些细节需要关注,比如转发请求的URL映射处理,Session过期,异常处理等。总的来说,没有用到什么高深的技术,只是针对各种跨域方案做了一个选择。</p><p><a href="http://www.zoneky.com/blog/2014/08/22/xdomain-using-proxy/">原文:使用Web代理实现Ajax跨域</a></p><img src="http://counter.cnblogs.com/blog/rss/3930326" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/lzkwin/p/xdomain-proxy.html" target="_blank">使用Web代理实现Ajax跨域</a>,转载请注明。</p>http://www.cnblogs.com/biwork/p/3930331.html微软BI 之SSRS 系列 - 实现 Excel 中图表结合的报表设计 - BIWORK来自群里面讨论的一个问题,EXCEL 中有类似于这样的图形,上面是 Chart, Chart X轴上的值正好就是下方 Table 的列头,这个在 SSRS 中应该如何实现?SSRS 2008、2008RS,2012 中实际上没有这种对应的控件,我们通常想到的方式可能是上方一个单独的 Chart 图,...2014-08-22T14:13:00Z2014-08-22T14:13:00ZBIWORKhttp://www.cnblogs.com/biwork/<div><span style="font-family: Microsoft YaHei; font-size: 13px;">来自群里面讨论的一个问题,EXCEL 中有类似于这样的图形,上面是 Chart, Chart X轴上的值正好就是下方 Table 的列头,这个在 SSRS 中应该如何实现?</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222051006121234.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">SSRS 2008、2008RS,2012 中实际上没有这种对应的控件,我们通常想到的方式可能是上方一个单独的 Chart 图,下方一个 Table 然后合并在一起。但是这样会存在一些问题,因为 Chart 轴的值不是固定的,会随着聚合值的增加而扩展。并且对于表中的列头来说也没有办法完全能够和 Chart X 轴上的坐标对应在一条竖线上,一定会有偏差。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">在 SSRS 中,我们可以通过一些技巧来实现类似于这样的效果。下面是我用 Column Chart 来展现的这种类似于 Excel 中的效果,虽然实现比较接近了,但我并不推荐这么做! 因为我们完全可以通过合理的变通,并使用其它的方式将这些数据展现的更加合理!&nbsp;&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222051165349731.png" alt="" />&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">原始测试数据 (AdventureWorksDW2012)就是一个简单的查询 -&nbsp;</span></div><div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="font-family: Microsoft YaHei; font-size: 13px;"><span style="color: #0000ff;">SELECT</span><span style="color: #000000;"> DPC.EnglishProductCategoryName,<br/> FIS.OrderDate,<br/> FIS.OrderDateKey,<br/> DD.CalendarYear,<br/> FIS.SalesOrderNumber,<br/> FIS.SalesOrderLineNumber,<br/> FIS.SalesAmount </span><span style="color: #0000ff;">AS</span><span style="color: #000000;"> Internet_Sales<br/></span><span style="color: #0000ff;">FROM</span> FactInternetSales <span style="color: #0000ff;">AS</span><span style="color: #000000;"> FIS<br/></span><span style="color: #0000ff;">INNER</span> <span style="color: #808080;">JOIN</span> DimProduct <span style="color: #0000ff;">AS</span><span style="color: #000000;"> DP<br/></span><span style="color: #0000ff;">ON</span> FIS.ProductKey <span style="color: #808080;">=</span><span style="color: #000000;"> DP.ProductKey<br/></span><span style="color: #0000ff;">INNER</span> <span style="color: #808080;">JOIN</span> DimProductSubcategory <span style="color: #0000ff;">AS</span><span style="color: #000000;"> DPS<br/></span><span style="color: #0000ff;">ON</span> DP.ProductSubcategoryKey <span style="color: #808080;">=</span><span style="color: #000000;"> DPS.ProductSubcategoryKey<br/></span><span style="color: #0000ff;">INNER</span> <span style="color: #808080;">JOIN</span> DimProductCategory <span style="color: #0000ff;">AS</span><span style="color: #000000;"> DPC<br/></span><span style="color: #0000ff;">ON</span> DPS.ProductCategoryKey <span style="color: #808080;">=</span><span style="color: #000000;"> DPC.ProductCategoryKey<br/></span><span style="color: #0000ff;">INNER</span> <span style="color: #808080;">JOIN</span> DimDate <span style="color: #0000ff;">AS</span><span style="color: #000000;"> DD<br/></span><span style="color: #0000ff;">ON</span> FIS.OrderDate <span style="color: #808080;">=</span> DD.FullDateAlternateKey</span></div><p><span style="font-family: Microsoft YaHei; font-size: 13px;">新建报表和数据集 Dataset,选择 Matrix 行分组为 Product Category ,列分组为 Calendar year。</span></p></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222052391439436.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">添加两个新行 - 选择 Outside Group - Above。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222052495961230.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">并删除第一行。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222053052686184.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">删除的时候只删除行,但是不要连组给删除了。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222053114094260.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">删除之后的样子就是这样的。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222053172685250.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">第二行是表头要显示的年份,第一行用作行内 Column Chart 图。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222053274878271.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">在第一行的这个位置插入 Data Bar,选择 Stacked Column。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222053420341783.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">填充 Stacked Column 中的聚合值,分组以及系列。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222053482849358.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">填充完毕后就可以看到大致的效果了,每一列都是从 Matrix 列分组展开的,而每一列当中的 Column Chart 的分组又依赖于当前的列分组中的年份。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222053585033380.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">自行做一些美化工作,包括格式化等等。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222054041129827.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">存在一个问题,不同的系列是出来了,但是无法区分各个系列与产品分类的关系。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">思路是 - Chart 图中的颜色模板使用自定义的,然后使用自定义函数获取指定的颜色,这几种颜色需要和自定义模板中的颜色是一样的。</span></div><div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="font-family: Microsoft YaHei; font-size: 13px;"><span style="color: #0000ff;">Private</span> colorPalette <span style="color: #0000ff;">As</span> <span style="color: #0000ff;">String</span>() = {<span style="color: #800000;">"</span><span style="color: #800000;">SeaGreen</span><span style="color: #800000;">"</span>, <span style="color: #800000;">"</span><span style="color: #800000;">Orange</span><span style="color: #800000;">"</span>, <span style="color: #800000;">"</span><span style="color: #800000;">RoyalBlue</span><span style="color: #800000;">"</span><span style="color: #000000;">}<br/> </span><span style="color: #0000ff;">Private</span> count <span style="color: #0000ff;">As</span> <span style="color: #0000ff;">Integer</span> = <span style="color: #800080;">0</span><br/> <span style="color: #0000ff;">Private</span> mapping <span style="color: #0000ff;">As</span> <span style="color: #0000ff;">New</span><span style="color: #000000;"> System.Collections.Hashtable()<br/><br/> </span><span style="color: #0000ff;">Public</span> <span style="color: #0000ff;">Function</span> GetColor(<span style="color: #0000ff;">ByVal</span> groupingValue <span style="color: #0000ff;">As</span> <span style="color: #0000ff;">String</span>) <span style="color: #0000ff;">As</span> <span style="color: #0000ff;">String</span><br/> <span style="color: #0000ff;">If</span> mapping.ContainsKey(groupingValue) <span style="color: #0000ff;">Then</span><br/> <span style="color: #0000ff;">Return</span><span style="color: #000000;"> mapping(groupingValue)<br/> </span><span style="color: #0000ff;">End</span> <span style="color: #0000ff;">If</span><br/><br/> <span style="color: #0000ff;">Dim</span> c <span style="color: #0000ff;">As</span> <span style="color: #0000ff;">String</span> = colorPalette(count <span style="color: #0000ff;">Mod</span><span style="color: #000000;"> colorPalette.Length)<br/> count </span>= count + <span style="color: #800080;">1</span><span style="color: #000000;"><br/> mapping.Add(groupingValue, c)<br/> </span><span style="color: #0000ff;">Return</span><span style="color: #000000;"> c<br/> </span><span style="color: #0000ff;">End Function</span></span></div><p><span style="font-family: Microsoft YaHei; font-size: 13px;">报表的自定义函数。</span></p></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222054101907118.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">设置 Stacked Column 图的自定义模板颜色。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222054167531022.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">选中 English Product Category 文本框,设置它的背景颜色,调用函数。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">=Code.GetColor(Fields!EnglishProductCategoryName.Value)</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222103153938480.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">最后,要做出这种效果还需要进一步的格式化,调整边框等操作使得这个 Stacked Column 像一个整体,而非单一的个体,这些美化的细节就不一一描述了。可以参考 <a href="http://blog.tianshansoft.com/archives/1795.html" target="_blank">天善学院微软SSRS2012报表课程</a>&nbsp;基础案例中的 - 案例6,案例13,案例18,案例24 以及 SSRS 报表设计提高篇中的报表美化与设计等案例。所以本文只讲解实现的流程,过于细节的设计技巧,包括一些聚合分组的概念,自定义函数等不是本文的重点。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;"><img src="http://images.cnitblog.com/blog/477275/201408/222054231745585.png" alt="" /></span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">对于文章开头看到的这种 Line Chart 形式可不可以做,也可以做,可以通过调整 Chart 区域折线的平面宽度使得几个独立的折线区域也连接成一个整体。但是像最左侧的 Y 轴线如何实现,几个产品不使用底线区分而要求使用 Chart 中自带的系列又该显示且不重复?这些都不是拖拖拽拽就可以实现的,这需要熟练掌握 SSRS 报表中的各种知识点和技能点,不同控件的组合使用技巧等。当然,如果就这个案例来说,假设实际开发中如果是我,我会拒绝设计这种样子的报表,因为上图很明显在阅读性和数据呈现上完全体现不出来像 Excel 展现出来的那种效果。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">所以,在实际报表开发过程中,不能为了模仿而模仿,因为很有可能模仿成一个四不像。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">吐吐口水吧! 在实际项目中,在引导客户改变已有使用习惯,适应新的适应习惯的过程中往往确实是伴着口水,Fighting 的,包括遭受到客户对自身能力的怀疑,这些在项目开发中都是很正常的。像有很多客户往往会提出他们认为很好实现但实则很难实现的一些效果要求必须实现,这些极端情况也是客观存在的。对于我来说,我也会遇到一些变态的和不合理的需求,但通常情况下我也会直接拒绝,坦诚告诉客户无法实现,因为它已经违背了 SSRS 自身特性。但同时我也会尝试拿出另多的替代的解决方案让客户来选择,这样客户通常会比较容易接受和妥协。就像出门从北京到上海出差,告诉他们飞机确实因为技术原因就是不能飞,在客户非要证明但又无法证明飞机就是能起飞的同时给他们选择高铁,汽车和自行车三选一,一般情况下客户还是会从中选择他最能够接受的一种最优方案,因为谁也耗不起时间。</span></div><div><span style="font-family: Microsoft YaHei; font-size: 13px;">&nbsp;</span></div><img src="http://counter.cnblogs.com/blog/rss/3930331" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/biwork/p/3930331.html" target="_blank">微软BI 之SSRS 系列 - 实现 Excel 中图表结合的报表设计</a>,转载请注明。</p>http://www.cnblogs.com/dolphin0520/p/3920373.htmlJava并发编程:volatile关键字解析 - 海 子Java并发编程:volatile关键字解析 volatile这个关键字可能很多朋友都听说过,或许也都用过。在Java 5之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果。在Java 5之后,volatile关键字才得以重获生机。 volatile关键字虽然从字面上理...2014-08-22T14:07:00Z2014-08-22T14:07:00Z海 子http://www.cnblogs.com/dolphin0520/<p style="text-align: center;">Java并发编程:volatile关键字解析</p><p style="text-align: left;">   volatile这个关键字可能很多朋友都听说过,或许也都用过。在Java 5之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果。在Java 5之后,volatile关键字才得以重获生机。</p><p style="text-align: left;">  volatile关键字虽然从字面上理解起来比较简单,但是要用好不是一件容易的事情。由于volatile关键字是与Java的内存模型有关的,因此在讲述volatile关键之前,我们先来了解一下与内存模型相关的概念和知识,然后分析了volatile关键字的实现原理,最后给出了几个使用volatile关键字的场景。</p><p style="text-align: left;">  以下是本文的目录大纲:</p><p style="text-align: left;">  一.内存模型的相关概念</p><p style="text-align: left;">  二.并发编程中的三个概念</p><p style="text-align: left;">  三.Java内存模型</p><p style="text-align: left;">  四..深入剖析volatile关键字</p><p style="text-align: left;">  五.使用volatile关键字的场景</p><p style="text-align: left;">  若有不正之处请多多谅解,并欢迎批评指正。</p><p style="text-align: left;">  请尊重作者劳动成果,转载请标明原文链接:</p><p style="text-align: left;">  http://www.cnblogs.com/dolphin0520/p/3920373.html</p><h2 style="text-align: left;">一.内存模型的相关概念</strong></p><p style="text-align: left;">  大家都知道,计算机在执行程序时,每条指令都是在CPU中执行的,而执行指令过程中,势必涉及到数据的读取和写入。由于程序运行过程中的临时数据是存放在主存(物理内存)当中的,这时就存在一个问题,由于CPU执行速度很快,而从内存读取数据和向内存写入数据的过程跟CPU执行指令的速度比起来要慢的多,因此如果任何时候对数据的操作都要通过和内存的交互来进行,会大大降低指令执行的速度。因此在CPU里面就有了高速缓存。</p><p style="text-align: left;">  也就是,当程序在运行过程中,会将运算需要的数据从主存复制一份到CPU的高速缓存当中,那么CPU进行计算时就可以直接从它的高速缓存读取数据和向其中写入数据,当运算结束之后,再将高速缓存中的数据刷新到主存当中。举个简单的例子,比如下面的这段代码:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">i = i + 1;<br/></div><p>&nbsp;  当线程执行这个语句时,会先从主存当中读取i的值,然后复制一份到高速缓存当中,然后CPU执行指令对i进行加1操作,然后将数据写入高速缓存,最后将高速缓存中i最新的值刷新到主存当中。</p><p>  这个代码在单线程中运行是没有任何问题的,但是在多线程中运行就会有问题了。在多核CPU中,每条线程可能运行于不同的CPU中,因此每个线程运行时有自己的高速缓存(对单核CPU来说,其实也会出现这种问题,只不过是以线程调度的形式来分别执行的)。本文我们以多核CPU为例。</p><p>  比如同时有2个线程执行这段代码,假如初始时i的值为0,那么我们希望两个线程执行完之后i的值变为2。但是事实会是这样吗?</p><p>  可能存在下面一种情况:初始时,两个线程分别读取i的值存入各自所在的CPU的高速缓存当中,然后线程1进行加1操作,然后把i的最新值1写入到内存。此时线程2的高速缓存当中i的值还是0,进行加1操作之后,i的值为1,然后线程2把i的值写入内存。</p><p>  最终结果i的值是1,而不是2。这就是著名的缓存一致性问题。通常称这种被多个线程访问的变量为共享变量。</p><p>  也就是说,如果一个变量在多个CPU中都存在缓存(一般在多线程编程时才会出现),那么就可能存在缓存不一致的问题。</p><p>  为了解决缓存不一致性问题,通常来说有以下2种解决方法:</p><p>  1)通过在总线加LOCK#锁的方式</p><p>  2)通过缓存一致性协议</p><p>  这2种方式都是硬件层面上提供的方式。</p><p>  在早期的CPU当中,是通过在总线上加LOCK#锁的形式来解决缓存不一致的问题。因为CPU和其他部件进行通信都是通过总线来进行的,如果对总线加LOCK#锁的话,也就是说阻塞了其他CPU对其他部件访问(如内存),从而使得只能有一个CPU能使用这个变量的内存。比如上面例子中 如果一个线程在执行 i = i +1,如果在执行这段代码的过程中,在总线上发出了LCOK#锁的信号,那么只有等待这段代码完全执行完毕之后,其他CPU才能从变量i所在的内存读取变量,然后进行相应的操作。这样就解决了缓存不一致的问题。</p><p>  但是上面的方式会有一个问题,由于在锁住总线期间,其他CPU无法访问内存,导致效率低下。</p><p>  所以就出现了缓存一致性协议。最出名的就是Intel 的MESI协议,MESI协议保证了每个缓存中使用的共享变量的副本是一致的。它核心的思想是:当CPU写数据时,如果发现操作的变量是共享变量,即在其他CPU中也存在该变量的副本,会发出信号通知其他CPU将该变量的缓存行置为无效状态,因此当其他CPU需要读取这个变量时,发现自己缓存中缓存该变量的缓存行是无效的,那么它就会从内存重新读取。</p><p><img style="margin-right: auto; margin-left: auto; display: block;" src="http://images.cnitblog.com/blog/288799/201408/212219343783699.jpg" alt="" /></p><p><strong>二.并发编程中的三个概念</strong></p><p>  在并发编程中,我们通常会遇到以下三个问题:原子性问题,可见性问题,有序性问题。我们先看具体看一下这三个概念:</p><p><strong>1.原子性</strong></p><p>  原子性:即一个操作或者多个操作 要么全部执行并且执行的过程不会被任何因素打断,要么就都不执行。</p><p>  一个很经典的例子就是银行账户转账问题:</p><p>  比如从账户A向账户B转1000元,那么必然包括2个操作:从账户A减去1000元,往账户B加上1000元。</p><p>  试想一下,如果这2个操作不具备原子性,会造成什么样的后果。假如从账户A减去1000元之后,操作突然中止。然后又从B取出了500元,取出500元之后,再执行 往账户B加上1000元 的操作。这样就会导致账户A虽然减去了1000元,但是账户B没有收到这个转过来的1000元。</p><p>  所以这2个操作必须要具备原子性才能保证不出现一些意外的问题。</p><p>  同样地反映到并发编程中会出现什么结果呢?</p><p>  举个最简单的例子,大家想一下假如为一个32位的变量赋值过程不具备原子性的话,会发生什么后果?</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">i = 9;<br/></div><p>&nbsp;  假若一个线程执行到这个语句时,我暂且假设为一个32位的变量赋值包括两个过程:为低16位赋值,为高16位赋值。</p><p>  那么就可能发生一种情况:当将低16位数值写入之后,突然被中断,而此时又有一个线程去读取i的值,那么读取到的就是错误的数据。</p><p><strong>2.可见性</strong></p><p>  可见性是指当多个线程访问同一个变量时,一个线程修改了这个变量的值,其他线程能够立即看得到修改的值。</p><p>  举个简单的例子,看下面这段代码:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">//线程1执行的代码<br/>int i = 0;<br/>i = 10;<br/><br/>//线程2执行的代码<br/>j = i;<br/></div><p>&nbsp;  假若执行线程1的是CPU1,执行线程2的是CPU2。由上面的分析可知,当线程1执行 i =10这句时,会先把i的初始值加载到CPU1的高速缓存中,然后赋值为10,那么在CPU1的高速缓存当中i的值变为10了,却没有立即写入到主存当中。</p><p>  此时线程2执行 j = i,它会先去主存读取i的值并加载到CPU2的缓存当中,注意此时内存当中i的值还是0,那么就会使得j的值为0,而不是10.</p><p>  这就是可见性问题,线程1对变量i修改了之后,线程2没有立即看到线程1修改的值。</p><p><strong>3.有序性</strong></p><p>  有序性:即程序执行的顺序按照代码的先后顺序执行。举个简单的例子,看下面这段代码:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">int i = 0; <br/>boolean flag = false;<br/>i = 1; //语句1 <br/>flag = true; //语句2<br/></div><p>&nbsp;  上面代码定义了一个int型变量,定义了一个boolean类型变量,然后分别对两个变量进行赋值操作。从代码顺序上看,语句1是在语句2前面的,那么JVM在真正执行这段代码的时候会保证语句1一定会在语句2前面执行吗?不一定,为什么呢?这里可能会发生指令重排序(Instruction Reorder)。</p><p>  下面解释一下什么是指令重排序,一般来说,处理器为了提高程序运行效率,可能会对输入代码进行优化,它不保证程序中各个语句的执行先后顺序同代码中的顺序一致,但是它会保证程序最终执行结果和代码顺序执行的结果是一致的。</p><p>  比如上面的代码中,语句1和语句2谁先执行对最终的程序结果并没有影响,那么就有可能在执行过程中,语句2先执行而语句1后执行。</p><p>  但是要注意,虽然处理器会对指令进行重排序,但是它会保证程序最终结果会和代码顺序执行结果相同,那么它靠什么保证的呢?再看下面一个例子:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">int a = 10; //语句1<br/>int r = 2; //语句2<br/>a = a + 3; //语句3<br/>r = a*a; //语句4<br/></div><p>&nbsp;  这段代码有4个语句,那么可能的一个执行顺序是:</p><p>  <img src="http://images.cnitblog.com/blog/288799/201408/212305263939989.jpg" alt="" /></p><p>  </p><p>  那么可不可能是这个执行顺序呢: 语句2&nbsp;&nbsp; 语句1&nbsp;&nbsp;&nbsp; 语句4&nbsp;&nbsp; 语句3</p><p>  不可能,因为处理器在进行重排序时是会考虑指令之间的数据依赖性,如果一个指令Instruction 2必须用到Instruction 1的结果,那么处理器会保证Instruction 1会在Instruction 2之前执行。</p><p>  虽然重排序不会影响单个线程内程序执行的结果,但是多线程呢?下面看一个例子:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">//线程1:<br/>context = loadContext(); //语句1<br/>inited = true; //语句2<br/><br/>//线程2:<br/>while(!inited ){<br/> sleep() <br/>}<br/>doSomethingwithconfig(context);<br/></div><p>&nbsp;  上面代码中,由于语句1和语句2没有数据依赖性,因此可能会被重排序。假如发生了重排序,在线程1执行过程中先执行语句2,而此是线程2会以为初始化工作已经完成,那么就会跳出while循环,去执行doSomethingwithconfig(context)方法,而此时context并没有被初始化,就会导致程序出错。</p><p>&nbsp;  从上面可以看出,指令重排序不会影响单个线程的执行,但是会影响到线程并发执行的正确性。</p><p>  也就是说,要想并发程序正确地执行,必须要保证原子性、可见性以及有序性。只要有一个没有被保证,就有可能会导致程序运行不正确。</p><h2 style="text-align: left;">三.Java内存模型</strong></p><p style="text-align: left;">  在前面谈到了一些关于内存模型以及并发编程中可能会出现的一些问题。下面我们来看一下Java内存模型,研究一下Java内存模型为我们提供了哪些保证以及在java中提供了哪些方法和机制来让我们在进行多线程编程时能够保证程序执行的正确性。</p><p style="text-align: left;">  在Java虚拟机规范中试图定义一种Java内存模型(Java Memory Model,JMM)来屏蔽各个硬件平台和操作系统的内存访问差异,以实现让Java程序在各种平台下都能达到一致的内存访问效果。那么Java内存模型规定了哪些东西呢,它定义了程序中变量的访问规则,往大一点说是定义了程序执行的次序。注意,为了获得较好的执行性能,Java内存模型并没有限制执行引擎使用处理器的寄存器或者高速缓存来提升指令执行速度,也没有限制编译器对指令进行重排序。也就是说,在java内存模型中,也会存在缓存一致性问题和指令重排序的问题。</p><p style="text-align: left;">  Java内存模型规定所有的变量都是存在主存当中(类似于前面说的物理内存),每个线程都有自己的工作内存(类似于前面的高速缓存)。线程对变量的所有操作都必须在工作内存中进行,而不能直接对主存进行操作。并且每个线程不能访问其他线程的工作内存。</p><p style="text-align: left;">  举个简单的例子:在java中,执行下面这个语句:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">i = 10;<br/></div><p>&nbsp;  执行线程必须先在自己的工作线程中对变量i所在的缓存行进行赋值操作,然后再写入主存当中。而不是直接将数值10写入主存当中。</p><p>  那么Java语言 本身对 原子性、可见性以及有序性提供了哪些保证呢?</p><p><strong>1.原子性</strong></p><p>  在Java中,对基本数据类型的变量的读取和赋值操作是原子性操作,即这些操作是不可被中断的,要么执行,要么不执行。</p><p>  上面一句话虽然看起来简单,但是理解起来并不是那么容易。看下面一个例子i:</p><p>  请分析以下哪些操作是原子性操作:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">x = 10; //语句1<br/>y = x; //语句2<br/>x++; //语句3<br/>x = x + 1; //语句4<br/></div><p>&nbsp;  咋一看,有些朋友可能会说上面的4个语句中的操作都是原子性操作。其实只有语句1是原子性操作,其他三个语句都不是原子性操作。</p><p>  语句1是直接将数值10赋值给x,也就是说线程执行这个语句的会直接将数值10写入到工作内存中。</p><p>  语句2实际上包含2个操作,它先要去读取x的值,再将x的值写入工作内存,虽然读取x的值以及 将x的值写入工作内存 这2个操作都是原子性操作,但是合起来就不是原子性操作了。</p><p>  同样的,x++和 x = x+1包括3个操作:读取x的值,进行加1操作,写入新的值。</p><p>&nbsp;  所以上面4个语句只有语句1的操作具备原子性。</p><p>  也就是说,只有简单的读取、赋值(而且必须是将数字赋值给某个变量,变量之间的相互赋值不是原子操作)才是原子操作。</p><p>  不过这里有一点需要注意:在32位平台下,对64位数据的读取和赋值是需要通过两个操作来完成的,不能保证其原子性。但是好像在最新的JDK中,JVM已经保证对64位数据的读取和赋值也是原子性操作了。</p><p>  从上面可以看出,Java内存模型只保证了基本读取和赋值是原子性操作,如果要实现更大范围操作的原子性,可以通过synchronized和Lock来实现。由于synchronized和Lock能够保证任一时刻只有一个线程执行该代码块,那么自然就不存在原子性问题了,从而保证了原子性。</p><p><strong>2.可见性</strong></p><p>  对于可见性,Java提供了volatile关键字来保证可见性。</p><p>  当一个共享变量被volatile修饰时,它会保证修改的值会立即被更新到主存,当有其他线程需要读取时,它会去内存中读取新值。</p><p>  而普通的共享变量不能保证可见性,因为普通共享变量被修改之后,什么时候被写入主存是不确定的,当其他线程去读取时,此时内存中可能还是原来的旧值,因此无法保证可见性。</p><p>  另外,通过synchronized和Lock也能够保证可见性,synchronized和Lock能保证同一时刻只有一个线程获取锁然后执行同步代码,并且在释放锁之前会将对变量的修改刷新到主存当中。因此可以保证可见性。</p><p><strong>3.有序性</strong></p><p>  在Java内存模型中,允许编译器和处理器对指令进行重排序,但是重排序过程不会影响到单线程程序的执行,却会影响到多线程并发执行的正确性。</p><p>  在Java里面,可以通过volatile关键字来保证一定的&ldquo;有序性&rdquo;(具体原理在下一节讲述)。另外可以通过synchronized和Lock来保证有序性,很显然,synchronized和Lock保证每个时刻是有一个线程执行同步代码,相当于是让线程顺序执行同步代码,自然就保证了有序性。</p><p>  另外,<span style="color: #ff0000;">Java内存模型具备一些先天的&ldquo;有序性&rdquo;,即不需要通过任何手段就能够得到保证的有序性,这个通常也称为 happens-before 原则。如果两个操作的执行次序无法从happens-before原则推导出来,那么它们就不能保证它们的有序性,虚拟机可以随意地对它们进行重排序。</span></p><p>  下面就来具体介绍下happens-before原则(先行发生原则):</p><ul><li style="text-align: left;">程序次序规则:一个线程内,按照代码顺序,书写在前面的操作先行发生于书写在后面的操作</li><li style="text-align: left;">锁定规则:一个unLock操作先行发生于后面对同一个锁额lock操作</li><li style="text-align: left;">volatile变量规则:对一个变量的写操作先行发生于后面对这个变量的读操作</li><li style="text-align: left;">传递规则:如果操作A先行发生于操作B,而操作B又先行发生于操作C,则可以得出操作A先行发生于操作C</li><li style="text-align: left;">线程启动规则:Thread对象的start()方法先行发生于此线程的每个一个动作</li><li style="text-align: left;">线程中断规则:对线程interrupt()方法的调用先行发生于被中断线程的代码检测到中断事件的发生</li><li style="text-align: left;">线程终结规则:线程中所有的操作都先行发生于线程的终止检测,我们可以通过Thread.join()方法结束、Thread.isAlive()的返回值手段检测到线程已经终止执行</li><li style="text-align: left;">对象终结规则:一个对象的初始化完成先行发生于他的finalize()方法的开始</li></ul><p style="text-align: left;">  这8条原则摘自《深入理解Java虚拟机》。</p><p style="text-align: left;">  这8条规则中,前4条规则是比较重要的,后4条规则都是显而易见的。</p><p style="text-align: left;">  下面我们来解释一下前4条规则:</p><p style="text-align: left;">  对于程序次序规则来说,我的理解就是一段程序代码的执行在单个线程中看起来是有序的。注意,虽然这条规则中提到&ldquo;书写在前面的操作先行发生于书写在后面的操作&rdquo;,这个应该是程序看起来执行的顺序是按照代码顺序执行的,因为虚拟机可能会对程序代码进行指令重排序。虽然进行重排序,但是最终执行的结果是与程序顺序执行的结果一致的,它只会对不存在数据依赖性的指令进行重排序。因此,在单个线程中,程序执行看起来是有序执行的,这一点要注意理解。事实上,这个规则是用来保证程序在单线程中执行结果的正确性,但无法保证程序在多线程中执行的正确性。</p><p style="text-align: left;">  第二条规则也比较容易理解,也就是说无论在单线程中还是多线程中,同一个锁如果出于被锁定的状态,那么必须先对锁进行了释放操作,后面才能继续进行lock操作。</p><p style="text-align: left;">  第三条规则是一条比较重要的规则,也是后文将要重点讲述的内容。直观地解释就是,如果一个线程先去写一个变量,然后一个线程去进行读取,那么写入操作肯定会先行发生于读操作。</p><p style="text-align: left;">  第四条规则实际上就是体现happens-before原则具备传递性。</p><h2 style="text-align: left;">四.深入剖析volatile关键字</strong></p><p style="text-align: left;">  在前面讲述了很多东西,其实都是为讲述volatile关键字作铺垫,那么接下来我们就进入主题。</p><p style="text-align: left;"><strong>1.volatile关键字的两层语义</strong></p><p style="text-align: left;">  一旦一个共享变量(类的成员变量、类的静态成员变量)被volatile修饰之后,那么就具备了两层语义:</p><p style="text-align: left;">  1)保证了不同线程对这个变量进行操作时的可见性,即一个线程修改了某个变量的值,这新值对其他线程来说是立即可见的。</p><p style="text-align: left;">  2)禁止进行指令重排序。</p><p style="text-align: left;">  先看一段代码,假如线程1先执行,线程2后执行:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">//线程1<br/>boolean stop = false;<br/>while(!stop){<br/>doSomething();<br/>}<br/><br/>//线程2<br/>stop = true;<br/></div><p>&nbsp;  这段代码是很典型的一段代码,很多人在中断线程时可能都会采用这种标记办法。但是事实上,这段代码会完全运行正确么?即一定会将线程中断么?不一定,也许在大多数时候,这个代码能够把线程中断,但是也有可能会导致无法中断线程(虽然这个可能性很小,但是只要一旦发生这种情况就会造成死循环了)。</p><p>  下面解释一下这段代码为何有可能导致无法中断线程。在前面已经解释过,每个线程在运行过程中都有自己的工作内存,那么线程1在运行的时候,会将stop变量的值拷贝一份放在自己的工作内存当中。</p><p>  那么当线程2更改了stop变量的值之后,但是还没来得及写入主存当中,线程2转去做其他事情了,那么线程1由于不知道线程2对stop变量的更改,因此还会一直循环下去。</p><p>  但是用volatile修饰之后就变得不一样了:</p><p>  第一:使用volatile关键字会强制将修改的值立即写入主存;</p><p>  第二:使用volatile关键字的话,当线程2进行修改时,会导致线程1的工作内存中缓存变量stop的缓存行无效(反映到硬件层的话,就是CPU的L1或者L2缓存中对应的缓存行无效);</p><p>  第三:由于线程1的工作内存中缓存变量stop的缓存行无效,所以线程1再次读取变量stop的值时会去主存读取。</p><p>  那么在线程2修改stop值时(当然这里包括2个操作,修改线程2工作内存中的值,然后将修改后的值写入内存),会使得线程1的工作内存中缓存变量stop的缓存行无效,然后线程1读取时,发现自己的缓存行无效,它会等待缓存行对应的主存地址被更新之后,然后去对应的主存读取最新的值。</p><p>  那么线程1读取到的就是最新的正确的值。</p><p style="text-align: left;"><strong>2.volatile保证原子性吗?</strong></p><p style="text-align: left;">  从上面知道volatile关键字保证了操作的可见性,但是volatile能保证对变量的操作是原子性吗?</p><p style="text-align: left;">  下面看一个例子:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">public class Test {<br/>public volatile int inc = 0;<br/><br/>public void increase() {<br/>inc++;<br/>}<br/><br/>public static void main(String[] args) {<br/>final Test test = new Test();<br/>for(int i=0;i&lt;10;i++){<br/>new Thread(){<br/>public void run() {<br/>for(int j=0;j&lt;1000;j++)<br/>test.increase();<br/>};<br/>}.start();<br/>}<br/><br/>while(Thread.activeCount()&gt;1) //保证前面的线程都执行完<br/>Thread.yield();<br/>System.out.println(test.inc);<br/>}<br/>}<br/></div><p>&nbsp;  大家想一下这段程序的输出结果是多少?也许有些朋友认为是10000。但是事实上运行它会发现每次运行结果都不一致,都是一个小于10000的数字。</p><p>  可能有的朋友就会有疑问,不对啊,上面是对变量inc进行自增操作,由于volatile保证了可见性,那么在每个线程中对inc自增完之后,在其他线程中都能看到修改后的值啊,所以有10个线程分别进行了1000次操作,那么最终inc的值应该是1000*10=10000。</p><p>  这里面就有一个误区了,volatile关键字能保证可见性没有错,但是上面的程序错在没能保证原子性。可见性只能保证每次读取的是最新的值,但是volatile没办法保证对变量的操作的原子性。</p><p>  在前面已经提到过,自增操作是不具备原子性的,它包括读取变量的原始值、进行加1操作、写入工作内存。那么就是说自增操作的三个子操作可能会分割开执行,就有可能导致下面这种情况出现:</p><p>  假如某个时刻变量inc的值为10,</p><p>  线程1对变量进行自增操作,线程1先读取了变量inc的原始值,然后线程1被阻塞了;</p><p>  然后线程2对变量进行自增操作,线程2也去读取变量inc的原始值,由于线程1只是对变量inc进行读取操作,而没有对变量进行修改操作,所以不会导致线程2的工作内存中缓存变量inc的缓存行无效,所以线程2会直接去主存读取inc的值,发现inc的值时10,然后进行加1操作,并把11写入工作内存,最后写入主存。</p><p>  然后线程1接着进行加1操作,由于已经读取了inc的值,注意此时在线程1的工作内存中inc的值仍然为10,所以线程1对inc进行加1操作后inc的值为11,然后将11写入工作内存,最后写入主存。</p><p>  那么两个线程分别进行了一次自增操作后,inc只增加了1。</p><p>  解释到这里,可能有朋友会有疑问,不对啊,前面不是保证一个变量在修改volatile变量时,会让缓存行无效吗?然后其他线程去读就会读到新的值,对,这个没错。这个就是上面的happens-before规则中的volatile变量规则,但是要注意,线程1对变量进行读取操作之后,被阻塞了的话,并没有对inc值进行修改。然后虽然volatile能保证线程2对变量inc的值读取是从内存中读取的,但是线程1没有进行修改,所以线程2根本就不会看到修改的值。</p><p>  根源就在这里,自增操作不是原子性操作,而且volatile也无法保证对变量的任何操作都是原子性的。</p><p>  把上面的代码改成以下任何一种都可以达到效果:</p><p>  采用synchronized:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;" onclick="cnblogs_code_show('376be879-56eb-4886-94b0-f89c5496c1d1')"><div id="cnblogs_code_open_376be879-56eb-4886-94b0-f89c5496c1d1" class="cnblogs_code_hide"><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">class</span><span style="color: #000000;"> Test {<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">int</span> inc = 0<span style="color: #000000;">;<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">synchronized</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> increase() {<br/> inc</span>++<span style="color: #000000;">;<br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {<br/> </span><span style="color: #0000ff;">final</span> Test test = <span style="color: #0000ff;">new</span><span style="color: #000000;"> Test();<br/> </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;10;i++<span style="color: #000000;">){<br/> </span><span style="color: #0000ff;">new</span><span style="color: #000000;"> Thread(){<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> run() {<br/> </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> j=0;j&lt;1000;j++<span style="color: #000000;">)<br/> test.increase();<br/> };<br/> }.start();<br/> }<br/> <br/> </span><span style="color: #0000ff;">while</span>(Thread.activeCount()&gt;1) <span style="color: #008000;">//</span><span style="color: #008000;">保证前面的线程都执行完</span><br/><span style="color: #000000;"> Thread.yield();<br/> System.out.println(test.inc);<br/> }<br/>}</span></div></div><p>  采用Lock:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;" onclick="cnblogs_code_show('7a877386-237d-4c2b-b28d-ff760550abc7')"><div id="cnblogs_code_open_7a877386-237d-4c2b-b28d-ff760550abc7" class="cnblogs_code_hide"><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">class</span><span style="color: #000000;"> Test {<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">int</span> inc = 0<span style="color: #000000;">;<br/> Lock lock </span>= <span style="color: #0000ff;">new</span><span style="color: #000000;"> ReentrantLock();<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> increase() {<br/> lock.lock();<br/> </span><span style="color: #0000ff;">try</span><span style="color: #000000;"> {<br/> inc</span>++<span style="color: #000000;">;<br/> } </span><span style="color: #0000ff;">finally</span><span style="color: #000000;">{<br/> lock.unlock();<br/> }<br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {<br/> </span><span style="color: #0000ff;">final</span> Test test = <span style="color: #0000ff;">new</span><span style="color: #000000;"> Test();<br/> </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;10;i++<span style="color: #000000;">){<br/> </span><span style="color: #0000ff;">new</span><span style="color: #000000;"> Thread(){<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> run() {<br/> </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> j=0;j&lt;1000;j++<span style="color: #000000;">)<br/> test.increase();<br/> };<br/> }.start();<br/> }<br/> <br/> </span><span style="color: #0000ff;">while</span>(Thread.activeCount()&gt;1) <span style="color: #008000;">//</span><span style="color: #008000;">保证前面的线程都执行完</span><br/><span style="color: #000000;"> Thread.yield();<br/> System.out.println(test.inc);<br/> }<br/>}</span></div></div><p>  采用AtomicInteger:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;" onclick="cnblogs_code_show('6858213b-e442-4d04-86bb-6e2ed73bfa52')"><div id="cnblogs_code_open_6858213b-e442-4d04-86bb-6e2ed73bfa52" class="cnblogs_code_hide"><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">class</span><span style="color: #000000;"> Test {<br/> </span><span style="color: #0000ff;">public</span> AtomicInteger inc = <span style="color: #0000ff;">new</span><span style="color: #000000;"> AtomicInteger();<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> increase() {<br/> inc.getAndIncrement();<br/> }<br/> <br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {<br/> </span><span style="color: #0000ff;">final</span> Test test = <span style="color: #0000ff;">new</span><span style="color: #000000;"> Test();<br/> </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;10;i++<span style="color: #000000;">){<br/> </span><span style="color: #0000ff;">new</span><span style="color: #000000;"> Thread(){<br/> </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> run() {<br/> </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> j=0;j&lt;1000;j++<span style="color: #000000;">)<br/> test.increase();<br/> };<br/> }.start();<br/> }<br/> <br/> </span><span style="color: #0000ff;">while</span>(Thread.activeCount()&gt;1) <span style="color: #008000;">//</span><span style="color: #008000;">保证前面的线程都执行完</span><br/><span style="color: #000000;"> Thread.yield();<br/> System.out.println(test.inc);<br/> }<br/>}</span></div></div><p>  在java 1.5的java.util.concurrent.atomic包下提供了一些原子操作类,即对基本数据类型的 自增(加1操作),自减(减1操作)、以及加法操作(加一个数),减法操作(减一个数)进行了封装,保证这些操作是原子性操作。atomic是利用CAS来实现原子性操作的(Compare And Swap),CAS实际上是利用处理器提供的CMPXCHG指令实现的,而处理器执行CMPXCHG指令是一个原子性操作。</p><p style="text-align: left;"><strong>3.volatile能保证有序性吗?</strong></p><p style="text-align: left;">  在前面提到volatile关键字能禁止指令重排序,所以volatile能在一定程度上保证有序性。</p><p style="text-align: left;">  volatile关键字禁止指令重排序有两层意思:</p><p style="text-align: left;">  1)当程序执行到volatile变量的读操作或者写操作时,在其前面的操作的更改肯定全部已经进行,且结果已经对后面的操作可见;在其后面的操作肯定还没有进行;</p><p style="text-align: left;">  2)在进行指令优化时,不能将在对volatile变量访问的语句放在其后面执行,也不能把volatile变量后面的语句放到其前面执行。</p><p style="text-align: left;">  可能上面说的比较绕,举个简单的例子:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">//x、y为非volatile变量<br/>//flag为volatile变量<br/><br/>x = 2; //语句1<br/>y = 0; //语句2<br/>flag = true; //语句3<br/>x = 4; //语句4<br/>y = -1; //语句5<br/></div><p>&nbsp;  由于flag变量为volatile变量,那么在进行指令重排序的过程的时候,不会将语句3放到语句1、语句2前面,也不会讲语句3放到语句4、语句5后面。但是要注意语句1和语句2的顺序、语句4和语句5的顺序是不作任何保证的。</p><p>  并且volatile关键字能保证,执行到语句3时,语句1和语句2必定是执行完毕了的,且语句1和语句2的执行结果对语句3、语句4、语句5是可见的。</p><p>  那么我们回到前面举的一个例子:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">//线程1:<br/>context = loadContext(); //语句1<br/>inited = true; //语句2<br/><br/>//线程2:<br/>while(!inited ){<br/> sleep() <br/>}<br/>doSomethingwithconfig(context);<br/></div><p>&nbsp;  前面举这个例子的时候,提到有可能语句2会在语句1之前执行,那么久可能导致context还没被初始化,而线程2中就使用未初始化的context去进行操作,导致程序出错。</p><p>  这里如果用volatile关键字对inited变量进行修饰,就不会出现这种问题了,因为当执行到语句2时,必定能保证context已经初始化完毕。</p><p style="text-align: left;"><strong>4.volatile的原理和实现机制</strong></p><p style="text-align: left;">  前面讲述了源于volatile关键字的一些使用,下面我们来探讨一下volatile到底如何保证可见性和禁止指令重排序的。</p><p style="text-align: left;">  下面这段话摘自《深入理解Java虚拟机》:</p><p style="text-align: left;">  &ldquo;观察加入volatile关键字和没有加入volatile关键字时所生成的汇编代码发现,加入volatile关键字时,会多出一个lock前缀指令&rdquo;</p><p style="text-align: left;">  lock前缀指令实际上相当于一个内存屏障(也成内存栅栏),内存屏障会提供3个功能:</p><p style="text-align: left;">  1)它确保指令重排序时不会把其后面的指令排到内存屏障之前的位置,也不会把前面的指令排到内存屏障的后面;即在执行到内存屏障这句指令时,在它前面的操作已经全部完成;</p><p style="text-align: left;">  2)它会强制将对缓存的修改操作立即写入主存;</p><p style="text-align: left;">  3)如果是写操作,它会导致其他CPU中对应的缓存行无效。</p><h2 style="text-align: left;">五.使用volatile关键字的场景</strong></p><p style="text-align: left;">  synchronized关键字是防止多个线程同时执行一段代码,那么就会很影响程序执行效率,而volatile关键字在某些情况下性能要优于synchronized,但是要注意volatile关键字是无法替代volatile关键字的,因为volatile关键字无法保证操作的原子性。通常来说,使用volatile必须具备以下2个条件:</p><p style="text-align: left;">  1)对变量的写操作不依赖于当前值</p><p style="text-align: left;">  2)该变量没有包含在具有其他变量的不变式中</p><p style="text-align: left;">  实际上,这些条件表明,可以被写入 volatile 变量的这些有效值独立于任何程序的状态,包括变量的当前状态。</p><p style="text-align: left;">  事实上,我的理解就是上面的2个条件需要保证操作是原子性操作,才能保证使用volatile关键字的程序在并发时能够正确执行。</p><p style="text-align: left;">  下面列举几个Java中使用volatile的几个场景。</p><p style="text-align: left;"><strong>1.状态标记量</strong></p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">volatile boolean flag = false;<br/><br/>while(!flag){<br/>doSomething();<br/>}<br/><br/>public void setFlag() {<br/>flag = true;<br/>}<br/></div><p>&nbsp;</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">volatile boolean inited = false;<br/>//线程1:<br/>context = loadContext(); <br/>inited = true; <br/><br/>//线程2:<br/>while(!inited ){<br/>sleep() <br/>}<br/>doSomethingwithconfig(context);<br/></div><p>&nbsp;</p><p><strong>2.double check</strong></p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">class Singleton{<br/>private volatile static Singleton instance = null;<br/><br/>private Singleton() {<br/><br/>}<br/><br/>public static Singleton getInstance() {<br/>if(instance==null) {<br/>synchronized (Singleton.class) {<br/>if(instance==null)<br/>instance = new Singleton();<br/>}<br/>}<br/>return instance;<br/>}<br/>}<br/></div><p>&nbsp;  至于为何需要这么写请参考:</p><p>  《Java 中的双重检查(Double-Check)》<a href="http://blog.csdn.net/dl88250/article/details/5439024" target="_blank">http://blog.csdn.net/dl88250/article/details/5439024</a></p><p style="text-align: left;">  和<a href="http://www.iteye.com/topic/652440">http://www.iteye.com/topic/652440</a></p><p style="text-align: left;">  参考资料:</p><p style="text-align: left;">  《Java编程思想》</p><p style="text-align: left;">  《深入理解Java虚拟机》</p><p style="text-align: left;">  <a href="http://jiangzhengjun.iteye.com/blog/652532" target="_blank">http://jiangzhengjun.iteye.com/blog/652532</a></p><p style="text-align: left;">  <a href="http://blog.sina.com.cn/s/blog_7bee8dd50101fu8n.html" target="_blank">http://blog.sina.com.cn/s/blog_7bee8dd50101fu8n.html</a></p><p style="text-align: left;">  <a href="http://ifeve.com/volatile/" target="_blank">http://ifeve.com/volatile/</a></p><p style="text-align: left;">  <a href="http://blog.csdn.net/ccit0519/article/details/11241403" target="_blank">http://blog.csdn.net/ccit0519/article/details/11241403</a></p><p style="text-align: left;">  <a href="http://blog.csdn.net/ns_code/article/details/17101369" target="_blank">http://blog.csdn.net/ns_code/article/details/17101369</a></p><p style="text-align: left;">  <a href="http://www.cnblogs.com/kevinwu/archive/2012/05/02/2479464.html" target="_blank">http://www.cnblogs.com/kevinwu/archive/2012/05/02/2479464.html</a></p><p style="text-align: left;">  <a href="http://www.cppblog.com/elva/archive/2011/01/21/139019.html" target="_blank">http://www.cppblog.com/elva/archive/2011/01/21/139019.html</a></p><p style="text-align: left;">  <a href="http://ifeve.com/volatile-array-visiblity/" target="_blank">http://ifeve.com/volatile-array-visiblity/</a></p><p style="text-align: left;">  <a href="http://www.bdqn.cn/news/201312/12579.shtml" target="_blank">http://www.bdqn.cn/news/201312/12579.shtml</a></p><p style="text-align: left;">  <a href="http://exploer.blog.51cto.com/7123589/1193399" target="_blank">http://exploer.blog.51cto.com/7123589/1193399</a></p><p style="text-align: left;">  <a href="http://www.cnblogs.com/Mainz/p/3556430.html" target="_blank">http://www.cnblogs.com/Mainz/p/3556430.html</a></p><img src="http://counter.cnblogs.com/blog/rss/3920373" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/dolphin0520/p/3920373.html" target="_blank">Java并发编程:volatile关键字解析</a>,转载请注明。</p>http://www.cnblogs.com/jianyus/p/3923316.htmlOffice 365 - SharePoint 2013 Online 之应用程序开发 - 霖雨1、给站点添加完Napa后,在网站内容里点击Napa,如下图: 2、创建一个新的app,如下图: 3、可以在Napa里添加新的项目,如下图: 4、添加新的文件,可以添加web页面、样式表、脚本,如下图: 5、可以设置Napa,如下图: 6、设置用Visual C#语言,这样vs打开可以...2014-08-22T13:51:00Z2014-08-22T13:51:00Z霖雨http://www.cnblogs.com/jianyus/<p>  1、给站点添加完Napa后,在网站内容里点击Napa,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224448937257.png"><img style="display: inline; border: 0px;" title="clip_image001" src="http://images.cnitblog.com/blog/256729/201408/192224458933430.png" alt="clip_image001" width="700" height="545" border="0" /></a></p><p>  2、创建一个新的app,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224465496031.png"><img style="display: inline; border: 0px;" title="clip_image002" src="http://images.cnitblog.com/blog/256729/201408/192224479243018.png" alt="clip_image002" width="700" height="377" border="0" /></a></p><p>  3、可以在Napa里添加新的项目,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224485037659.png"><img style="display: inline; border: 0px;" title="clip_image003" src="http://images.cnitblog.com/blog/256729/201408/192224491126246.png" alt="clip_image003" width="700" height="366" border="0" /></a></p><p>  4、添加新的文件,可以添加web页面、样式表、脚本,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224496749818.png"><img style="display: inline; border: 0px;" title="clip_image004" src="http://images.cnitblog.com/blog/256729/201408/192224503782732.png" alt="clip_image004" width="700" height="388" border="0" /></a></p><p>  5、可以设置Napa,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224508932291.png"><img style="display: inline; border: 0px;" title="clip_image005" src="http://images.cnitblog.com/blog/256729/201408/192224512218091.png" alt="clip_image005" width="250" height="133" border="0" /></a></p><p>  6、设置用Visual C#语言,这样vs打开可以用C#,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224518463449.png"><img style="display: inline; border: 0px;" title="clip_image006" src="http://images.cnitblog.com/blog/256729/201408/192224524566334.png" alt="clip_image006" width="700" height="428" border="0" /></a></p><p>  7、可以点击左侧菜单,Open in Visual Studio,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224529564420.png"><img style="display: inline; border: 0px;" title="clip_image007" src="http://images.cnitblog.com/blog/256729/201408/192224535818777.png" alt="clip_image007" width="700" height="306" border="0" /></a></p><p>  8、弹出菜单,选择Visual C#,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224543785421.png"><img style="display: inline; border: 0px;" title="clip_image008" src="http://images.cnitblog.com/blog/256729/201408/192224549717237.png" alt="clip_image008" width="471" height="352" border="0" /></a></p><p>  9、可能会弹出菜单,安装Web Platform 5.0,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224554563850.png"><img style="display: inline; border: 0px;" title="clip_image009" src="http://images.cnitblog.com/blog/256729/201408/192224562065479.png" alt="clip_image009" width="528" height="418" border="0" /></a></p><p>  10、Visual Studio打开应用程序,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224569093095.png"><img style="display: inline; border: 0px;" title="clip_image010" src="http://images.cnitblog.com/blog/256729/201408/192224580962025.png" alt="clip_image010" width="700" height="498" border="0" /></a></p><p>  11、需要登录Office 365,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224587847167.png"><img style="display: inline; border: 0px;" title="clip_image011" src="http://images.cnitblog.com/blog/256729/201408/192224593319267.png" alt="clip_image011" width="441" height="407" border="0" /></a></p><p>  12、在Visual Studio中打开,可以添加自定义列表,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192224599099610.png"><img style="display: inline; border: 0px;" title="clip_image012" src="http://images.cnitblog.com/blog/256729/201408/192225004879954.png" alt="clip_image012" width="700" height="515" border="0" /></a></p><p>  13、为列表实例添加测试数据,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192225010494527.png"><img style="display: inline; border: 0px;" title="clip_image013" src="http://images.cnitblog.com/blog/256729/201408/192225015341141.png" alt="clip_image013" width="700" height="473" border="0" /></a></p><p>  14、添加JS脚本,读取列表内容,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192225023151014.png"><img style="display: inline; border: 0px;" title="clip_image014" src="http://images.cnitblog.com/blog/256729/201408/192225030652643.png" alt="clip_image014" width="700" height="463" border="0" /></a></p><p>  15、在Default页面,添加HTML读取展示列表内容,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192225037215243.png"><img style="display: inline; border: 0px;" title="clip_image015" src="http://images.cnitblog.com/blog/256729/201408/192225042844114.png" alt="clip_image015" width="700" height="348" border="0" /></a></p><p>  16、可以部署应用程序,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192225047993673.png"><img style="display: inline; border: 0px;" title="clip_image016" src="http://images.cnitblog.com/blog/256729/201408/192225052532043.png" alt="clip_image016" width="612" height="308" border="0" /></a></p><p>  17、打开部署的开发人员站点,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192225061432417.png"><img style="display: inline; border: 0px;" title="clip_image017" src="http://images.cnitblog.com/blog/256729/201408/192225069408059.png" alt="clip_image017" width="700" height="585" border="0" /></a></p><p>  18、点击Custom Show,可以进入该应用程序,点击GetData获取数据,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192225078156960.png"><img style="display: inline; border: 0px;" title="clip_image018" src="http://images.cnitblog.com/blog/256729/201408/192225083785831.png" alt="clip_image018" width="700" height="477" border="0" /></a></p><p>  19、点击List可以进入列表,如下图:</p><p><a href="http://images.cnitblog.com/blog/256729/201408/192225089566174.png"><img style="display: inline; border: 0px;" title="clip_image019" src="http://images.cnitblog.com/blog/256729/201408/192225094872504.png" alt="clip_image019" width="700" height="506" border="0" /></a></p><p><strong>总 结</strong></p><p>  Office 365中使用VS打开App后,不会同步到Napa,所以在Napa中发布,会覆盖掉VS发布的,不过,SharePoint Online中开发App和Server中基本一样。</p><img src="http://counter.cnblogs.com/blog/rss/3923316" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/jianyus/p/3923316.html" target="_blank">Office 365 - SharePoint 2013 Online 之应用程序开发</a>,转载请注明。</p>http://www.cnblogs.com/lszwhb/p/3928050.html010-CALayer(图层) - ゴルツの惠斌纳閣下问题一:什么是CALayer(图层)?简而言之:使UIView可以显示在屏幕上的功能属性,UIView之所以可以显示在屏幕上完全是因为UIView内部含有一个CALayer属性|在iOS中,能看得见摸得着的东西基本上都是UIView,比如一个按钮、一个文本标签、一个文本输入框、一个图标等等,这些都是...2014-08-22T13:32:00Z2014-08-22T13:32:00Zゴルツの惠斌纳閣下http://www.cnblogs.com/lszwhb/<p><strong><span style="color: #0000ff;">问题一:什么是CALayer(图层)?</span></strong></p><p><span style="color: #0000ff;">简而言之:使UIView可以显示在屏幕上的功能属性,UIView之所以可以显示在屏幕上完全是因为UIView内部含有一个CALayer属性</span></p><div>&bull;在iOS中,能看得见摸得着的东西基本上都是UIView,比如一个按钮、一个文本标签、一个文本输入框、一个图标等等,这些都是UIView</div><div>&nbsp;</div><div>&bull;UIView本身不具备显示的功能,之所以能显示完全是因为它内部的图层属性(具有显示功能)</div><div>&nbsp;</div><div>&bull;在创建UIView对象时,UIView内部会自动创建一个图层(即CALayer对象),通过UIView的layer属性可以访问这个层</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property(nonatomic,readonly,retain) CALayer *layer;</div><div>&nbsp;</div><div>&bull;当UIView需要显示到屏幕上时,会调用drawRect:方法进行绘图,将所有内容绘制在自己的图层上,绘图完毕后,系统将图层映射到屏幕上,于是就完成了UIView的显示</div><div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008000;">//</span><span style="color: #008000;"> view的完整显示过程<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> 1. view.layer准备了一个Layer Graphics Context(图层类型的上下文)<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> 2. 调用view.layer.delegate(view)的drawLayer:inContext:,并传入刚才准备好的上下文<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> 3. view的drawLayer:inContext:方法内部又会调用view的drawRect:方法<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> 4. view就可以在drawRect:方法中实现绘图代码, 所有东西最终都绘制到view.layer上面<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> 5. 系统再将view.layer的内容拷贝到屏幕, 于是完成了view的显示</span></div></div><p><strong><span style="color: #0000ff;">问题二:CALayer具有哪些属性?</span></strong></p><p><span style="color: #0000ff;">九大属性:</span></p><p>&bull;1.内容(比如设置为图片CGImageRef)</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property(retain) id contents;</div><div><div>&bull;2.位置(默认指中点,具体由anchorPoint决定)</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CGPoint position; <span style="color: #008000;">// 用来设置CALayer在父层中的位置 以父层的左上角为原点(0, 0)</span></div>&bull;3.宽度和高度</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CGRect bounds;</div><div><span style="font-size: 14px; line-height: 1.5;">&bull;4.锚点(x,y的范围都是0-1),决定了自身视图在position的坐落点</span></div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CGPoint anchorPoint;&nbsp;<span style="color: #008000;">// 称为&ldquo;定位点&rdquo;、&ldquo;锚点&rdquo; 决定着CALayer身上的哪个点会在position属性所指的位置 以自己的左上角为原点(0, 0) 它的x、y取值范围都是0~1,默认值为(0.5, 0.5)</span></div><div>&bull;5.背景颜色(CGColorRef类型)</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CGColorRef backgroundColor;</div><div>&bull;6.形变属性</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CATransform3D transform;</div><div>&bull;7.边框颜色(CGColorRef类型)</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CGColorRef borderColor;</div><div>&bull;8.边框宽度</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CGFloat borderWidth;</div><div>&bull;9.圆角半径</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">@property CGColorRef borderColor;</div><p><span style="color: #0000ff; font-size: 1.17em; line-height: 1.5;">问题三:CALayer可以用来做什么呢?</span></p><div>&bull;通过操作CALayer对象,可以很方便地调整UIView的一些外观属性,比如:</div><div>&Oslash;阴影</div><div>&Oslash;圆角大小</div><div>&Oslash;边框宽度和颜色</div><div>&Oslash;&hellip; &hellip;</div><div>&Oslash;代码示例:显示效果:<img src="http://images.cnitblog.com/blog/513720/201408/212105047844146.png" alt="" width="83" height="79" /><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">- (<span style="color: #0000ff;">void</span><span style="color: #000000;">)testView<br/>{<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 边框宽度</span><br/> self.purpleView.layer.borderWidth = <span style="color: #800080;">10</span><span style="color: #000000;">;<br/> </span><span style="color: #008000;">//</span> <span style="color: #008000;">//</span><span style="color: #008000;"> 边框颜色</span><br/> self.purpleView.layer.borderColor =<span style="color: #000000;"> [UIColor blueColor].CGColor;<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 圆角</span><br/> self.purpleView.layer.cornerRadius = <span style="color: #800080;">10</span><span style="color: #000000;">;<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> self.purpleView.layer.masksToBounds = YES; // 超出主层边框范围的内容都剪掉<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 阴影颜色</span><br/> self.purpleView.layer.shadowColor =<span style="color: #000000;"> [UIColor blueColor].CGColor;<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 阴影偏差</span><br/> self.purpleView.layer.shadowOffset = CGSizeMake(<span style="color: #800080;">10</span>, <span style="color: #800080;">10</span><span style="color: #000000;">);<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 阴影不透明度</span><br/> self.purpleView.layer.shadowOpacity = <span style="color: #800080;">0.5</span><span style="color: #000000;">;<br/>}</span></div></div><div>&bull;还可以<span style="color: #ff0000;">给图层添加动画</span>,来<span style="color: #0000ff;">实现</span>一些比较<span style="color: #0000ff;">炫酷的效果(见问题六)</span></div><div><span style="color: #0000ff;">ps:新建</span><span style="color: #0000ff;"><span style="color: #0000ff;">图层</span></span><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;" onclick="cnblogs_code_show('0b8a024b-1bdf-4497-aeb6-2ee178d9249c')"><div id="cnblogs_code_open_0b8a024b-1bdf-4497-aeb6-2ee178d9249c" class="cnblogs_code_hide">- (<span style="color: #0000ff;">void</span><span style="color: #000000;">)viewDidLoad<br/>{<br/> [super viewDidLoad];<br/><br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 新建图层<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> CALayer *layer = [[CALayer alloc] init];</span><br/> CALayer *layer =<span style="color: #000000;"> [CALayer layer];<br/> layer.backgroundColor </span>=<span style="color: #000000;"> [UIColor redColor].CGColor;<br/> layer.bounds </span>= CGRectMake(<span style="color: #800080;">0</span>, <span style="color: #800080;">0</span>, <span style="color: #800080;">100</span>, <span style="color: #800080;">100</span><span style="color: #000000;">);<br/> layer.position </span>= CGPointMake(<span style="color: #800080;">200</span>, <span style="color: #800080;">200</span><span style="color: #000000;">);<br/> layer.anchorPoint </span>= CGPointMake(<span style="color: #800080;">1</span>, <span style="color: #800080;">1</span><span style="color: #000000;">);<br/> layer.cornerRadius </span>= <span style="color: #800080;">10</span><span style="color: #000000;">;<br/> layer.masksToBounds </span>=<span style="color: #000000;"> YES;<br/> layer.contents </span>= (<span style="color: #0000ff;">id</span>)[UIImage imageNamed:<span style="color: #800000;">@"</span><span style="color: #800000;">lufy</span><span style="color: #800000;">"</span><span style="color: #000000;">].CGImage;<br/> [self.view.layer addSublayer:layer];<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 新建图层<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> CALayer *layer1 = [[CALayer alloc] init];</span><br/> CALayer *layer1 =<span style="color: #000000;"> [CALayer layer];<br/> layer1.backgroundColor </span>=<span style="color: #000000;"> [UIColor redColor].CGColor;<br/> layer1.bounds </span>= CGRectMake(<span style="color: #800080;">0</span>, <span style="color: #800080;">0</span>, <span style="color: #800080;">100</span>, <span style="color: #800080;">100</span><span style="color: #000000;">);<br/> layer1.position </span>= CGPointMake(<span style="color: #800080;">200</span>, <span style="color: #800080;">200</span><span style="color: #000000;">);<br/> layer1.anchorPoint </span>= CGPointMake(<span style="color: #800080;">0</span>, <span style="color: #800080;">0</span><span style="color: #000000;">);<br/> layer1.cornerRadius </span>= <span style="color: #800080;">10</span><span style="color: #000000;">;<br/> layer1.masksToBounds </span>=<span style="color: #000000;"> YES;<br/> layer1.contents </span>= (<span style="color: #0000ff;">id</span>)[UIImage imageNamed:<span style="color: #800000;">@"</span><span style="color: #800000;">lufy</span><span style="color: #800000;">"</span><span style="color: #000000;">].CGImage;<br/> [self.view.layer addSublayer:layer1];<br/>}</span></div></div><p><span style="color: #0000ff;">ps:自定义图层</span></p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;" onclick="cnblogs_code_show('fcd72539-d8b8-4832-bda9-528b9841b54e')"><div id="cnblogs_code_open_fcd72539-d8b8-4832-bda9-528b9841b54e" class="cnblogs_code_hide"><span style="color: #0000ff;">@implementation</span><span style="color: #000000;"> WHBLayer<br/></span><span style="color: #008000;">/*</span><span style="color: #008000;">*<br/> * 只有明显地调用setNeedsDisplay方法,才会调用drawInContext:方法进行绘制<br/> </span><span style="color: #008000;">*/</span><br/>- (<span style="color: #0000ff;">void</span><span style="color: #000000;">)drawInContext:(CGContextRef)ctx<br/>{<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 红色</span><br/> CGContextSetRGBFillColor(ctx, <span style="color: #800080;">1</span>, <span style="color: #800080;">0</span>, <span style="color: #800080;">0</span>, <span style="color: #800080;">1</span><span style="color: #000000;">);<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 添加圆</span><br/> CGContextAddEllipseInRect(ctx, CGRectMake(<span style="color: #800080;">0</span>, <span style="color: #800080;">0</span>, <span style="color: #800080;">50</span>, <span style="color: #800080;">50</span><span style="color: #000000;">));<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> 实心绘制</span><br/><span style="color: #000000;"> CGContextFillPath(ctx);<br/>}</span></div></div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;" onclick="cnblogs_code_show('94724902-2ab1-40bf-8d05-2d26a34836b6')"><div id="cnblogs_code_open_94724902-2ab1-40bf-8d05-2d26a34836b6" class="cnblogs_code_hide">- (<span style="color: #0000ff;">void</span><span style="color: #000000;">)viewDidLoad<br/>{<br/> [super viewDidLoad];<br/> WHBLayer </span>*layer =<span style="color: #000000;"> [WHBLayer layer];<br/> layer.bounds </span>= CGRectMake(<span style="color: #800080;">0</span>, <span style="color: #800080;">0</span>, <span style="color: #800080;">100</span>, <span style="color: #800080;">100</span><span style="color: #000000;">);<br/> layer.backgroundColor </span>=<span style="color: #000000;"> [UIColor greenColor].CGColor;<br/> layer.anchorPoint </span>=<span style="color: #000000;"> CGPointZero;<br/> [layer setNeedsDisplay]; </span><span style="color: #008000;">//</span><span style="color: #008000;"> 绘图 会自动调用WHBLayer的drawInContext:(CGContextRef)ctf 方法</span><br/><span style="color: #000000;"> [self.view.layer addSublayer:layer];<br/>}</span></div></div></div><p><strong><span style="color: #0000ff;">问题四:为什么在CALayer中我们使用CGImageRef、CGColorRef两种数据类型来对其设置,而不用UIColor、UIImage???</span></strong></p><p><span style="color: #0000ff;">原因主要:考虑移植性,跨平台性</span></p><div><div>&bull;QuartzCore框架和CoreGraphics框架是可以跨平台使用的,在iOS和Mac OS X上都能使用&nbsp;</div><div>  &Oslash;CALayer是定义在QuartzCore框架中的</div><div>  &Oslash;CGImageRef、CGColorRef两种数据类型是定义在CoreGraphics框架中的</div><div>&bull;但是UIKit只能在iOS中使用</div><div>  &Oslash;UIColor、UIImage是定义在UIKit框架中的</div><div>&bull;为了保证可移植性,QuartzCore不能使用UIImage、UIColor,只能使用CGImageRef、CGColorRef</div></div><p><strong><span style="color: #0000ff;">&nbsp;问题五:既然UIView和CALayer均可在屏幕上显示图像,那又如何在二者之间做选择?</span></strong></p><div>&bull;通过CALayer,就能做出跟UIImageView一样的界面效果</div><div>&bull;既然CALayer和UIView都能实现相同的显示效果,那究竟该选择谁好呢?</div><div>&Oslash;对比CALayer,UIView多了一个事件处理的功能。也就是说,CALayer不能处理用户的触摸事件,而UIView可以</div><div>&Oslash;如果显示出来的东西<span style="color: #0000ff;">需要跟用户进行交互</span>的话,<span style="color: #0000ff;">用UIView</span>;如果<span style="color: #0000ff;">不需要跟用户进行交互</span>,用<span style="color: #0000ff;">UIView</span>或者<span style="color: #0000ff;">CALayer</span>都可以</div><div>&Oslash;当然,CALayer的性能会高一些,因为它少了事件处理的功能,更加轻量级</div><p><strong><span style="color: #0000ff;">问题六:如何利用CALayer来产生一些动画效果?什么是隐式动画?</span></strong></p><div>&bull;所有<span style="color: #0000ff;">手动创建的CALayer对象</span>,都存在着隐式动画</div><div>&bull;什么是隐式动画?</div><div>&Oslash;当对非Root Layer的部分属性进行修改时,默认会自动产生一些动画效果,称这些属性为Animatable Properties(可动画属性)</div><div>&bull;列举几个常见的Animatable Properties:</div><div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008000;">//</span><span style="color: #008000;"> bounds:用于设置CALayer的宽度和高度。修改这个属性会产生缩放动画<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> backgroundColor:用于设置CALayer的背景色。修改这个属性会产生背景色的渐变动画<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> position:用于设置CALayer的位置。修改这个属性会产生平移动画</span></div></div><div>&bull;可以通过动画事务(CATransaction)<span style="color: #800080;">关闭默认</span>的隐式动画效果</div><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #000000;">[CATransaction begin];<br/>[CATransaction setDisableActions:YES];<br/>self.myview.layer.position </span>= CGPointMake(<span style="color: #800080;">10</span>, <span style="color: #800080;">10</span><span style="color: #000000;">);<br/>[CATransaction commit];</span></div><p>&nbsp;代码示例:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;" onclick="cnblogs_code_show('5e37fb1d-8fc5-4adb-b9d0-5fb8bfc79abe')"><div id="cnblogs_code_open_5e37fb1d-8fc5-4adb-b9d0-5fb8bfc79abe" class="cnblogs_code_hide">- (<span style="color: #0000ff;">void</span><span style="color: #000000;">)viewDidLoad<br/>{<br/> [super viewDidLoad];<br/> </span><span style="color: #008000;">//</span><span style="color: #008000;"> Do any additional setup after loading the view, typically from a nib.</span><br/><span style="color: #000000;"> <br/> CALayer </span>*layer =<span style="color: #000000;"> [CALayer layer];<br/> layer.bounds </span>= CGRectMake(<span style="color: #800080;">0</span>, <span style="color: #800080;">0</span>, <span style="color: #800080;">100</span>, <span style="color: #800080;">100</span><span style="color: #000000;">);<br/> layer.backgroundColor </span>=<span style="color: #000000;"> [UIColor redColor].CGColor;<br/> layer.position </span>=<span style="color: #000000;"> CGPointZero;<br/> layer.anchorPoint </span>=<span style="color: #000000;"> CGPointZero;<br/> [self.view.layer addSublayer:layer];<br/> self.layer </span>=<span style="color: #000000;"> layer;<br/>}<br/><br/></span>- (<span style="color: #0000ff;">void</span>)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)<span style="color: #0000ff;">event</span><span style="color: #000000;"><br/>{<br/> self.layer.backgroundColor </span>=<span style="color: #000000;"> [UIColor blueColor].CGColor;<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> [CATransaction begin]; </span><span style="color: #008000;">//</span><span style="color: #008000;"> 开启事务<br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> [CATransaction setDisableActions:YES];</span><br/><span style="color: #000000;"> <br/> self.layer.position </span>= CGPointMake(<span style="color: #800080;">100</span>, <span style="color: #800080;">100</span><span style="color: #000000;">);<br/> self.layer.opacity </span>= <span style="color: #800080;">0.1</span><span style="color: #000000;">;<br/> <br/></span><span style="color: #008000;">//</span><span style="color: #008000;"> [CATransaction commit]; </span><span style="color: #008000;">//</span><span style="color: #008000;"> 提交事务</span><br/>}</div></div><p>&nbsp;</p><div>&nbsp;</div><div>&nbsp;</div><img src="http://counter.cnblogs.com/blog/rss/3928050" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/lszwhb/p/3928050.html" target="_blank">010-CALayer(图层)</a>,转载请注明。</p>http://www.cnblogs.com/MigCoder/p/3930264.htmlAngularJs ng-repeat 必须注意的性能问题 - 灰机_不会飞AngularJs 的 ng-repeat 让我们非常方便的遍历数组生成 Dom 元素,但是使用不当也会有性能问题。在项目中我们使用 ng-repeat 加载完一个列表后,如果再次请求数据,然后过滤列表,代码可能会这么写: 请求新数据 {{use...2014-08-22T13:18:00Z2014-08-22T13:18:00Z灰机_不会飞http://www.cnblogs.com/MigCoder/<p>AngularJs 的 ng-repeat 让我们非常方便的遍历数组生成 Dom 元素,但是使用不当也会有性能问题。</p><p>在项目中我们使用 ng-repeat 加载完一个列表后,如果再次请求数据,然后过滤列表,代码可能会这么写:</p><code>&lt;div ng-controller=&quot;Test&quot;&gt;<br/> &lt;button ng-click=&quot;request()&quot;&gt;请求新数据&lt;/button&gt;<br/> &lt;div ng-repeat=&quot;user in users&quot;&gt;<br/> {{user.name}}<br/> &lt;/div&gt;<br/>&lt;/div&gt;</code><p>Controller 的代码:</p><code>app.controller('Test', function($scope) {<br/> var users = [];<br/> for (var i = 0; i &lt; 100; i++) {<br/> users[i] = {<br/> id: i,<br/> name: &quot;User: &quot; + i<br/> };<br/> }<br/> $scope.users = users;<br/><br/> $scope.request = function () {<br/> // 从服务器加载新数据<br/> var result = [];<br/><br/> // 直接重新赋值给 users<br/> $scope.users = result;<br/> };<br/>});</code><p>查看 ng-repeat 的源码可以发现,当 ng-repeat 的数组被替换时, 它默认并不会重新利用已有的 Dom 元素,而是直接将其全部删除并重新生成新的数组 Dom 元素:</p><code>// 将上次生成的所有 dom 移除<br/>for (key in lastBlockMap) {<br/> if (lastBlockMap.hasOwnProperty(key)) {<br/> block = lastBlockMap[key];<br/> elementsToRemove = getBlockElements(block.clone);<br/> $animate.leave(elementsToRemove);<br/> forEach(elementsToRemove, function(element) { element[NG_REMOVED] = true; });<br/> block.scope.$destroy();<br/> }<br/>}</code><p>Dom 的频繁操作是非常不友好的,为什么 ng-repeat 不能利用已有的 dom 元素去更新数据呢?因为你没有把数组元素的标识属性告诉它,那么两次替换的时候它就没办法追踪了,我们可以看到 ng-repeat 往数组里每个元素加了一个 $$hashKey 的属性:</p><p><img src="http://images.cnitblog.com/blog/433262/201408/222116005656478.jpg" alt="key" /></p><p>这个 key 是由 Angular 内部的 nextUid() 方法生成,类似数据库自增,但是是使用字符串。</p><p>现在我们明白了,因为每次替换数组都会导致 ng-repeat 为每个元素生成一个新 key, 所以根本没办法重用已有的 Dom 元素,那么我们可以使用下边的语法来避免这个问题:</p><code>&lt;div ng-controller=&quot;Test&quot;&gt;<br/> &lt;button ng-click=&quot;request()&quot;&gt;请求新数据&lt;/button&gt;<br/>// 使用 track by 标识<br/> &lt;div ng-repeat=&quot;user in users track by user.id&quot;&gt;<br/> {{user.name}}<br/> &lt;/div&gt;<br/>&lt;/div&gt;</code><p>这样 ng-repeat 就用将其缓存起来啦,当然可能你的数组元素没有一个标识属性,如果元素数量不多那么可以接受,不然还是建议你手动为其生成一个标识属性。</p><img src="http://counter.cnblogs.com/blog/rss/3930264" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/MigCoder/p/3930264.html" target="_blank">AngularJs ng-repeat 必须注意的性能问题</a>,转载请注明。</p>http://www.cnblogs.com/reach296/p/3930252.html[性能分析]端口限制 - reach2961、 什么是协议端口?在Internet上,各主机间通过TCP/TP协议发送和接收数据报,各个数据报根据其目的主机的ip地址来进行互联网络中的路由选择。可见,把数据报顺利的传送到目的主机是没有问题的。问题出在哪里呢?我们知道大多数操作系统都支持多程序(进程)同时运行,那么目的主机应该把接收到的数据报...2014-08-22T13:08:00Z2014-08-22T13:08:00Zreach296http://www.cnblogs.com/reach296/<p>1、&nbsp; 什么是协议端口?</p><p>在Internet上,各主机间通过TCP/TP协议发送和接收数据报,各个数据报根据其目的主机的ip地址来进行互联网络中的路由选择。可见,把数据报顺利的传送到目的主机是没有问题的。问题出在哪里呢?我们知道大多数操作系统都支持多程序(进程)同时运行,那么目的主机应该把接收到的数据报传送给众多同时运行的进程中的哪一个呢?显然这个问题有待解决,端口机制便由此被引入进来。</p><p>&nbsp;&nbsp;&nbsp; 本地操作系统会给那些有需求的进程分配协议端口(protocal&nbsp;port,即我们常说的端口),每个协议端口由一个正整数标识,如:80,139,445,等等。当目的主机接收到数据报后,将根据报文首部的目的端口号,把数据发送到相应端口,而与此端口相对应的那个进程将会领取数据并等待下一组数据的到来。说到这里,端口的概念似乎仍然抽象,那么继续跟我来,别走开。</p><p>端口其实就是队,操作系统为各个进程分配了不同的队,数据报按照目的端口被推入相应的队中,等待被进程取用,在极特殊的情况下,这个队也是有可能溢出的,不过操作系统允许各进程指定和调整自己的队的大小。<br />不光接受数据报的进程需要开启它自己的端口,发送数据报的进程也需要开启端口,这样,数据报中将会标识有源端口,以便接受方能顺利的回传数据报到这个端口。</p><p>&nbsp; &nbsp;&nbsp;&nbsp;在Internet上,按照协议类型分类,端口被分为TCP端口和UDP端口两类,虽然他们都用正整数标识,但这并不会引起歧义,比如TCP的80端口和UDP的80端口,因为数据报在标明端口的同时,还将标明端口的类型。&nbsp;</p><p>&nbsp;&nbsp;&nbsp; &nbsp;从端口的分配来看,端口被分为固定端口和动态端口两大类(一些教程还将极少被用到的高端口划分为第三类:私有端口):<br />&nbsp;&nbsp;&nbsp; 固定端口(0-1023):<br />&nbsp;&nbsp;&nbsp; &nbsp;使用集中式管理机制,即服从一个管理机构对端口的指派,这个机构负责发布这些指派。由于这些端口紧绑于一些服务,所以我们会经常扫描这些端口来判断对方是否开启了这些服务,如TCP的21(ftp),80(http),139(netbios),UDP的7(echo),69(tftp)等等一些大家熟知的端口;<br />&nbsp;&nbsp;&nbsp; 动态端口(1024-65535):<br />&nbsp;&nbsp;&nbsp; &nbsp;这些端口并不被固定的捆绑于某一服务,操作系统将这些端口动态的分配给各个进程,同一进程两次分配有可能分配到不同的端口。</p><p>&nbsp;</p><p>2、&nbsp; 端口限制对性能的影响</p><p>理论上,linux和windows可用的端口个数为65535个Linux和windows动态端口的数量都存在限制,linux系统(如red hat)的动态端口限制为1024-4999,windows 2003SOCKET 端口数量默认5000,正常情况下,这些端口数量是够用的,但是如果服务器是用来提供web服务或者用来对其他服务器加压,这时候,服务器会产生大量的TCP连接,由于每个TCP客户端连接都要占用一个唯一的本地端口号,如果现有的TCP客户端连接已将所有的本地端口号占满,则此时就无法为新的TCP客户端连接分配一个本地端口号了,因此系统会在这种情况下在connect()调用中返回失败,并提示错误消息:"Can't assignrequested&nbsp;address".</p><p>&nbsp;</p><p>&nbsp;</p><p>3、&nbsp; 如何查看端口限制</p><p>首先可以通过cat /proc/sys/net/ipv4/ip_local_port_range来查看服务器端口的极限值</p><p>&nbsp;</p><p>&nbsp;</p><p>再通过</p><p>netstat -an | awk '/^tcp/ {++state[$6]} END&nbsp;&nbsp; {for (key in state) printkey,"\t",state[key]}'</p><p>命令来查看端口的占用情况</p><p>&nbsp;<img src="http://images.cnitblog.com/blog/638228/201408/222105471129864.png" alt="" /></p><p>如果端口极限值设置较小,而端口的已经被大量使用,尤其被大量TIME_WAIT占用</p><p>就可以确认端口已经被耗尽了。</p><p>&nbsp;<img src="http://images.cnitblog.com/blog/638228/201408/222106020035946.png" alt="" /></p><p>4、&nbsp; 如何修改端口限制</p><p>&nbsp;</p><p>#Vi&nbsp;/etc/sysctl.conf </p><p>修改net.ipv4.ip_local_port_range = 1024 65000</p><p># /sbin/sysctl -p</p><p>试配置生效即可。</p><img src="http://counter.cnblogs.com/blog/rss/3930252" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/reach296/p/3930252.html" target="_blank">[性能分析]端口限制</a>,转载请注明。</p>http://www.cnblogs.com/xiaomanon/p/3930242.htmlQt中配置OpenCV环境 - XiaoManon在Qt中开发基于OpenCV的应用时,需要配置对应函数库到环境变量,这时候我们需要使用到qmake能够识别的变量来指定环境变量。INCLUDEPATH: 用于指定搜索头文件到文件夹路径。LIBS: 包含链接时的库文件列表。使用-L来指定文件夹路径,-l来指定库的名称。而这些配置,我们需要写入到对应到...2014-08-22T12:59:00Z2014-08-22T12:59:00ZXiaoManonhttp://www.cnblogs.com/xiaomanon/<p>在Qt中开发基于OpenCV的应用时,需要配置对应函数库到环境变量,这时候我们需要使用到qmake能够识别的变量来指定环境变量。</p><p><span style="background-color: #00ff00;">INCLUDEPATH</span>: 用于指定搜索头文件到文件夹路径。</p><p><span style="background-color: #00ff00;">LIBS</span>: 包含链接时的库文件列表。使用-L来指定文件夹路径,-l来指定库的名称。</p><p>而这些配置,我们需要写入到对应到工程文件(*.pro)当中,可以有两种写法。</p><p>第一种:直接指定包含到库文件名称。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;">1</span> INCLUDEPATH += /usr/local/include/opencv2/<br/><span style="color: #008080;">2</span> <br/><span style="color: #008080;">3</span> LIBS += /usr/local/lib/<span style="color: #000000;">libopencv_core.so \<br/></span><span style="color: #008080;">4</span> /usr/local/lib/<span style="color: #000000;">libopencv_highgui.so \<br/></span><span style="color: #008080;">5</span> /usr/local/lib/libopencv_imgproc.so</div><p>这种方法必须指定整个文件到完整路径及名称,编写起来相对麻烦一点。</p><p>&nbsp;</p><p>第二种:指定库文件路径和库名称。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;">1</span> INCLUDEPATH += /usr/local/include/opencv2/<br/><span style="color: #008080;">2</span> LIBS += -L/usr/local/lib/<span style="color: #000000;"> \<br/></span><span style="color: #008080;">3</span> -<span style="color: #000000;">lopencv_core\<br/></span><span style="color: #008080;">4</span> -<span style="color: #000000;">lopencv_highgui\<br/></span><span style="color: #008080;">5</span> -lopencv_imgproc</div><p>这里只需要先声明库文件的位置,然后使用-l参数来添加库名称即可。</p><img src="http://counter.cnblogs.com/blog/rss/3930242" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/xiaomanon/p/3930242.html" target="_blank">Qt中配置OpenCV环境</a>,转载请注明。</p>http://www.cnblogs.com/xiaoheimiaoer/p/3930220.html炫酷吊炸天的nodeppt - 黑MAO由于要做一个关于node的分享,要准备写一个ppt方便就行交流。之前用的比较多的是slides(http://www.slides.com),最近知道了一个node写的工具,可以生成ppt,号称很强大,我就做了一次尝试(原因是使用markdown语法写ppt)。 nodeppt是@三水清大大的...2014-08-22T12:46:00Z2014-08-22T12:46:00Z黑MAOhttp://www.cnblogs.com/xiaoheimiaoer/<p>  由于要做一个关于node的分享,要准备写一个ppt方便就行交流。之前用的比较多的是slides(http://www.slides.com),最近知道了一个node写的工具,可以生成ppt,号称很强大,我就做了一次尝试(原因是使用markdown语法写ppt)。</p><p>&nbsp;</p><p>  nodeppt是@三水清大大的力作,想要了解nodeppt有哪些强大的功能,可以<a href="https://github.com/ksky521/nodePPT" target="_blank">来这儿</a>看看</p><p>&nbsp;</p><p>  我这里主要讲一下,我在使用过程需要注意的问题:</p><p>&nbsp;</p><p>  1.安装</p><p>  根据文档说明安装应该是这样子的</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">npm install -g nodeppt<br/></div><p>  本想着这样就能成功,结果安装过程中报了错误,发现是权限不够,所以就加了个sudo,果然成功安装了nodeppt。检测是否成功安装nodeppt可以使用下面的命令</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">nodeppt -V<br/>//记得是大写V<br/>//0.8.9-1<br/><br/>nodeppt -h<br/>//查看帮助<br/></div><p>  2.启动</p><p>  如果直接使用文档中的启动命令的话,有可能会报错,应该是ERROR: port 8080 is in use!意思就是说nodeppt默认开启的8080端口被占用了,这个时候需要使用设置端口号的命令</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">nodeppt start -p &lt;未被使用的端口&gt;<br/></div><p>  这样就可以成功开启服务了,这个时候就可以访问到nodeppt提供的默认demo了,上面介绍了nodeppt的特性和相关功能实用,那么如何执行自己的ppt呢?</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">nodeppt start -d &lt;ppt所在目录&gt;<br/></div><p>  nodeppt会读取设置目录下的md文件,作为ppt的入口。没有设置目录的话,或默认/usr/local/lib/node_modules/nodeppt/ppts/目录下的ppt。</p><p>  3.配置</p><p>  看demo例子可以发现,可以发现下面一些配置</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">title: nodeppt markdown 演示<br/>speaker: Theo Wang<br/>url: https://github.com/ksky521/nodePPT<br/>transition: slide<br/>files: /js/demo.js,/css/demo.css<br/><br/>/*<br/>title: 这是演讲的题目<br/>speaker: 演讲者名字<br/>url: 可以设置链接<br/>transition: 转场效果,例如:zoomin/cards/slide<br/>files: 引入js和css的地址,如果有的话~自动放在页面底部<br/>*/<br/></div><p>  关于files这配置,如果没有设置,默认从/usr/local/lib/node_modules/nodeppt/assets/读取静态资源。想要了解有那些专场效果可以<a href="https://github.com/ksky521/nodePPT" target="_blank">来这儿</a>查看。</p><p>  4.关闭</p><p>  关闭nodeppt服务的时候,我用ctrl+z实现了关闭,发现我之前开启的端口号都不能使用了,原因我虽然把服务关了,但是并没有关闭进程,这样的话,之前开启的端口都不能使用了。解决办法:可以使用ps -ef | grep username 可以查看这个username开启的进程有哪些,也可以使用netsta -nap | grep node查看node相关的端口状态,然后使用kill -9 &lt;进程ID&gt;杀死进程即可。</p><p>  当然最简单的办法就是:使用ctrl+c关闭nodeppt服务,这样的话进程同时会被干掉。</p><p>  5.其他</p><ul><li>nodeppt有一个比较好玩的功能,可以通过手机控制PPT展示,就是所谓的&ldquo;摇一摇&rdquo;切换ppt,很酷,很炫</li><li>可以导出pdf格式和html格式的文件</li></ul><p>&nbsp;</p><p>  可以<a href="http://qdemo.sinaapp.com/" target="_blank">看看demo</a>例子,看完之后,<span style="line-height: 1.5;">各位也可以去玩玩,挺不错的~</span></p><p>&nbsp;</p><p>  参考资料:https://github.com/ksky521/nodePPT</p><img src="http://counter.cnblogs.com/blog/rss/3930220" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/xiaoheimiaoer/p/3930220.html" target="_blank">炫酷吊炸天的nodeppt</a>,转载请注明。</p>http://www.cnblogs.com/reent/p/3930186.html关于CSS设置和获取元素不透明度的问题与解决方案 - ZREENT对于浏览器对元素不透明度度opacity的支持情况和如何设置,相信大家已经都熟悉了。这里就不再分析。简单的,要支持主流浏览器需要同时设置IE特有的滤镜和W3C标准的opacity属性。例如:设置不透明度为20% filter:alpha(opacity=20); opacity:0....2014-08-22T12:23:00Z2014-08-22T12:23:00ZZREENThttp://www.cnblogs.com/reent/<p>对于浏览器对元素不透明度度opacity的支持情况和如何设置,相信大家已经都熟悉了。这里就不再分析。</p><p>简单的,要支持主流浏览器需要同时设置IE特有的滤镜和W3C标准的opacity属性。</p><p><span style="color: #008000;">例如:设置不透明度为20% </span></p><p><span style="color: #008000;">    filter:alpha(opacity=20); </span></p><p><span style="color: #008000;">    opacity:0.2;</span></p><p>我需要重点分享的是CSS不透明度通过JS获取的问题。</p><p>首先,经过本人的大量有效分析测试得出:</p><p><span style="color: #ff6600;">    IE5 - IE8&nbsp;&nbsp; 只支持filter滤镜设置CSS不透明度,不支持opacity属性设置CSS不透明度</span></p><p><span style="color: #ff6600;">    IE9&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp; &nbsp;既支持filter滤镜设置CSS不透明度,也支持opacity属性设置CSS不透明度</span></p><p><span style="color: #ff6600;">    IE10+&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 不支持filter滤镜设置CSS不透明度,只支持opacity属性设置CSS不透明度</span></p><p><span style="color: #ff6600;">    Firefox Chrome Safari Opera 不支持filter滤镜设置CSS不透明度,只支持opacity属性设置CSS不透明度</span></p><p>&nbsp;</p><p><span style="color: #ff6600;">    IE9+ 支持currentStyle,也支持document.defaultView.getComputedStyle</span></p><p><br /><span style="color: #ff6600;">    IE5 - IE8 支持currentStyle,不支持document.defaultView.getComputedStyle</span></p><p><br /><span style="color: #ff6600;">    非IE 不支持currentStyle,只支持document.defaultView.getComputedStyle</span></p><p>&nbsp;</p><p><span style="color: #ff6600;">    IE 支持获取CSS用户自定义的属性,但是只能通过访问currentStyle返回,而访问document.defaultView.getComputedStyle无法获得自定义属性</span></p><p><span style="color: #ff6600;">    非IE 不支持获取CSS自定义属性</span></p><p>&nbsp;</p><p>所以,用JS获取不透明度属性的时候,因为默认的opacity属性值为1,但是IE8及其以下的版本不支持,所以会返回undefined,而其他支持该属性的浏览器返回1。</p><p>那么在如上面的例子中那样设置了不透明度为20%不透明,那么我们要通过IE支持的currentStyle来获取filter的值:alpha(opacity=20),在通过正则表达式得到数值20,再乘以0.01得到正确的值0.2,</p><p>对于非IE和支持opacity的浏览器,可以直接获取。</p><p>&nbsp;</p><p><strong><span style="color: #ff0000;">但是</span></strong></p><p>这里有一个需要注意的地方就是,返回值的浮点数与设置值不相等的问题。</p><p>在Chrome Safari Opera这3大浏览器中,设置opacity:0.2,当我们用JS获取opacity值得时候返回的却不是0.2,而是一个约等于0.2的很长的浮点数,浏览器测试这时判断返回值是否等于0.2,得到false。</p><p>对于这个问题我没有找到原因,但是在这3大浏览器中测试确实存在,返回值与设置值有微量误差,本人猜测这也许与获取CSS的方式是通过计算后的样式,其中浏览器内部的浮点数运算存在误差的缘故。</p><p>所以可以用 <span style="color: #ff6600;">0.01*Math.round(100*CSS.opacity)&nbsp;</span>方法处理返回值来得到正确的值。</p><p>在IE和Firefox中没有这样的问题,他们都返回设置的值。</p><p>&nbsp;</p><p>以上就是本人的一点小小的经验,非常高兴和大家分享!</p><img src="http://counter.cnblogs.com/blog/rss/3930186" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/reent/p/3930186.html" target="_blank">关于CSS设置和获取元素不透明度的问题与解决方案</a>,转载请注明。</p>http://www.cnblogs.com/travellife/p/3930179.htmlAndroid随笔之——Activity中启动另一应用 - 登天路最近在写语音交互程序,在语音打开应用这块碰到如何用代码控制应用启动的问题。百度了一下,有两种方案:1、获取应用的包名;2、获取应用的包名、入口类名。 之前对两种方案都进行了尝试,发现方案二中存在一个弊端,那就是系统级应用只能获取到包名,不能获取到入口类名,这会导致程序报错,当然你也可以不获取系...2014-08-22T12:15:00Z2014-08-22T12:15:00Z登天路http://www.cnblogs.com/travellife/<p>  最近在写语音交互程序,在语音打开应用这块碰到如何用代码控制应用启动的问题。百度了一下,有两种方案:1、获取应用的包名;2、获取应用的包名、入口类名。</p><p>  之前对两种方案都进行了尝试,发现方案二中存在一个弊端,那就是系统级应用只能获取到包名,不能获取到入口类名,这会导致程序报错,当然你也可以不获取系统级应用的包名和入口类名。出于简单方便的考虑,选择了方案一,废话不多说,上代码。</p><p>方案一、</p><p>1、获取程序包名:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;"> 1</span>      <span style="color: #008000;">//</span><span style="color: #008000;"> 获取PackageManager对象</span><br/><span style="color: #008080;"> 2</span> PackageManager packageManager =<span style="color: #000000;"> getPackageManager();<br/></span><span style="color: #008080;"> 3</span> <span style="color: #008000;">//</span><span style="color: #008000;"> 得到系统安装的所有程序的PackageInfo对象</span><br/><span style="color: #008080;"> 4</span> List&lt;PackageInfo&gt; packages = packageManager.getInstalledPackages(0<span style="color: #000000;">);<br/></span><span style="color: #008080;"> 5</span> <span style="color: #008000;">//</span><span style="color: #008000;"> 创建String类型的数组,用来存放获取的包名</span><br/><span style="color: #008080;"> 6</span> ArrayList&lt;String&gt; packageList = <span style="color: #0000ff;">new</span> ArrayList&lt;String&gt;<span style="color: #000000;">();<br/></span><span style="color: #008080;"> 7</span> <br/><span style="color: #008080;"> 8</span> <span style="color: #0000ff;">for</span> (<span style="color: #0000ff;">int</span> i = 0; i &lt; packages.size(); i++<span style="color: #000000;">) {<br/></span><span style="color: #008080;"> 9</span> PackageInfo packageInfo =<span style="color: #000000;"> packages.get(i);<br/></span><span style="color: #008080;">10</span> <br/><span style="color: #008080;">11</span> <span style="color: #008000;">//</span><span style="color: #008000;"> 判断是否是系统级应用</span><br/><span style="color: #008080;">12</span> <span style="color: #0000ff;">if</span> ((packageInfo.applicationInfo.flags &amp; ApplicationInfo.FLAG_SYSTEM) == 0<span style="color: #000000;">) {<br/></span><span style="color: #008080;">13</span> <span style="color: #000000;"> packageList.add(packageInfo.packageName);<br/></span><span style="color: #008080;">14</span> <span style="color: #000000;"> }<br/></span><span style="color: #008080;">15</span> }</div><p>&nbsp;</p><p>2、根据获取的包名启动应用:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;"> 1</span>      <span style="color: #008000;">//</span><span style="color: #008000;"> 获取PackageManager对象</span><br/><span style="color: #008080;"> 2</span> PackageManager packageManager =<span style="color: #000000;"> getPackageManager();<br/></span><span style="color: #008080;"> 3</span> <span style="color: #008000;">//</span><span style="color: #008000;"> 获取索引为i的包名</span><br/><span style="color: #008080;"> 4</span> String packageName =<span style="color: #000000;"> packageList.get(i);<br/></span><span style="color: #008080;"> 5</span> <span style="color: #008000;">//</span><span style="color: #008000;"> 如果该程序不可启动(像系统自带的包,有很多是没有入口的)会返回NULL</span><br/><span style="color: #008080;"> 6</span> Intent intent =<span style="color: #000000;"> packageManager.getLaunchIntentForPackage(packageName);<br/></span><span style="color: #008080;"> 7</span> <br/><span style="color: #008080;"> 8</span> <span style="color: #0000ff;">if</span> (intent != <span style="color: #0000ff;">null</span><span style="color: #000000;">) {<br/></span><span style="color: #008080;"> 9</span> <span style="color: #000000;"> startActivity(intent);<br/></span><span style="color: #008080;">10</span> }</div><p>&nbsp;</p><p>方案二、还没仔细研究过,不过对那些主入口设置&lt;intent-filter&gt;的应用,方案二应该会优于方案一,明天找个时间把这个补上</p><img src="http://counter.cnblogs.com/blog/rss/3930179" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/travellife/p/3930179.html" target="_blank">Android随笔之——Activity中启动另一应用</a>,转载请注明。</p>http://www.cnblogs.com/ready-Yang/p/3930074.html4行代码实现js模板引擎 - 杨捷在平时编码中,经常要做拼接字符串的工作,如把json数据用HTML展示出来,以往字符串拼接与逻辑混在在一起会让代码晦涩不堪,加大了多人协作与维护的成本。而采用前端模板机制就能很好的解决这个问题。精妙的 tmpl前端模板类开源的不少,但最属 jQuery 作者 John Resig 开发的 “java...2014-08-22T11:52:00Z2014-08-22T11:52:00Z杨捷http://www.cnblogs.com/ready-Yang/<p>在平时编码中,经常要做拼接字符串的工作,如把json数据用HTML展示出来,以往字符串拼接与逻辑混在在一起会让代码晦涩不堪,加大了多人协作与维护的成本。而采用前端模板机制就能很好的解决这个问题。</p><p>&nbsp;</p><p><strong>精妙的 tmpl</strong></p><p>&nbsp;</p><p>前端模板类开源的不少,但最属 jQuery 作者 John Resig 开发的 &ldquo;javascript micro templating&rdquo; 最为精妙,寥寥几笔便实现了模板引擎核心功能。 它的介绍与使用方式请看作者博客:<a href="http://ejohn.org/blog/javascript-micro-templating/">http://ejohn.org/blog/javascript-micro-templating/</a></p><p>&nbsp;</p><p>麻雀虽小,五脏俱全,除了基本的数据附加外,还拥有缓存机制、逻辑支持。现在,若要我评出一个javascript 最节能的自定义函数排名,第一名是 $ 函数(document.getElementById 简版),而第二名就是 tmpl 了。</p><p>&nbsp;</p><p>当然,它并非完美,我使用过程中发现了一些问题:</p><p>&nbsp;</p><p><strong>tmpl 美中不足</strong></p><p>&nbsp;</p><p>&nbsp;&nbsp;&nbsp; 一、无法正确处理转义字符,如: tmpl('&lt;%=name%&gt;\\&lt;%=id%&gt; ', {name:'糖饼', id: '1987'});它就会报错。若正常工作,它应该输出:糖饼\1987</p><p>&nbsp;</p><p>&nbsp;&nbsp;&nbsp; 二、无法识别数据里的单引号</p><p>&nbsp;</p><p>&nbsp;&nbsp;&nbsp; 三、设置变量默认值复杂,如</p><p>&nbsp;</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;tmpl('&lt;%if(obj.name){%&gt;&lt;%=name%&gt;&lt;%}else{%&gt;默认值&lt;%}%&gt; ', {name:'糖饼'});&nbsp; //设置name默认为 &ldquo;默认值&rdquo;</p><p><strong>tmpl 优化版本</strong></p><p>废话不多说,先敬上代码:</p><p>&nbsp;</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;"> 1</span> <span style="color: #0000ff;">function</span><span style="color: #000000;"> tmpl(str, data) {<br/></span><span style="color: #008080;"> 2</span> <span style="color: #0000ff;">var</span> $ = '$' + (+ <span style="color: #0000ff;">new</span><span style="color: #000000;"> Date)<br/></span><span style="color: #008080;"> 3</span> , fn = <span style="color: #0000ff;">function</span><span style="color: #000000;"> (data) {<br/></span><span style="color: #008080;"> 4</span> <span style="color: #0000ff;">var</span> i, variable = [$], value =<span style="color: #000000;"> [[]];<br/></span><span style="color: #008080;"> 5</span> <span style="color: #0000ff;">for</span> (i <span style="color: #0000ff;">in</span><span style="color: #000000;"> data) {<br/></span><span style="color: #008080;"> 6</span> <span style="color: #000000;"> variable.push(i);<br/></span><span style="color: #008080;"> 7</span> <span style="color: #000000;"> value.push(data[i]);<br/></span><span style="color: #008080;"> 8</span> <span style="color: #000000;"> }<br/></span><span style="color: #008080;"> 9</span> <span style="color: #0000ff;">return</span> (<span style="color: #0000ff;">new</span><span style="color: #000000;"> Function(variable, fn.$))<br/></span><span style="color: #008080;">10</span> .apply(data, value).join(""<span style="color: #000000;">);<br/></span><span style="color: #008080;">11</span> <span style="color: #000000;"> };<br/></span><span style="color: #008080;">12</span> <br/><span style="color: #008080;">13</span> <span style="color: #008000;">//</span><span style="color: #008000;">将模板解析成函数</span><br/><span style="color: #008080;">14</span> fn.$ = fn.$ || $ + ".push('"<br/><span style="color: #008080;">15</span> + str.replace(/\\/g, "\\\\"<span style="color: #000000;">)<br/></span><span style="color: #008080;">16</span> .replace(/'/g, "\\'") <span style="color: #008000;">//</span><span style="color: #008000;">防止单括号错误</span><br/><span style="color: #008080;">17</span> .replace(/[\r\t\n]/g, " "<span style="color: #000000;">)<br/></span><span style="color: #008080;">18</span> .split("[:").join("\t"<span style="color: #000000;">)<br/></span><span style="color: #008080;">19</span> .replace(/((^|:])[^\t]*)'/g, "$1\r"<span style="color: #000000;">)<br/></span><span style="color: #008080;">20</span> .replace(/\t=([^\?]*?):]/g, "',$1,'"<span style="color: #000000;">)<br/></span><span style="color: #008080;">21</span> .replace(/\t=([^\?]*?)\?(.*?):]/g, "',this.$1||'$2','") <span style="color: #008000;">//</span><span style="color: #008000;"> [:=data?:] [:=data?任何内容:]</span><br/><span style="color: #008080;">22</span> .split("\t").join("');"<span style="color: #000000;">)<br/></span><span style="color: #008080;">23</span> .split(":]").join($ + ".push('"<span style="color: #000000;">)<br/></span><span style="color: #008080;">24</span> .split("\r").join("\\'"<span style="color: #000000;">)<br/></span><span style="color: #008080;">25</span> + "');return " +<span style="color: #000000;"> $;<br/></span><span style="color: #008080;">26</span> <br/><span style="color: #008080;">27</span> <span style="color: #008000;">//</span><span style="color: #008000;">如果未定义data则返回编译好的函数,使用时直接传入数据即可,</span><br/><span style="color: #008080;">28</span> <span style="color: #008000;">//</span><span style="color: #008000;">省去每次解析成函数的时间</span><br/><span style="color: #008080;">29</span> <span style="color: #0000ff;">return</span> data ?<span style="color: #000000;"> fn(data) : fn;<br/></span><span style="color: #008080;">30</span> <br/><span style="color: #008080;">31</span> }; </div><p>好吧,上面的代码看起来超出了4行,原谅我标题党。不过这段代码经过压缩后,确实只有四行^_^。下面我们就来详细解构它。</p><p>首先看一下使用示例:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;"> 1</span> <span style="color: #008000;">//</span><span style="color: #008000;">循环结构</span><br/><span style="color: #008080;"> 2</span> <br/><span style="color: #008080;"> 3</span> <span style="color: #0000ff;">var</span> tpl = '[: for(var k in ary){ var one=ary[k]; :]'<br/><span style="color: #008080;"> 4</span> + '&lt;p&gt;[:=one:]&lt;/p&gt;'<br/><span style="color: #008080;"> 5</span> + '[: } :]'<span style="color: #000000;">;<br/></span><span style="color: #008080;"> 6</span> <span style="color: #0000ff;">var</span> data = {ary:[123,'abc'<span style="color: #000000;">]};</span><br/><span style="color: #008080;"> 8</span> <span style="color: #0000ff;">var</span> div =<span style="color: #000000;"> tmpl(tpl,data);</span><br/><span style="color: #008080;">10</span> console.log(div); <span style="color: #008000;">//&lt;/p&gt;</span><span style="color: #008000;">123&lt;/p&gt;&lt;p&gt;abc&lt;/p&gt;</span><br/><span style="color: #008080;">11</span> <br/><span style="color: #008080;">12</span> <br/><span style="color: #008080;">13</span> <span style="color: #008000;">//</span><span style="color: #008000;">变量验证</span><br/><span style="color: #008080;">14</span> <br/><span style="color: #008080;">15</span> <span style="color: #0000ff;">var</span> tpl = '[: if(this.name!==undefined){ :]' <span style="color: #008000;">//</span><span style="color: #008000;">注意必须使用 this.name,直接使用name,如果未定义就会报错</span><br/><span style="color: #008080;">16</span> + '&lt;p&gt;[:=name:]&lt;/p&gt;'<br/><span style="color: #008080;">17</span> + '[: } :]'<span style="color: #000000;">;<br/></span><span style="color: #008080;">18</span> <span style="color: #0000ff;">var</span> data = {name:'abc'<span style="color: #000000;">};</span><br/><span style="color: #008080;">20</span> <span style="color: #0000ff;">var</span> div =<span style="color: #000000;"> tmpl(tpl,data);</span><br/><span style="color: #008080;">22</span> <br /><br/><span style="color: #008080;">23</span> <span style="color: #008000;">//</span><span style="color: #008000;">你还可以这样方便地使用未定义的变量:</span><br/><span style="color: #008080;">24</span> <br /><br/><span style="color: #008080;">25</span> <span style="color: #0000ff;">var</span> tpl = '&lt;p&gt;name:[:=name?:], name:[:=name?默认值:]&lt;/p&gt;'<span style="color: #000000;">;<br/></span><span style="color: #008080;">26</span> <span style="color: #0000ff;">var</span> data = {no:'abc'<span style="color: #000000;">};<br/></span><span style="color: #008080;">27</span> <span style="color: #0000ff;">var</span> div =<span style="color: #000000;"> tmpl(tpl,data);<br/></span><span style="color: #008080;">28</span> <br/><span style="color: #008080;">29</span> console.log(div); <span style="color: #008000;">//</span><span style="color: #008000;"> &lt;p&gt;name:, name:默认值&lt;/p&gt;</span><br/><span style="color: #008080;">30</span> <br/><span style="color: #008080;">31</span> <br/><span style="color: #008080;">32</span> <span style="color: #008000;">//</span><span style="color: #008000;">缓存编译结果</span><br/><span style="color: #008080;">33</span> <br/><span style="color: #008080;">34</span> <br/><span style="color: #008080;">35</span> <span style="color: #0000ff;">var</span> tpl = '[: for(var k in ary){ var one=ary[k]; :]'<br/><span style="color: #008080;">36</span> + '&lt;<span style="color: #008000;">p&gt;</span>[:=one:]&lt;/p&gt;'<br/><span style="color: #008080;">37</span> + '[: } :]'<span style="color: #000000;">;<br/></span><span style="color: #008080;">38</span> <span style="color: #0000ff;">var</span> data = {ary:[123,'abc'<span style="color: #000000;">]};</span><br/><span style="color: #008080;">40</span> <span style="color: #0000ff;">var</span> render = tmpl(tpl); <span style="color: #008000;">//</span><span style="color: #008000;">不传入data,则生成缓存,多次使用缓存节约大量正则运算</span><br/><span style="color: #008080;">42</span> <span style="color: #0000ff;">var</span> div = render(data); <span style="color: #008000;">//</span><span style="color: #008000;">传入data,代入变量,解析成最终结果</span><br/><span style="color: #008080;">43</span> <br/><span style="color: #008080;">44</span> console.log(div); <span style="color: #008000;">//&lt;</span><span style="color: #008000;">p&gt;123&lt;/p&gt;&lt;p&gt;abc&lt;/p&gt;</span></div><p>使用方法:在 [: 与&nbsp;:] 之间使用任何js代码,并且通过 [:=data:] 方式以字符串形式输出变量。更加详细的使用方法/手册,请查看:<a href="http://docs.codekart.jojoin.com/p/tool_tmpl">http://docs.codekart.jojoin.com/p/tool_tmpl</a></p><p><strong>优化的地方:</strong></p><p>&nbsp;&nbsp;&nbsp; 一. 正确处理转义字符 \ ' 等转义字符</p><p>&nbsp;&nbsp;&nbsp; 二. 修改包裹符 &lt;% %&gt; 为 [: :] 防止与 html标签&lt;/&gt;和求余运算符%产生冲突。</p><p>&nbsp;&nbsp;&nbsp; 三. 修改环境变量 obj 为 this</p><p>&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp;&nbsp; tmpl('&lt;%if(obj.name=="name")%&gt;')&nbsp;&nbsp; //旧版本</p><p>&nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp; tmpl('[: if(this.name=="name") :]')&nbsp;&nbsp; //新版本</p><p>&nbsp;&nbsp;&nbsp; 四. 为变量添加默认值</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &nbsp; tmpl('&lt;%if(obj.name){%&gt;&lt;%=name%&gt;&lt;%}else{%&gt;默认值&lt;%}%&gt;')&nbsp;&nbsp; //旧版本</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &nbsp; tmpl('[:=name?默认值:]')&nbsp;&nbsp; //新版本</p><p>&nbsp;&nbsp;&nbsp; 五. 去掉 with 语句,大幅提升引擎性能</p><p>&nbsp;&nbsp;&nbsp; 六. 删除可有可无的功能,保持精简</p><p>&nbsp;&nbsp;&nbsp; 七. 增加调试模式</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//打印模板编译中间结果:</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;console(tmpl('&lt;p&gt;[:=name:]&lt;/p&gt;').$);&nbsp;</p><p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //$1408707567855.push('&lt;p&gt;',name,'&lt;/p&gt;');return $1408707567855</p><p>此引擎函数大致分为两部分:</p><p>&nbsp;&nbsp;&nbsp; 一. 上半部分:模板函数解析执行</p><p>&nbsp;&nbsp;&nbsp; 二. 下半部分:正则运算生成模板函数</p><p>可以看出,引擎实现的重点主要在下半部分的一堆正则表达式,也就是&ldquo;模板编译&rdquo;的过程。这里我不打算把每一个正则的功能都说清楚,那样篇幅太大(好吧是我懒),各位看官请自行阅读研究。</p><p>另外,此模板引擎已被集成到 Node.js&nbsp;web开发框架 <a title="Codekart" href="http://www.cnblogs.com/codekart.jojoin.com" target="_blank">Codekart</a>&nbsp;中。</p><p><strong>Codekart</strong> 是一套给 Node.js 开发者使用的应用程序开发框架和工具包。 它提供一套丰富的标准库以及简单的接口和逻辑结构, 其目的是使开发人员更快速地进行项目开发。 使用 Codekart 可以减少代码的编写量, 并将你的精力投入到项目的创造性开发上。</p><p>它已经帮你出色的完成了下面这些事情:</p><p><strong>优雅的框架思维</strong></p><p>如果你需要一个真正的框架,而不是一个模块/中间件/工具箱,如果你需要简约与便捷,需要一目了然、理所当然的舒适感,那么 Codekart 将是最好的选择。</p><p><strong>高性能 HTTP 服务器</strong></p><p>Codekart 处理 http 请求的性能接近原生 Node.js 代码: http.createServer(), 原因是框架只是对此函数做了简单的封装,其性能的损耗仅仅只有一个 url 正则匹配运算,路由请求处理程序。</p><p><strong>便捷的静态文件服务器</strong></p><p>把文件放入 static/ 目录下,启动Codekart,url 访问,搞定!</p><p><strong>web 页面模块化支持</strong></p><p>实际上,这是Codekart最出色的部分!它是前后端一体化的,可以像写配置文件一样编写web页面, 框架自动完成 js、css 、tpl 文件的模块化加载、合并、压缩, 并在html里引用,自动完成 html 模板的解析,并且支持页面继承和多态,一切就是那么简单轻松!</p><p><strong>丰富的工具箱</strong></p><p>Codekart 准备了一系列强大的前后端工具集合,涉及进程通信,数据缓存,文件读取,文件上传,数据采集与处理,流程控制,任务计划等诸多方面。</p><p>框架源码托管在Github:<a href="https://github.com/yangjiePro/Codekart">https://github.com/yangjiePro/Codekart</a>&nbsp;&nbsp;&nbsp;欢迎提交新的代码!</p><p>&nbsp;</p><img src="http://counter.cnblogs.com/blog/rss/3930074" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/ready-Yang/p/3930074.html" target="_blank">4行代码实现js模板引擎</a>,转载请注明。</p>http://www.cnblogs.com/lizhenghn/p/3930107.html使用autotools系列工具自动部署源代码编译安装 - lizhenghn在Linux系统下开发一个较大的项目,完全手动建立Makefile是一件费力而又容易出错的工作。autotools系列工具只需用户输入简单的目标文件、依赖文件、文件目录等就可以比较轻松地生成Makefile了。这极大地简化了Makefile的编写和维护,作者也是刚体验到其威力,所以将其过程记录下来。...2014-08-22T11:08:00Z2014-08-22T11:08:00Zlizhenghnhttp://www.cnblogs.com/lizhenghn/<p>在Linux系统下开发一个较大的项目,完全手动建立Makefile是一件费力而又容易出错的工作。autotools系列工具只需用户输入简单的目标文件、依赖文件、文件目录等就可以比较轻松地生成Makefile了。</p><p>这极大地简化了Makefile的编写和维护,作者也是刚体验到其威力,所以将其过程记录下来。</p><p>本文以一个简单的hello.c文件进行验证。</p><p><strong>准备工作</strong></p><p>首先需要安装autotools系列工具,包括aclocal、autoscan、automake、autoheader、autoconf等。</p><p>如果在centos或redhat系统下可以通过rpm &ndash;qa | grep auto来查看各个应用程序,如果没有,直接yum install automake即可安装。</p><p><strong>大致流程</strong></p><ul><li>1. 编写源代码以及手动书写Makefile.am文件(稍后介绍);</li><li>2. 依次使用各种autotools系列工具配置编译属性;</li></ul><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #000000;"> autoscan #扫面当前目录、源文件,生成configure.scan文件<br/> aclocal #根据configure.in生成aclocal.m4文件以及autom4te.cache文件夹<br/> autoconf #根据configure.in和aclocal.m4来产生configure文件<br/> autoheader #生成配置头文件的模板config.h.</span><span style="color: #0000ff;">in</span><br/> <span style="color: #0000ff;">touch</span><span style="color: #000000;"> README NEWS AUTHORS ChangeLog #生成一些声明性文件<br/> automake </span>--add-missing #生成Makefiel.in和所需要的脚本</div><ul><li><span style="line-height: 1.5;">3. 编译、生成、部署:</span></li></ul><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">make</span><br/><span style="color: #0000ff;">make</span> <span style="color: #0000ff;">install</span><br/><span style="color: #0000ff;">make</span><span style="color: #000000;"> clean<br/></span><span style="color: #0000ff;">make</span><span style="color: #000000;"> dist #制作发布的软件包<br/></span><span style="color: #0000ff;">make</span> distcheck #检验完整性并制作软件包 </div><p><strong>详细步骤</strong></p><p><strong><span style="color: #ff0000;">1. &nbsp;autoscan</span></strong></p><p>将生成configure.scan和autoscan.log文件,它会在给定目录及其子目录树中检查源文件,若没有给定目录,就在当前目录及其子目录树中进行检查。它会搜索源文件以寻找一般的移植性问题并且创建一个文件configure.scan,通过这个文件我们可以创建autoconf需要的模版文件。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">[root@lzv6 hello]# <span style="color: #0000ff;">ls</span><span style="color: #000000;"><br/>hello.c<br/>[root@lzv6 hello]# autoscan<br/>[root@lzv6 hello]# </span><span style="color: #0000ff;">ls</span><span style="color: #000000;"><br/>autoscan.log configure.scan hello.c</span></div><p><strong><span style="color: #ff0000;">2. &nbsp;重命名configure.scan,并修改configure.in</span></strong></p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #0000ff;">mv</span> configure.scan configure.<span style="color: #0000ff;">in</span> #将上步生成的configure.scan更改为autoconf需要的文件模版configure.<span style="color: #0000ff;">in</span></div><p>在configure.in中增加以下内容:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">AC_INIT([hello], [<span style="color: #800080;">0.1</span><span style="color: #000000;">], [lizhenghn@gmail.com])<br/>AM_INIT_AUTOMAKE(hello,</span><span style="color: #800080;">1.0</span><span style="color: #000000;">)<br/>AC_PROG_CXX<br/>AC_PROG_LIBTOOL<br/>AC_CONFIG_FILES(Makefile)</span></div><p>修改后的configure.in内容如下:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #008080;"> 1</span> # -*- Autoconf -*-<br/><span style="color: #008080;"> 2</span> # Process this <span style="color: #0000ff;">file</span><span style="color: #000000;"> with autoconf to produce a configure script.<br/></span><span style="color: #008080;"> 3</span> <br/><span style="color: #008080;"> 4</span> AC_PREREQ([<span style="color: #800080;">2.63</span><span style="color: #000000;">])<br/></span><span style="color: #008080;"> 5</span> AC_INIT([hello], [<span style="color: #800080;">0.1</span><span style="color: #000000;">], [lizhenghn@gmail.com]) #用来定义软件的名字、版本,以及邮箱地址信息<br/></span><span style="color: #008080;"> 6</span> #AC_INIT([hello], [<span style="color: #800080;">0.1</span><span style="color: #000000;">])<br/></span><span style="color: #008080;"> 7</span> AM_INIT_AUTOMAKE(hello,<span style="color: #800080;">0.1</span><span style="color: #000000;">) #必须添加此行,描述了要生成的软件包的名字及其版本号<br/></span><span style="color: #008080;"> 8</span> <span style="color: #000000;">AC_CONFIG_SRCDIR([hello.c]) #用来侦测所指定的源码文件是否存在<br/></span><span style="color: #008080;"> 9</span> <span style="color: #000000;">AC_CONFIG_HEADERS([config.h])<br/></span><span style="color: #008080;">10</span> <br/><span style="color: #008080;">11</span> # Checks <span style="color: #0000ff;">for</span><span style="color: #000000;"> programs.<br/></span><span style="color: #008080;">12</span> <span style="color: #000000;">AC_PROG_CC #检测系统所用的C编译器<br/></span><span style="color: #008080;">13</span> <span style="color: #000000;">AC_PROG_CXX<br/></span><span style="color: #008080;">14</span> <br/><span style="color: #008080;">15</span> # Checks <span style="color: #0000ff;">for</span><span style="color: #000000;"> libraries.<br/></span><span style="color: #008080;">16</span> <span style="color: #000000;">#AC_PROG_RANLIB #需要需要静态库<br/></span><span style="color: #008080;">17</span> <span style="color: #000000;">AC_PROG_LIBTOOL #如果需要动态库<br/></span><span style="color: #008080;">18</span> <br/><span style="color: #008080;">19</span> # Checks <span style="color: #0000ff;">for</span><span style="color: #000000;"> header files.<br/></span><span style="color: #008080;">20</span> <br/><span style="color: #008080;">21</span> # Checks <span style="color: #0000ff;">for</span><span style="color: #000000;"> typedefs, structures, and compiler characteristics.<br/></span><span style="color: #008080;">22</span> <br/><span style="color: #008080;">23</span> # Checks <span style="color: #0000ff;">for</span><span style="color: #000000;"> library functions.<br/></span><span style="color: #008080;">24</span> <span style="color: #000000;">AC_CONFIG_FILES(Makefile)<br/></span><span style="color: #008080;">25</span> AC_OUTPUT</div><p><strong><span style="color: #ff0000;">3. aclocal :复制所有的宏命令</span></strong></p><p>configure.in 里面包含了一系列的宏命令,运行aclocal的目的是把工程需要的宏命令展开。aclocal.m4 就是configure.in中用到的宏定义。本步会生成aclocal.m4文件和autom4te.cache文件夹。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #000000;">[root@lzv6 hello]# aclocal<br/>[root@lzv6 hello]# </span><span style="color: #0000ff;">ls</span><span style="color: #000000;"><br/>aclocal.m4 autom4te.cache autoscan.log configure.</span><span style="color: #0000ff;">in</span> hello.c</div><p><strong><span style="color: #ff0000;">4. autoheader :生成config.h.in文件</span></strong></p><p>生成配置头文件的模板config.h.in文件。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #000000;">[root@lzv6 hello]# autoheader<br/>[root@lzv6 hello]# </span><span style="color: #0000ff;">ls</span><span style="color: #000000;"><br/>aclocal.m4 autom4te.cache autoscan.log config.h.</span><span style="color: #0000ff;">in</span> configure.<span style="color: #0000ff;">in</span> hello.c</div><p><strong><span style="color: #ff0000;">5. 创建一些声明性文件</span></strong></p><p>此处创建一些声明性,但非必须的文件,文件内容可以暂时都为空。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">[root@lzv6 hello]# <span style="color: #0000ff;">touch</span> README NEWS AUTHORS ChangeLog</div><p><strong><span style="color: #ff0000;">6. autoconf :生成configure脚本</span></strong></p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;"><span style="color: #000000;">[root@lzv6 hello]# autoconf<br/>[root@lzv6 hello]# </span><span style="color: #0000ff;">ls</span><span style="color: #000000;"><br/>aclocal.m4 AUTHORS autom4te.cache autoscan.log ChangeLog config.h.</span><span style="color: #0000ff;">in</span> configure configure.<span style="color: #0000ff;">in</span> hello.c NEWS README</div><p><strong><span style="color: #ff0000;">7. 编写Makefile.am</span></strong></p><p>如下是一个简单的Makefile.am</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">AUTOMAKE_OPTIONS=<span style="color: #000000;">foreign<br/>bin_PROGRAMS</span>=<span style="color: #000000;">hello<br/>hello_SOURCES</span>=hello.c</div><p>说明:AUTOMAKE_OPTIONS为设置automake的选项。automake提供了3种软件等级:foreign、gnu、gnits,默认等级是gnu。此处使用的foreign表示只是检测必要的文件。<br />bin_PROGRAMS定义了要产生的执行文件名。如果产生多个可执行文件,每个文件名用空格隔开。<br />file_SOURCES定义file这个执行程序的依赖文件。同样的,对于多个执行文件,那就要定义相应的file_SOURCES。</p><p><strong><span style="color: #ff0000;">8. automake --add-missing</span></strong></p><p>生成Makefiel.in和所需要的脚本,其中add-missing选项会让automake自动添加一些必须的脚本文件。</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">[root@lzv6 hello]# automake --add-<span style="color: #000000;">missing <br/>configure.</span><span style="color: #0000ff;">in</span>:<span style="color: #800080;">16</span>: installing `./config.guess<span style="color: #800000;">'<br/></span>configure.<span style="color: #0000ff;">in</span>:<span style="color: #800080;">16</span>: installing `./config.sub<span style="color: #800000;">'<br/></span>configure.<span style="color: #0000ff;">in</span>:<span style="color: #800080;">7</span>: installing `./<span style="color: #0000ff;">install</span>-<span style="color: #0000ff;">sh</span><span style="color: #800000;">'<br/></span>configure.<span style="color: #0000ff;">in</span>:<span style="color: #800080;">16</span>: required <span style="color: #0000ff;">file</span> `./ltmain.<span style="color: #0000ff;">sh</span><span style="color: #800000;">'</span><span style="color: #800000;"> not found</span><br/>configure.<span style="color: #0000ff;">in</span>:<span style="color: #800080;">7</span>: installing `./missing<span style="color: #800000;">'<br/></span>Makefile.am: installing `./depcomp<span style="color: #800000;">'</span> <br/>[root@lzv6 hello]# libtoolize --automake --copy --debug --<span style="color: #000000;">force #注意此处步骤!!!<br/>[root@lzv6 hello]# automake </span>--add-<span style="color: #000000;">missing <br/>[root@lzv6 hello]# </span><span style="color: #0000ff;">ls</span><span style="color: #000000;"><br/>aclocal.m4 autom4te.cache ChangeLog config.h.</span><span style="color: #0000ff;">in</span> configure depcomp <span style="color: #0000ff;">install</span>-<span style="color: #0000ff;">sh</span><span style="color: #000000;"> Makefile.am missing README<br/>AUTHORS autoscan.log config.guess config.sub configure.</span><span style="color: #0000ff;">in</span> hello.c ltmain.<span style="color: #0000ff;">sh</span> Makefile.<span style="color: #0000ff;">in</span> NEWS </div><p>说明:第一次运行时报错,此时运行命令libtoolize --automake --copy --debug --force即可解决。</p><p>至此该工程的Makefile就配置完成了。</p><p>&nbsp;</p><p>在以后的使用中进行编译时可以直接运行命令:</p><div style="background-color: #F5F5F5;border: 1px solid #CCCCCC;padding:10px;">[root@lzv6 hello]# ./<span style="color: #000000;">configure<br/>[root@lzv6 hello]# </span><span style="color: #0000ff;">make</span><span style="color: #000000;"> <br/>[root@lzv6 hello]# </span><span style="color: #0000ff;">make</span> <span style="color: #0000ff;">install</span> #默认安装在/usr/local/bin/<br />[root@lzv6 hello]# make dist #打包,会在当前目前下生成hello-0.1.tar.gz</div><p>当你增加代码文件时,其实主需要修改Makefile.am即可,这里就不再详述了。</p><p>这里附上一张整个步骤的流程图:</p><p><img src="http://images.cnitblog.com/blog/601232/201408/221946358789101.png" alt="" width="712" height="617" /></p><p>&nbsp;</p><img src="http://counter.cnblogs.com/blog/rss/3930107" width="1" height="1" alt=""/><br/><p>本文链接:<a href="http://www.cnblogs.com/lizhenghn/p/3930107.html" target="_blank">使用autotools系列工具自动部署源代码编译安装</a>,转载请注明。</p>

高级...
微信扫一扫
关注无线梦工厂
返回主页 手机网推广 加入Timewe 反馈意见 关于我们 加入收藏 点击这里给我发消息加关注加微信好友

触屏版,3G,Wap,手机网站,浏览器,pc版,在线,嵌入式,Wap浏览器,电脑上的Wap浏览器 各主流搜索引擎收录Timewe
(R)2009-2019 Timewe 时维网络科技有限公司 (86)0755-36601136
简体/繁体