2013年2月1日星期五

架設一台VNC Server(for CentOS 5.5 or 6.2)

一、安裝與設定VNC Server

Step 1) 安裝vnc-server package

(此為CentOS 5.5安裝package方式)
[root]$ yum install -y vnc-server

(此為CentOS 6.2安裝package方式)
[root]$ yum install -y tigervnc-server

Step 2) 修改vnc-server設定檔

[root]$ vim /etc/sysconfig/vncservers
------------------------------------------------------------
#VNCSERVERS="2:myusername" #預設值
#VNCSERVERARGS[2]="-geometry 800x600 -nolisten tcp -nohttpd -localhost"

#指定vnc server要開啟的編號 (此處編號設定1)
VNCSERVERS="1:vncuser"
#以vnc server預設的基底port為5900,再加上編號1
#所以我們會開啟5901這個port給vncuser這個權限使用!

#指定vnc連接時的Xwindows解析度
VNCSERVERARGS[2]="-geometry 1024x768"
------------------------------------------------------------



Step 3) 新增vncuser在Linux下的帳號與家目錄

[root]$ useradd -m vncuser



Step 4) 切換權限到vncuser

[root]$ su vncuser



Step 5) 指定vncuser的編號 (此處編號設定1),故登入時要使用5900+1=5901 port

[vncuser]$ vncserver :1



Step 6) 設定vncuser登入時的密碼

[vncuser]$ vncpasswd



Step 7) 設定vncuser登入時使用Gnome圖形介面 (此部驟在CentOS 6.2不用做)

[vncuser]$ vim ~/.vnc/xstartup
------------------------------------------------------------
unset SESSION_MANAGER #把最前面的註解拿掉
exec /etc/X11/xinit/xinitrc  #把最前面的註解拿掉

[ -x /etc/vnc/xstartup ] && exec /etc/vnc/xstartup
[ -r $HOME/.Xresources ] && xrdb $HOME/.Xresources
xsetroot -solid grey
vncconfig -iconic &
xterm -geometry 80x24+10+10 -ls -title "$VNCDESKTOP Desktop" &
twm &
------------------------------------------------------------

[vncuser]$ exit



Step 8) 此時可以看到5901 port已在使用,並且也啟動了相關的ports

[root]$ netstat -tulnp | grep X
------------------------------------------------------------
tcp   0   0   0.0.0.0:5901   0.0.0.0:*   LISTEN   13945/Xvnc
tcp   0   0   0.0.0.0:6001   0.0.0.0:*   LISTEN   13945/Xvnc
tcp   0   0           :::6001            :::*   LISTEN   13945/Xvnc
------------------------------------------------------------



Step 9) 開啟防火牆的5091 port

[root]$ iptables -A INPUT -i eth0 -s 0.0.0.0 -p tcp --dport 5901 -j ACCEPT
[root]$ /etc/init.d/iptables save



Step 10) 若要讓系統重開機後vnc server會自動啟動的話,請執行:

[root]$ chkconfig vncserver on





二、額外新增VNC Server的登入帳號

Step 11) 我們來新增一個mis的帳號吧!

[root]$ useradd -m mis



Step 12) 修改vncserver config檔! 讓每次重新啟動OS會自行開啟ports讓使用VNC的使用者連線

[root]$ vim /etc/sysconfig/vncservers
------------------------------------------------------------
#指定使用者mis要開啟的編號 (此處編號設定2)
VNCSERVERS="1:vncuser 2:mis"
#以vnc server預設的基底port為5900,再加上編號2
#所以我們會開啟5902這個port給mis這個權限使用!

VNCSERVERARGS[2]="-geometry 1024x768"
------------------------------------------------------------



Step 13) 切換權限到mis

[root]$ su mis



Step 14) 指定mis的編號 (此處編號設定2),故登入時要使用5900+2=5902 port

[mis]$ vncserver :2



Step 15) 設定mis登入時的密碼

[mis]$ vncpasswd



Step 16) 設定mis登入時使用Gnome圖形介面 (此部驟在CentOS 6.2不用做)

[mis]$ vim ~/.vnc/xstartup
------------------------------------------------------------
unset SESSION_MANAGER #把最前面的註解拿掉
exec /etc/X11/xinit/xinitrc  #把最前面的註解拿掉

[ -x /etc/vnc/xstartup ] && exec /etc/vnc/xstartup
[ -r $HOME/.Xresources ] && xrdb $HOME/.Xresources
xsetroot -solid grey
vncconfig -iconic &
xterm -geometry 80x24+10+10 -ls -title "$VNCDESKTOP Desktop" &
twm &
------------------------------------------------------------

[mis]$ exit



Step 17) 開啟防火牆的5092 port

[root]$ iptables -A INPUT -i eth0 -s 0.0.0.0 -p tcp --dport 5902 -j ACCEPT
[root]$ /etc/init.d/iptables save





三、移除VNC的登入帳號

Step 18) 我們來把剛剛新增的mis的帳號移除吧!

[root]$ userdel -r mis



Step 19) 將mis所屬的編號給移除,此時5902 port就無法使用VNC連線了。

[root]$ vncserver -kill :2



Step 20) 修改vncserver config檔! 移除mis這個vnc的登入port

[root]$ vim /etc/sysconfig/vncservers
------------------------------------------------------------
#我們移除了2:mis
VNCSERVERS="1:vncuser"
VNCSERVERARGS[2]="-geometry 1024x768"
------------------------------------------------------------



Step 21) 移除防火牆的5092 port

[root]$ iptables -D INPUT -i eth0 -s 0.0.0.0 -p tcp --dport 5902 -j ACCEPT
[root]$ /etc/init.d/iptables save

Groupinstall Xfce安装失败




今天博主按照以前介绍的《在CentOS中搭建远程桌面管理》的步骤,在CentOS中搭建远程桌面。在连接的时候碰到一个问题,使用TightVNC客户端连接后页面一直显示黑屏,鼠标显示黑色叉叉。xstartup中的配置和权限设置也是正确的。

这时博主开始找原因。首先查看/root/.vnc中的log文件,提示如下错误“/root/.vnc/xstartup: line 2: /usr/bin/startxfce4: No such file or directory”。startxfce4这个文件没有找到,说明xfce可能安装失败了。然后回到《在CentOS中搭建远程桌面管理》介绍中的第一步:安装Xfce 4.4。输入yum groupinstall xfce-4.4命令,运行后提示如下信息:

Loaded plugins: fastestmirror
Loading mirror speeds from cached hostfile
base: centos.sonn.com
extras: mirror.stanford.edu
updates: mirror.5ninesolutions.com
Setting up Group Process
Checking for new repos for mirrors
Warning: Group xfce does not exist.
No packages in any requested group available to install or update

输入yum grouplist,发现列表中也没有xfce,但是yum search xfce的时候可以看到很多包。所以最后博主就试着使用yum install xfce*安装。安装完后登录正常。。。

在CentOS中搭建远程桌面管理:xfce+VNC


Linux的安全与性能向来为开发者所称道,你可以轻松地在搜索引擎中找到各种Linux优越性的说辞,其中不乏Linux的激进者。特别是当你步入VPS领域,更多地接触Linux之后,你会更加觉得这个开源操作系统的强大之处!本文将简单带领大家在Linux VPS上安装桌面系统Xfce,并使用VNC对其进行图形界面的使用与管理。

Xfce与KDE,Gnome一样,都是可视化的桌面前端,其特点是占用资源更小。根据网上言论,资源占用情况大致为Xfce<KDE<Gnome。
第一步:安装Xfce 4.4
yum groupinstall xfce-4.4
第二步:安装VNC
yum install vnc vnc-server
第三步:配置VNC
1.修改配置文件
vi /etc/sysconfig/vncservers
加入以下内容:
VNCSERVERS=”1:root”
VNCSERVERARGS[1]=”-geometry 800×600″
2.设置VNC密码
vncpasswd
3.启动VNC服务(很多朋友在使用DirectSpace默认的桌面VNC的时候,遇到无法连接“10061错误”,即可在ssh下输入下面命令解决!)
vncserver
4.修改vnc文件
vi /root/.vnc/xstartup
将文件内容替换为以下内容
#!/bin/sh
/usr/bin/startxfce4
5.设置vnc权限
chmod +x ~/.vnc/xstartup
6.重启VNC服务
service vncserver restart
7.设置VNC开机启动
chkconfig vncserver on
第四步:使用TightVNC登录
下载地址:http://115.com/file/bew5lqah (由于115网盘关闭大众分享功能,推荐使用下载二下载)下载二 ;
打开软件,输入ip:1(自己设置的数字:VNCSERVERS=”1:root”),然后输入密码就可以登录了。

android 下拉列表(Spinner)的学习与应用


第一步,还是先新建一个工程
第二步。修改已经生成的res/layout/main.xml。整体替换为:

[java] view plaincopy
  1. <?xml version="1.0" encoding="utf-8"?>      
  2. <LinearLayout      
  3.     android:id="@+id/widget28"      
  4.     android:layout_width="fill_parent"      
  5.     android:layout_height="fill_parent"      
  6.     android:orientation="vertical"      
  7.     xmlns:android="http://schemas.android.com/apk/res/android" >      
  8.     <TextView      
  9.         android:id="@+id/TextView_Show"      
  10.         android:layout_width="fill_parent"      
  11.         android:layout_height="wrap_content"      
  12.         android:text="你选择的是"      
  13.         android:textSize="25sp">      
  14.     </TextView>      
  15.     <Spinner      
  16.         android:id="@+id/spinner_City"      
  17.         android:layout_width="fill_parent"      
  18.         android:layout_height="wrap_content">      
  19.     </Spinner><!-- 定义一个下拉菜单-->      
  20. </LinearLayout>      


第三步。添加关键代码


[java] view plaincopy
  1. import java.util.ArrayList;       
  2. import java.util.List;       
  3. import android.app.Activity;       
  4. import android.os.Bundle;       
  5. import android.view.MotionEvent;       
  6. import android.view.View;       
  7. import android.widget.AdapterView;       
  8. import android.widget.ArrayAdapter;       
  9. import android.widget.Spinner;       
  10. import android.widget.TextView;       
  11.       
  12. public class spinner extends Activity {       
  13.     /** Called when the activity is first created. */      
  14.     private List<String> list = new ArrayList<String>();       
  15.     private TextView myTextView;       
  16.     private Spinner mySpinner;       
  17.     private ArrayAdapter<String> adapter;       
  18.     
  19.     @Override      
  20.     public void onCreate(Bundle savedInstanceState) {       
  21.         super.onCreate(savedInstanceState);       
  22.         setContentView(R.layout.main);       
  23.         //第一步:添加一个下拉列表项的list,这里添加的项就是下拉列表的菜单项       
  24.         list.add("北京");       
  25.         list.add("上海");       
  26.         list.add("深圳");       
  27.         list.add("南京");       
  28.         list.add("重庆");       
  29.         myTextView = (TextView)findViewById(R.id.TextView_Show);       
  30.         mySpinner = (Spinner)findViewById(R.id.spinner_City);       
  31.         //第二步:为下拉列表定义一个适配器,这里就用到里前面定义的list。       
  32.         adapter = new ArrayAdapter<String>(this,android.R.layout.simple_spinner_item, list);       
  33.         //第三步:为适配器设置下拉列表下拉时的菜单样式。       
  34.         adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);       
  35.         //第四步:将适配器添加到下拉列表上       
  36.         mySpinner.setAdapter(adapter);       
  37.         //第五步:为下拉列表设置各种事件的响应,这个事响应菜单被选中       
  38.         mySpinner.setOnItemSelectedListener(new Spinner.OnItemSelectedListener(){       
  39.             @SuppressWarnings("unchecked")    
  40.             public void onItemSelected(AdapterView arg0, View arg1, int arg2, long arg3) {       
  41.                 // TODO Auto-generated method stub       
  42.                 /* 将所选mySpinner 的值带入myTextView 中*/      
  43.                 myTextView.setText("您选择的是:"+ adapter.getItem(arg2));       
  44.                 /* 将mySpinner 显示*/      
  45.                 arg0.setVisibility(View.VISIBLE);       
  46.             }       
  47.             @SuppressWarnings("unchecked")    
  48.             public void onNothingSelected(AdapterView arg0) {       
  49.                 // TODO Auto-generated method stub       
  50.                 myTextView.setText("NONE");       
  51.                 arg0.setVisibility(View.VISIBLE);       
  52.             }       
  53.         });       
  54.         /*下拉菜单弹出的内容选项触屏事件处理*/      
  55.         mySpinner.setOnTouchListener(new Spinner.OnTouchListener(){       
  56.             public boolean onTouch(View v, MotionEvent event) {       
  57.                 // TODO Auto-generated method stub       
  58.                 /* 将mySpinner 隐藏,不隐藏也可以,看自己爱好*/      
  59.                 v.setVisibility(View.INVISIBLE);       
  60.                 return false;       
  61.             }       
  62.         });       
  63.         /*下拉菜单弹出的内容选项焦点改变事件处理*/      
  64.         mySpinner.setOnFocusChangeListener(new Spinner.OnFocusChangeListener(){       
  65.         public void onFocusChange(View v, boolean hasFocus) {       
  66.         // TODO Auto-generated method stub       
  67.             v.setVisibility(View.VISIBLE);       
  68.         }       
  69.         });       
  70.     }       
  71. }    

小结:
第一步:添加一个下拉列表项的list,这里添加的项就是下拉列表的菜单项  
第二步:为下拉列表定义一个适配器,这里就用到里前面定义的list。  
第三步:为适配器设置下拉列表下拉时的菜单样式。  
第四步:将适配器添加到下拉列表上  
第五步:为下拉列表设置各种事件的响应,这个事响应菜单被选中  

常用的android弹出对话框


我们在平时做开发的时候,免不了会用到各种各样的对话框,相信有过其他平台开发经验的朋友都会知道,大部分的平台都只提供了几个最简单的实现,如果我们想实现自己特定需求的对话框,大家可能首先会想到,通过继承等方式,重写我们自己的对话框。当然,这也是不失为一个不错的解决方式,但是一般的情况却是这样,我们重写的对话框,也许只在一个特定的地方会用到,为了这一次的使用,而去创建一个新类,往往有点杀鸡用牛刀的感觉,甚至会对我们的程序增加不必要的复杂性,对于这种情形的对话框有没有更优雅的解决方案呢? 
    幸运的是,android提供了这种问题的解决方案,刚开始接触android的时候,我在做一个自定义对话框的时候,也是通过继承的方式来实现,后来随着对文档了解的深入,发现了android起始已经提供了相应的接口Dialog Builder ,下面我就吧相关的内容在这里分享一下,也能让更多的初学者少走弯路。


首先是一个最简单的应用,就是弹出一个消息框,在android中可以这样实现
new AlertDialog.Builder(self)    
                .setTitle("标题")  
                .setMessage("简单消息框")  
                .setPositiveButton("确定", null)  
                .show();  


效果如下:

1.png


上面的代码中我们新建了一个AlertDialog,并用Builder方法形成了一个对象链,通过一系列的设置方法,构造出我们需要的对话框,然后调用show方法显示出来,注意到Builder方法的参数 self,这个其实是Activity对象的引用,根据你所处的上下文来传入相应的引用就可以了。例如在onCreate方法中调用,只需传入this即可。下面是带确认和取消按钮的对话框

new AlertDialog.Builder(self)   
.setTitle("确认")  
.setMessage("确定吗?")  
.setPositiveButton("是", null)  
.setNegativeButton("否", null)  
.show();  



2.png

注意到,这里有两个null参数,这里要放的其实是这两个按钮点击的监听程序,由于我们这里不需要监听这些动作,所以传入null值简单忽略掉,但是实际开发的时候一般都是需要传入监听器的,用来响应用户的操作。

下面是一个可以输入文本的对话框


new AlertDialog.Builder(self)  
.setTitle("请输入")  
.setIcon(android.R.drawable.ic_dialog_info)  
.setView(new EditText(self))  
.setPositiveButton("确定", null)  
.setNegativeButton("取消", null)  

.show();  3.png


如上代码,我们用setView方法,为我们的对话框传入了一个文本编辑框,当然,你可以传入任何的视图对象,比如图片框,WebView等。。尽情发挥你的想象力吧~:lol

下面是单选框与多选框,也是非常有用的两种对话框

new AlertDialog.Builder(self)  
.setTitle("请选择")  
.setIcon(android.R.drawable.ic_dialog_info)                  
.setSingleChoiceItems(new String[] {"选项1","选项2","选项3","选项4"}, 0,   
  new DialogInterface.OnClickListener() {  
                              
     public void onClick(DialogInterface dialog, int which) {  
        dialog.dismiss();  
     }  
  }  
)  
.setNegativeButton("取消", null)  
.show();         



4.png
                 

new AlertDialog.Builder(self)  
.setTitle("多选框")  
.setMultiChoiceItems(new String[] {"选项1","选项2","选项3","选项4"}, null, null)  
.setPositiveButton("确定", null)                  
.setNegativeButton("取消", null)  
.show();  
            

多选对话框

单选和多选对话框应该是我们平时用的非常多的,代码应该很好理解,下面再最后介绍两个、

列表对话框
new AlertDialog.Builder(self)  
.setTitle("列表框")  
.setItems(new String[] {"列表项1","列表项2","列表项3"}, null)  
.setNegativeButton("确定", null)  
.show();  


6.png


最后,在对话框中显示图片
ImageView img = new ImageView(self);  


img.setImageResource(R.drawable.icon);  
new AlertDialog.Builder(self)  
.setTitle("图片框")  
.setView(img)  
.setPositiveButton("确定", null)  
.show();  


7.png

      我们传入了一个ImageView来显示图片,这里显示了一个经典的android小绿人图标~ ~,当然这里还可以放上网络图片,具体的实现方法就不介绍了,留给大家来练习吧~:lol

      最后总结一下,android平台为我们开发提供了极大的便利,DialogBuilder能做的不止这些,这里给大家展示的只是冰山一角,我们可以尽情的发挥想象,创造我们自己的对话框。

修改phpcms文件夹名称,修改index.php,修改后台登录路径



第一次发的PHPCMS教程,这个应该很多人需要。弄完以后安全性提高了很多。修改前记得备份。
修改phpcms文件夹名称方法(例如把phpcms修改成 test)
找到根目录 api.php ,
include PHPCMS_PATH.'phpcms/base.php';
修改成
include PHPCMS_PATH.'test/base.php';
找到根目录 index.php ,
include PHPCMS_PATH.'/phpcms/base.php';
修改成
include PHPCMS_PATH.'/test/base.php';
修改index.php文件名方法。(例如修改成 test.php)
批量替换phpcms程序的文件: 查找 index.php 替换成 test.php(这个步骤应该不适用所有做过二次开发的。自己斟酌,看下是否有哪些文件里面的index.php不是根目录的)
修改后台登录的地址。 (例如修改成 localhost/admin)
.htaccess 文件加这句(链接重写的教程百度)
RewriteRule ^admin$ test.php?m=admin&c=index&a=login
如果想让localhost/admin 作为唯一登陆口
在根目录 index.php 添加一下代码
if ($_SERVER['REQUEST_URI']=='/index.php?m=admin' or
$_SERVER['REQUEST_URI']=='/index.php?m=admin&c=index' or
$_SERVER['REQUEST_URI']=='/index.php?m=admin&c=index&a=login' or
preg_replace('/hash=(.*)/','hash=no',$_SERVER['REQUEST_URI'])=='/index.php?m=admin&c=index&a=login&pc_hash=no'
){Header("Location: http://修改成首页链接或者 404 页面");}

PHPCMS短信接口



更新时间:2012-6-1
版本:2.0
短信平台网站: http://sms.phpcms.cn/
所有接口同时支持get / post二种方式获取:
发送短信接口:
http://sms.phpcms.cn/api.php?op=sms_service&sms_uid=$sms_uid&sms_pid=$sms_pid&sms_passwd=$sms_passwd&mobile=$mobile&send_txt=".urlencode($send_txt)."&charset=".CHARSET 
2012.12.15日起,将使用新的发送短信接口:
http://sms.phpcms.cn/api.php?op=sms_service&sms_uid=$sms_uid&sms_pid=$sms_pid&sms_passwd=$sms_passwd&mobile=$mobile&send_txt=".urlencode($send_txt)."&charset=".CHARSET."&tplid=".$tplid
重点说明一下,新接口中$send_txt 和 $tplid 参数

参数
含义
备注
$send_txt
短信内容
新接口根据使用短信模版所包含的参数数量进行组合短信内容,举例如下:

短信模版样式
欢迎您:___,你的验证码是___,请于五分钟内完成验证!

$send_txt参数为:
小王||334466

也就是说将参数使用|| 符号连接起来传递。
$tplid
短信模版ID
可与网站后台发送短信处查看模版ID,具体请看图一所示。


正常返回结果:0#9573

#号前状态码,#号之后,返回短信平台的查询id

代码
说明
0
成功
1
手机号码非法
2
用户存在于黑名单列表
3
接入用户名或密码错误
4
产品代码不存在
5
IP非法
6
源号码错误
7
调用网关错误
8
消息长度超过60
9
发送短信内容参数为空
10
用户已主动暂停该业务
11
wap链接地址或域名非法


错误返回状态:
-1
单个号码每分钟发送短信数量超过限制3 
-2
API mobile phone error
-11
帐号验证失败
-10
SNDA接口没有返回结果
-12
剩余短信数不足以发送
-15
产品已停用
-16
因发垃圾信息该产品被停用
-17
内容包含敏感词
-18
全英文140,带中文60,长短信全英文420,长短信带中文180
-19
该运营商暂停短信发送
-20
自定义签名长度超过规定长度
-40
接口返回异常,请联系管理员
  
获取短信产品列表信息:
获取短信剩余条数: 
返回值:
正常,{"surplus":"987654","allow_send_ip":["192.168.1.2","114.251.167.194"]}
帐号验证失败返回:{"msg":"-1","tips":"验证失败"}

获取充值记录: 
最多返回:30条 
http://sms.phpcms.cn/api.php?op=sms_get_paylist&sms_uid=$sms_uid&sms_pid=$sms_pid&sms_passwd=$sms_passwd 
返回值:
正常,
0 =>array(
         'productid'=> 1,
         'price'=> 10,
         'totalnum'=> 100,
         'name'=>'10元套餐',
         'give_away'=>0,
         'description'=>'充50元送10条短信',
         'recharge_time'=>'2121212121',
         ),

帐号验证失败返回:{"msg":"-1","tips":"验证失败"} 
获取消费记录:
默认:返回20条记录
$page 当前分页
返回数组:tatal 总条数
列表:datas