`
ddbl7
  • 浏览: 11711 次
  • 性别: Icon_minigender_1
  • 来自: 北京
最近访客 更多访客>>
社区版块
存档分类
最新评论

fried cake server ----我的油炸糕(1)

阅读更多
fried cake server(油炸糕) 是我前一段时间比较闲地时候,自己动手写的一个服务器,目标是做成类似于jboss一样的application server。不过目前做了几个月,还只是个不支持tagLib的jsp容器而已,好在支持jndi和datasource,加上不完善的jta……勉强能看得过……
没办法,上班也忙,下班也忙……断断续续写了好久,也仅仅是个雏形,一个人的力量毕竟是有限的
有兴趣的朋友们可以访问以下网址:
https://sourceforge.net/projects/freadcakeserver/
cvs 地址host:  freadcakeserver.cvs.sourceforge.net
       path:/cvsroot/freadcakeserver
可以匿名访问
在今后的日子里,我会逐渐的把代码贴出来,一边反省,一边重构……当前版本alpha build 1004,完善中……
今天是第一贴,放出两个核心类server.MyServer和server.SocketProcess
其实就是一个多路通道加上一个执行线程。
本来应该先发个设计图,可是类图经常变化,以后再发吧
/**
 * 整个服务器的核心,本身也是一个静态多路服务器的骨架
 * @author 刘宇航
 */
package server;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.util.Calendar;
import java.util.Iterator;
import java.util.Set;

import util.Constant;

public class MyServer
{
	public static final int setTimeOut = 8000;

	private  int port = 80;//默认端口

	Selector selector;
	ByteBuffer buffer = ByteBuffer.allocate(1024 * 8);

	/**
	 * 初始化一个server
	 * @throws Exception
	 */
	public  MyServer() throws Exception
	{
		init();
	}
	/**
	 * 实际初始化一个server
	 * 包括webLoader加载、解析appRuntime、注册通道等工作
	 * @throws Exception
	 */
	public void init() throws Exception
	{
		WebLoader webLoader = WebLoader.getInstance();
		webLoader.loadAll();// 这里已经隐含着将appRuntime初始化了
		port = (Integer)AppRunTime.getInstance().getProperty(Constant.SERVER_PORT); 
		ServerSocketChannel channel = ServerSocketChannel.open();
		channel.configureBlocking(false);
		channel.socket().bind(new InetSocketAddress(port));
		selector = Selector.open();
		channel.register(selector, SelectionKey.OP_ACCEPT);
	}

	/**
	 * 请不要误会,这并不是一个多线程的run()方法
	 * 另外,server只接受isAcceptable为真的请求
	 * 单向的消息请求暂不提供
	 * @throws Exception
	 */
	public void run() throws Exception
	{
		while (true)
		{
			int num = selector.select();
			if (num != 0)
			{
				Set<SelectionKey> set = selector.selectedKeys();
				Iterator<SelectionKey> it = set.iterator();

				while (it.hasNext())
				{
					SelectionKey key = it.next();

					if (key.isAcceptable())
					{
						accept((ServerSocketChannel) key.channel());
					}
					it.remove();
				}
			}
		}

	}
	
	
    /**
     * 对于isAcceptable的请求启动一个SoketProcess线程
     * @param serverSocket
     * @throws IOException
     */
	protected void accept(ServerSocketChannel serverSocket) throws IOException
	{
		SocketProcess pc = new SocketProcess(serverSocket);
		/*多线程模式---发布用*/
		pc.start();
		/*单线程模式---开发用*/
		//pc.run();
	}

    /**
     * 启动命令,发布版里使用bootstrap项目调用这个main方法,参数设置也挪至bootstrap项目中
      * @param args 暂无 待扩充
      * @throws Exception
     */
	public static void main(String args[]) throws Exception
	{
		long time = Calendar.getInstance().getTimeInMillis();
		MyServer server = new MyServer();
		time = Calendar.getInstance().getTimeInMillis()-time;
		System.out.println("服务器这正在监听端口"+AppRunTime.getInstance().getProperty(Constant.SERVER_PORT)+".....");
		System.out.println("启动成功,总耗时"+time+"毫秒.....");
		server.run();
	}
}

// end


package server;

import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;

import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;

import project.ProjectInfo;
import util.Constant;
import util.HttpError;
import util.ServerMessage;
import webHttp.HttpRequestImpl;
import webHttp.HttpResponseImpl;
import webHttp.RequestHanler;
import webHttp.SessionPool;
import exceptions.HttpException;
import executor.BaseExecutor;
import executor.Executor;
/**
 * 双线通信线程
 * 解释: 双线通信指,既接收请求,又返回响应
 * 相对应的还有 单向发送通讯 和 单向接受通讯
 * 后两个可以用于jms等功能的实现
 * 目前尚未考虑实现
 * @author 刘宇航
 *
 */
public class SocketProcess extends Thread {
	ServerSocketChannel serverSocket = null;

	/**
	 * 初始化一个线程
	 * @param serverSocket 必须保证双线通信
	 */
	public SocketProcess(ServerSocketChannel serverSocket) {
		this.serverSocket = serverSocket;
	}

	/**
	 * 从serverSocket中获取数据
	 * 然后判断是资源文件还是servlet
	 * 最有发送正确的信息给客户端
	 */
	public void run() {
		ByteBuffer buffer = ByteBuffer.allocate(1024 * 8);
		SocketChannel channel = null;
			// 一些读取数据的工作
			try{
			channel = serverSocket.accept();
			if (channel == null || channel.read(buffer) <= 0) return;
			}catch(Exception e){e.printStackTrace();return;}
			//初始化元素
			String requestInfo = new String(buffer.array(), 0, buffer.position());
			HttpRequestImpl request = RequestHanler.getHttpRequest(requestInfo);
			HttpServletResponse response = new HttpResponseImpl(channel);
			//项目解析
			AppRunTime runtime = AppRunTime.getInstance();
			String[] uri = request.getContextPath().split(Constant.WEB_SP);
			ProjectInfo project = runtime.getProjectMap().get(uri[1]);
	
			if(project==null)
			{
				sendExceptionMessage(channel,new HttpException(HttpError.ERROR_404));
				closeChannel(buffer,channel);
			}
			else
			{
			SessionPool sessionPool = SessionPool.getInstance(project.getId());
			HttpSession session = sessionPool.getSession(channel.socket().getInetAddress().getHostAddress());
			request.bind(session);
			// 执行资源
			try {
			Executor exe  = new BaseExecutor(project);
			exe.forward(request, response);		
			} 
			//返回错误到页面
			catch (Exception e) {
				sendExceptionMessage(channel,e);
			}
			// 至少要反回空串
			closeChannel(buffer,channel);
			}
		
	}
	/**
	 * 追加一个空格并关闭通道
	 * @param buffer
	 * @param channel
	 */
	private void closeChannel(ByteBuffer buffer, SocketChannel channel){
		buffer.clear();
		if (!channel.isOpen()) return ;
		buffer.put(" ".getBytes());
		buffer.flip();
		try {
			channel.write(buffer);
			channel.finishConnect();
			channel.close();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}

	/**
	 * 向客户端浏览器报告错误,比如error404 page not found之类的
	 * @param channel
	 * @param message
	 */
	private void sendExceptionMessage(SocketChannel channel ,Exception message){
		/*创建缓冲区*/
		ByteBuffer buffer = ByteBuffer.allocate(1024 * 8);
		/*获得错误消息堆栈*/
		StackTraceElement[]stackTraces =  message.getStackTrace();
		/*设置消息头*/
		buffer.put(message.toString().getBytes());
		/*设置堆栈体*/
		for(StackTraceElement trace:stackTraces)
		{
			buffer.put(trace.toString().getBytes());
			buffer.put(Constant.STR_NEXT_LINE.getBytes());
		}
		/*发送到浏览器*/
		try {
		buffer.flip();
		channel.write(buffer);
		} catch (IOException e) {
			System.err.println(ServerMessage.connection_exception);
		}
		buffer.clear();
		/*本机报错*/
		message.printStackTrace();
	}
}

// end

呵呵,先发这么多,以后的慢慢补上
0
0
分享到:
评论

相关推荐

    Fried Cake Application Server-开源

    作为一个处于开发和测试阶段的小型容器,Fried Cake Server为开发者提供了在JSP(JavaServer Pages)环境中的部署和管理应用的新选择。 J2EE,全称Java 2 Platform, Enterprise Edition,是由Oracle公司主导的Java...

    台湾小吃英文名称(Word档含图文).doc

    - 萝卜糕:Turnip Cake 或 Fried White Radish Patty - 盐鸭蛋:Salted Duck Egg - 锅贴:Fried Dumpling - 馍头:Steamed Bun - 葱油饼:Spring Onion Pancake - 饺子:Dumpling - 稀饭(粥):Rice ...

    transLucid (was Lucid Fried Eggs)-开源

    "transLucid (was Lucid Fried Eggs)" 这个标题表明这是一个名为 "transLucid" 的项目,之前可能被称为 "Lucid Fried Eggs"。这可能是因为项目经历了改名或更新,通常这样的变化伴随着软件的发展或品牌重塑。"开源" ...

    qtip:简单的文本显示服务

    QTip QTip是通过命令行管理的自托管... qtip serve /recipe/fried-chicken fried-chicken-recipe.md 产品特点 Markdown文件支持 自动将markdown中嵌入的本地图像上传和存储到您的首选CDN 100%自托管 :rocket: 演示版

    Rework - Jason Fried.rar

    GOMake a dent in the universeScratch your own itchStart making somethingNo time is no excuseDraw a line in the sandMission statement impossibleOutside money is Plan ZYou need less than you thinkStart ...

    Fried衰弱评估方法.pdf

    Fried衰弱评估方法.pdf

    Fried Rice-开源

    自动正确地回答来自www.freerice.com的问题,从而向联合国提供了大量大米,以养活世界。

    高二烹饪班英语毕业考试题.docx

    5. 煎蛋火鸡 - Fried egg with turkey 6. 饺子 - Dumplings 7. 甜品 - Dessert 8. 三明治 - Sandwich 9. 比萨 - Pizza 10. 汉堡 - Hamburger 11. 蛋糕 - Cake 12. 西瓜 - Watermelon 13. 梨 - Pear 14. 橙子 - ...

    epOS:用于Intel x86处理器的教育性操作系统

    epOS,Ramon Fried(2013-2016)的教育版英特尔8086 32位操作系统 强调 兼容Multiboot 1.0 位图页面框架分配器 高半内核 分页已启用 使用循环算法的可抢占式调度 环0-3内核与用户空间分离 交流会 APIC / IOAPIC ...

    广东省深圳市文汇中学2015届九年级英语上学期第6周周末作业(答案不全) 牛津深圳版

    - `油炸食品`: fried food - `体检`: physical examination - `减肥`: lose weight - `通常;大体上`: usually; generally - `加糖的咖啡`: sugary coffee - `a bit of`: 一点点 - `would rather do sth.`: ...

    最全版我国菜名英文翻译(数千种).doc

    1. **翻译原则**: - 以主料为主、配料为辅:如“白灵菇扣鸭掌”(Mushrooms with Duck Webs) - 以烹制方法为主、原料为辅:如“火爆腰花”(Sauteed Pig Kidney) - 以形状、口感为主、原料为辅:如“脆皮鸡”...

    国外生活必备英文词汇_饮食篇.doc

    - French fries:炸薯条,切片的土豆油炸而成。 - baked potato:烘马铃薯,整个土豆烘烤至软糯。 - mashed potatoes:马铃薯泥,捣碎的煮熟土豆。 - omelette:简蛋卷,煎熟的鸡蛋卷。 - pudding:布丁,甜点,通常...

    (课标卷)2014-2015学年高中英语 Unit 2 Healthy eating Warming Up Reading课时作

    - fried:油炸的,指通过热油烹制的食物。 - balanced:平衡的,形容饮食均衡,包含各种必要的营养成分。 - slim:苗条的,形容身材瘦削,符合现代审美标准。 2. 单项选择题: - interested:感兴趣的,用于...

    第六单元教案IlikemusicthatIcandanceto单元学案.pdf

    - 我不愿意吃太多的油炸食品:I am unwilling to eat too much fried food. - 主要是膳食平衡:mainly a balanced diet - 我同意多吃水果和蔬菜对身体好:I agree that eating more fruits and vegetables is ...

    餐厅英语情景对话-必备.doc

    - "I recommend crispy and fried duck."(我推荐香酥鸭。) 这些对话涵盖了餐厅服务的基本流程,包括迎接顾客、询问需求、推荐菜品、预订和等待座位、以及就座后的点餐过程。了解这些常用表达对于在餐厅工作的...

    fire-containment:Fried和Fried描述的火势遏制和初始攻击模型

    cbevins /消防设施一种基于1995年杰里米·弗里德(Jeremy S.Fried)和伯顿·弗雷德(Burton D.Fried)论文的火遏制算法和初始攻击模型,题为“用现实战术模拟野火遏制”。 它是根据Jeremy S.Fried(1991)的FCAT防火...

    河南省开封市第三十三中学九年级英语全册 Unit 6 I like music that I can dance to Read

    - **fry**:动词,意为“油煎,油炸”,形容词形式为**fried**,表示“油炸的,油煎的”。 示例: - 他炸了一个鸡蛋。He fried an egg. - 油炸鸡肉 fried chicken 2. **短语应用** - **be bad for**:对……...

    餐饮文化培训课程.pptx

    - 炒(stir-fried/sautéed):快速翻煎食材,如"炒鳝片"(Stir-fried eel slices) - 爆(quick-fried):快炒,通常用高温短时间烹制,如"爆炒肉丝" - 炸(deep-fried):包括软炸和酥炸,如"软炸里脊"(soft-...

    中国菜英文翻译2.doc

    * 萝卜糕(Fried white radish patty) * 芋头糕(Taro cake) 特殊饮料 * 果汁甘蔗汁(Sugar cane juice) * 酸梅汁(Plum juice) * 杨桃汁(Star fruit juice) * 青草茶(Herb juice) 特殊点心 * 当归鸭...

Global site tag (gtag.js) - Google Analytics