随笔-40  评论-66  文章-0  trackbacks-0
  2008年2月26日
springside3背景struts2.1.2 spring2.5.6 Hibernate3.4GA


1.struts2

使用ZeroConfig + CodeBehind插件,实现约定大于配置的零配置文件风格.

根本不用配置struts.xml文件





这里就是action实现annotation  CodeBehind。
如方法


通过 /user/user!input.action访问, 并转到 /user/user-xxx.jsp页面
即namespace + action name + "-" + "xxx.jsp"


 另外其中 action中需要注入的 service 使用annotation ,在set方法前 加入@request 或 @ autowired 或 @resource
注释(具体是用那个暂时无法搞清楚,总之我用request 就不行,另外两个都可以)



有了这个代码之后 就能注入spring 环境中的 id = userManager 的bean (我个人理解)

而spring中  id = userManager 的bean  也是通过自动注入完成的

主要代码是 applicationContext.xml文件中的
代码:




2.spring

spring2.5.6的annotation特性用的比较泛滥。 新手刚开始看的一头雾水很正常。

xml文件中配置自动注册bean,通过扫描包中的带注解的类。即这个代码:




扫描到下面的类,就自动注册成 id=userManager





3.hibernate
使用hiberante3 注解,不要XML配置,实体类注解不用多说。

需要注意的是entity类的扫描配置




看清楚是扫描包,不是扫描类! 所以实体类com.mylu.User是无法扫描到,要放在 com.mylu.xxx.User才能扫描到!




下边按照ss3风格做的例子,去掉spring security 框架的, 结构更清晰。

下载:实例代码


附:
类库
posted @ 2009-01-08 17:01 Super·shen BLOG 阅读(1788) | 评论 (2)编辑 收藏

在jsp中,其实jsp就是servlet,jsp和servlet也都是一个class:

1 .request.getRealPath(),这个方法已经不推荐使用,在servlet后继版本中将被取缔。

2.getServletContext().getRealPath("/")这个方法比较好用,可以直接在servlet和jsp中使用。

3.request.getSession().getServletContext().getRealPath()也可以在jsp和servlet使用。

4.this.getClass().getClassLoader().getResource("").getPath(),这个方法可以在任意jsp,servlet,java文件中使用,因为不管是jsp,servlet其实都是java程序,都是一个class。所以它应该是一个通用的方法。

posted @ 2008-09-17 14:55 Super·shen BLOG 阅读(492) | 评论 (0)编辑 收藏

普遍的,简单的权限系统要求:

1.系统所有资源定义 [资源表]   ( 还可以分为更小的权限表,操作表,这里通叫资源表)
2.定义角色 [角色表]
3.给角色指定资源(一个角色可以管理多个资源) [角色-资源表]
4.定义用户组 [用户表]
5.给用户组指定角色(一个用户组可以拥有多种角色) [用户组-角色表]
6.给用户指定角色(一个用户可以拥有多种角色,可以直接指定角色,也可以继承用户组的角色)[用户-角色表]

 


查找权限时:

根据用户ID[用户-角色表]或用户组ID[用户组-角色表],查到所有角色ID,再[角色-资源表]找到所有角色下的所有资源。

此就是用户拥有的资源。(资源一般为模块,当然也可以分更细的定义为页面,操作方法等)


此权限设计适合于模块化访问系统,如OA


当然很多系统因地而已,不可能完全满足,按照自己系统需求设计是最合适的设计。



posted @ 2008-08-18 16:17 Super·shen BLOG 阅读(793) | 评论 (0)编辑 收藏
提交页面

插入

用户:
密码:
处理页面add2.cgi 代码 #include #include #include #include "sqlite3.h" #include "cgic.h" int cgiMain() { printf("Content-type:text/html\n\n"); printf(""); sqlite3 *db=NULL; char *zErrMsg = 0; int rc; rc = sqlite3_open("test.db", &db); if(rc){ printf("Can't open database\n"); //这里改了。要是按原先的,会提示stderr未定义,我不知道为什么。哪位朋友知道一定要告诉我哦。 sqlite3_close(db); exit(1); } else printf("open test.db successfully!\n"); char username[241]; cgiFormString("username", username, 241); fprintf(cgiOut, "username:
\n");
cgiHtmlEscape(username);
fprintf(cgiOut, "
\n"); char password[241]; cgiFormString("password", password, 241); fprintf(cgiOut, "password:
\n");
cgiHtmlEscape(password);
fprintf(cgiOut, "
\n"); char sql[300]={'\0'}; //不能用指针! //插入数据 sprintf(sql, "INSERT INTO \"user\" VALUES('%s', '%s');", username,password); //sql = "INSERT INTO \"user\" VALUES('username', 'password');" ; sqlite3_exec( db , sql , 0 , 0 , &zErrMsg ); printf(sql); printf("插入数据成功!\n"); int nrow = 0, ncolumn = 0; char **azResult; //二维数组存放结果 //查询数据 /* int sqlite3_get_table(sqlite3*, const char *sql,char***result , int *nrow , int *ncolumn ,char **errmsg ); result中是以数组的形式存放你所查询的数据,首先是表名,再是数据。 nrow ,ncolumn分别为查询语句返回的结果集的行数,列数,没有查到结果时返回0 */ char *sql2 = "SELECT * FROM user"; sqlite3_get_table( db , sql2 , &azResult , &nrow , &ncolumn , &zErrMsg ); int i = 0 ; printf( "row:%d column=%d
" , nrow , ncolumn ); printf( "\nThe result of querying is : \n" ); for( i=0 ; i<( nrow + 1 ) * ncolumn ; i++ ) printf( "azResult[%d] = %s
", i , azResult[i] ); //释放掉 azResult 的内存空间 sqlite3_free_table( azResult ); sqlite3_close(db); //关闭数据库 return 0; } 请注意数据库文件 test.db的访问权限! 这里改成777!
posted @ 2008-03-01 17:11 Super·shen BLOG 阅读(1693) | 评论 (1)编辑 收藏

[转自] http://webdn.trueself.cn/archives/107

posted @ 2008-02-28 14:19 Super·shen BLOG 阅读(734) | 评论 (0)编辑 收藏

◆ 使用strtok函数分割。
     原型:char *strtok(char *s, char delim);
    strtok在s中查找包含在delim中的字符并用NULL('\0')来替换,直到找遍整个字符串。  
     功能:分解字符串为一组字符串。s为要分解的字符串,delim为分隔符字符串。
    说明:首次调用时,s指向要分解的字符串,之后再次调用要把s设成NULL。
                strtok在s中查找包含在delim中的字符并用NULL('\0')来替换,直到找遍整个字符串。
    返回值:从s开头开始的一个个被分割的串。当没有被分割的串时则返回NULL。
                  所有delim中包含的字符都会被滤掉,并将被滤掉的地方设为一处分割的节点。

使用例:
#include <stdio.h>
#include <string.h>
#include <stdio.h>
#include <string.h>

int main(int argc,char **argv)
{
    char * buf1="aaa, ,a, ,,,bbb-c,,,ee|abc";

    /* Establish string and get the first token: */
    char* token = strtok( buf1, ",-|");
    while( token != NULL )
     {
           /* While there are tokens in "string" */
           printf( "%s ", token );
          /* Get next token: */
          token = strtok( NULL, ",-|");
     }
    return 0;
}

OUT 值:
aaa

a

bbb
c
ee
abc

◆ 使用strstr函数分割。

    原型:extern char *strstr(char *haystack,char *needle);

    用法:#include <string.h>
   功能:从字符串haystack中寻找needle第一次出现的位置(不比较结束NULL)
   说明:返回指向第一次出现needle位置的指针,如果没找到则返回NULL。

使用例:
#include <stdio.h>
#include <string.h>

int main(int argc,char **argv)
{
     char *haystack="aaa||a||bbb||c||ee||";
     char *needle="||";
     char* buf = strstr( haystack, needle);
     while( buf != NULL )
     {
         buf[0]='\0';
         printf( "%s\n ", haystack);
          haystack = buf + strlen(needle);
          /* Get next token: */
          buf = strstr( haystack, needle);
     }
     return 0;
}

OUT 值:
aaa
a
bbb
c
ee

◆ strtok比较适合多个字符作分隔符的场合,而strstr适合用字符串作分隔符的场合。

posted @ 2008-02-27 16:35 Super·shen BLOG 阅读(1464) | 评论 (0)编辑 收藏
我们来看看到底如何从POST表单收集数据到CGI程序,下面給出了一個比较简单的C源代碼:     
    
 

 

#include<stdio.h>
#include<stdlib.h>    
#define MAXLEN 80    
#define EXTRA 5
/*   4个字节留给字段的名字"data",   1个字节留给"="   */
#define   MAXINPUT   MAXLEN+EXTRA+2 
/*   1个字节留给换行符,还有一个留给后面的NULL   */
#define DATAFILE "../data/data.txt"
/*   要被添加数据的文件   */

void   unencode(char   *src,   char   *last,   char   *dest)
{
 for(; src != last; src++, dest++)
  if(*src == "+")
   *dest = " ";
  else if(*src == "%") {    
   int   code;    
   if(sscanf(src+1,"%2x",&code)!=1)code="?";
   *dest=code;
   src   +=2;}
  else
   *dest=*src;
  *dest=" ";
  *++dest="";    
}    

int   main(void)    
{    
 char *lenstr;
 char input[MAXINPUT], data[MAXINPUT];
 long len;
 
 printf("%s%c%c", "Content-Type:text/html;charset=gb2312",13,10);
 printf("<TITLE>Response</TITLE>");

 lenstr=getenv("CONTENT_LENGTH");
 if(lenstr==NULL || sscanf(lenstr,"%ld",&len)!=1 || len>MAXLEN)
  printf("<P>表单提交错误");
 else{
  FILE *f;
  fgets(input,   len+1,   stdin);
  unencode(input+EXTRA,   input+len,   data);

  f =fopen(DATAFILE, "a");
  if(f == NULL)    
   printf("<P>对不起,意外错误,不能够保存你的数据");    
  else
   fputs(data,   f);    
  fclose(f);
  printf("<P>非常感谢,您的数据已经被保存<BR>%s",data);    
 }    
 return   0;    
}    

    
       从本质上来看,程序先从CONTENT_LENGTH环境变量中得到数据的字长,然后读取相应长度的字符串。因为数据内容在传输的过程中是经过了编码的,所以必须进行相应的解码。编码的规则很简单,主要的有这几条:     

1.   表单中每个每个字段用字段名后跟等号,再接上上这个字段的值来表示,每个字段之间的内容用&连结;    2.   所有的空格符号用加号代替,所以在编码码段中出现空格是非法的;    
3.   特殊的字符比如标点符号,和一些有特定意义的字符如“+”,用百分号后跟其对应的ACSII码值来表示。    

例如:如果用户输入的是:     
   
Hello   there!    

那么数据传送到服务器的时候经过编码,就变成了data=Hello+there%21   上面的unencode()函数就是用来把编码后的数据进行解码的。在解码完成后,数据被添加到data.txt文件的尾部,并在浏览其中回显出来。    

把文件编译完成后,把它改名为collect.cgi后放在CGI目录中就可以被表单调用了。下面给出了其相应的表单:    

<FORM   ACTION="/cgi-bin/collect.cgi"   METHOD="POST"   >
<P>请输入您的留言(最多80个字符):<BR>
<INPUT   NAME="data"   SIZE="60"   MAXLENGTH="80"   ><BR>
<INPUT   TYPE="SUBMIT"   VALUE="确定">
</FORM   >    
   
   
       事实上,这个程序只能作为例子,是不能够正式的使用的。它漏掉了很关键的一个问题:当有多个用户同时像文件写入数据是,肯定会有错误发生。而对于一个这样的程序而言,文件被同时写入的几率是很大的。因此,在比较正式的留言版程序中,都需要做一些更多的考虑,比如加入一个信号量,或者是借助于一个钥匙文件等。因为那只是编程的技巧问题,在这儿就不多说了。

posted @ 2008-02-27 13:52 Super·shen BLOG 阅读(2756) | 评论 (1)编辑 收藏
啥都不说,直接看代码!

简单输出代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
void main()
{
printf("Content-type:text/html\n\n");

printf("hello world!");

fflush(stdout);

}



处理get代码

#include <stdio.h>
#include <stdlib.h>
int zmain(void)
{char *data;
long m,n;
printf("%s%c%c\n","Content-Type:text/html;charset=iso-8859-1",13,10);
printf("<TITLE>Multiplication results</TITLE>\n");
printf("<H3>Multiplication results</H3>\n");
data = getenv("QUERY_STRING");
if(data == NULL) 
printf("<P>Error! Error in passing data from form to script.");
else if(sscanf(data,"m=%ld&n=%ld",&m,&n)!=2) 
printf("<P>Error! Invalid data. Data must be numeric.");
else 
printf("<P>The product of %ld and %ld is %ld.",m,n,m*n);
return 0;
}


处理post代码

#include<stdio.h>
#include<stdlib.h>
void main()
{
 int i,n;
 printf("Content-type:text/html\n\n");
 n=0;
 if(getenv("CONTENT_LENGTH"))
  n=atoi(getenv("CONTENT_LENGTH"));
 printf("%d",n);
 for(i=0;i<n;i++)
  putchar(getchar());
 putchar('\n');
 fflush(stdout);
}



还是代码


#include <stdio.h>
#include <stdlib.h>
#include <string.h>

/* 转换函数声明 */
int htoi(char *);

/*  主函数 */
void zmain() {
 int i,n;
 char c;
 printf ("Content-type: text/html\n\n");
 n=0;
 if (getenv("CONTENT_LENGTH"))
  n=atoi(getenv("CONTENT_LENGTH"));
 for (i=0; i<n;i++){
  
  int is_eq=0; //判断是否有等于号。
  
  c=getchar();
  switch(c){
  case '&':
   c='\n';
   break;
  case '+':
   c='+';
   break;
  case '%':
   {
    char s[3];
    s[0]=getchar();
    s[1]=getchar();
    s[2]=0;
    c=htoi(s);
    i+=2;
   }
   break;
  case '=':
   c='=';
   is_eq=1;
   break;
  };
  
  putchar(c);
  //if (is_eq) putchar(' ');
 }
 putchar ('\n');
 fflush(stdout);
}

/* 转换为小写 */
int islower (int ch ) 

{
    return (unsigned int) (ch - 'a') < 26u;
}


/* convert hex string to int 16进制转换成10进制 */
int htoi(char *s)

{
 
 char *digits="0123456789ABCDEF";
 
 if(islower(s[0])) s[0]=toupper(s[0]);
 if(islower(s[1])) s[1]=toupper(s[1]);
 
 return 16 * (strchr(digits, s[0]) -strchr(digits,'0') ) +(strchr(digits,s[1])-strchr(digits,'0'));
 
}


#include<stdio.h>
#include<stdlib.h>
void zzzmain()
{
 int i,n;
 printf("Content-type:text/html\n\n");
 n=0;
 if(getenv("CONTENT_LENGTH"))
  n=atoi(getenv("CONTENT_LENGTH"));
 printf("%d",n);
 for(i=0;i<n;i++)
  putchar(getchar());
 putchar('\n');
 fflush(stdout);
}

 

posted @ 2008-02-26 15:37 Super·shen BLOG 阅读(740) | 评论 (0)编辑 收藏