1.什么是ContentProvider

 数据库在Android其中是私有的,当然这些数据包含文件数据和数据库数据以及一些其它类型的数据。

 不能将数据库设为WORLD_READABLE,每一个数据库都仅仅能创建它的包訪问,

 这意味着仅仅有由创建数据库的进程可訪问它。假设须要在进程间传递数据,

 则能够使用AIDL/Binder或创建一个ContentProvider,可是不能跨越进程/包边界直接来使用数据库。

 一个Content Provider类实现了一组标准的方法接口,从而可以让其它的应用保存或读取此Content Provider的各种数据类型。

 也就是说,一个程序能够通过实现一个Content Provider的抽象接口将自己的数据暴露出去。

 外界根本看不到,也不用看到这个应用暴露的数据在应用其中是怎样存储的,或者是用数据库存储还是用文件存储,还是通过网上获得,这些一切都不重要,

 重要的是外界能够通过这一套标准及统一的接口和程序里的数据打交道,能够读取程序的数据,也能够删除程序的数据,

 当然,中间也会涉及一些权限的问题。下边列举一些较常见的接口,这些接口例如以下所看到的。

·  query(Uri uri, String[] projection, String selection, String[] selectionArgs,String sortOrder):通过Uri进行查询,返回一个Cursor。

·  insert(Uri url, ContentValues values):将一组数据插入到Uri 指定的地方。

·  update(Uri uri, ContentValues values, String where, String[] selectionArgs):更新Uri指定位置的数据。

·  delete(Uri url, String where, String[] selectionArgs):删除指定Uri而且符合一定条件的数据。

2.什么是ContentResolver

外界的程序通过ContentResolver接口能够訪问ContentProvider提供的数据,在Activity其中通过getContentResolver()能够得到当前应用的 ContentResolver实例。

 ContentResolver提供的接口和ContentProvider中须要实现的接口相应,主要有下面几个。

·  query(Uri uri, String[] projection, String selection, String[] selectionArgs,String sortOrder):通过Uri进行查询,返回一个Cursor。

·  insert(Uri url, ContentValues values):将一组数据插入到Uri 指定的地方。

·  update(Uri uri, ContentValues values, String where, String[] selectionArgs):更新Uri指定位置的数据。

·  delete(Uri url, String where, String[] selectionArgs):删除指定Uri而且符合一定条件的数据。

3.ContentProvider和ContentResolver中用到的Uri

在ContentProvider和 ContentResolver其中用到了Uri的形式通常有两种,一种是指定所有数据,还有一种是指定某个ID的数据。

我们看以下的样例。

·  content://contacts/people/  这个Uri指定的就是所有的联系人数据。

·  content://contacts/people/1 这个Uri指定的是ID为1的联系人的数据。 

在上边两个类中用到的Uri一般由3部分组成。

·  第一部分是方案:"content://" 这部分永远不变

·  第二部分是授权:"contacts"

·  第二部分是路径:"people/","people/1"(假设没有指定ID,那么表示返回所有)。

因为URI通常比較长,并且有时候easy出错,且难以理解。所以,在Android其中定义了一些辅助类,并且定义了一些常量来取代这些长字符串的使用,比例如以下边的代码:

·  Contacts.People.CONTENT_URI (联系人的URI)。

在我们的实例MyProvider中是例如以下定义的:

public static final String AUTHORITY="com.teleca.PeopleProvider";

public static final String PATH_SINGLE="people/#";

public static final String PATH_MULTIPLE="people";

public static final Uri content_URI=Uri.parse("content://"+AUTHORITY+"/"+PATH_MULTIPLE);

实例1:

文件MyProvider.java

package com.teleca.provider;

import java.util.HashMap;

import android.content.ContentProvider;

import android.content.ContentUris;

import android.content.ContentValues;

import android.content.Context;

import android.content.UriMatcher;

import android.database.Cursor;

import android.database.SQLException;

import android.database.sqlite.SQLiteDatabase;

import android.database.sqlite.SQLiteOpenHelper;

import android.database.sqlite.SQLiteQueryBuilder;

import android.net.Uri;

import android.text.TextUtils;

import android.util.Log;

public class MyProvider extends ContentProvider {

 public static final String MIME_DIR_PREFIX="vnd.android.cursor.dir";

 public static final String MIME_ITEM_PREFIX="vnd.android.cursor.item";

 public static final String MIME_ITEM="vnd.msi.people";

 public static final String MIME_TYPE_SINGLE=MIME_ITEM_PREFIX+"/"+MIME_ITEM;

 public static final String MIME_TYPE_MULTIPLE=MIME_DIR_PREFIX+"/"+MIME_ITEM;

 public static final String AUTHORITY="com.teleca.PeopleProvider";

 public static final String PATH_SINGLE="people/#";

 public static final String PATH_MULTIPLE="people";

 public static final Uri content_URI=Uri.parse("content://"+AUTHORITY+"/"+PATH_MULTIPLE);

 public static final String DEFAULT_SORT_ORDER="name DESC";

 public static final String _ID="_id";

 public static final String NAME="name";

 public static final String PHONE="phone";

 public static final String AGE="age";

 public static final int PEOPLE=1;

 public static final int PEOPLES=2;

 private static UriMatcher URI_MATCHER;

 private static HashMap<String,String> PROJECTION_MAP;

 public static String DB_NAME="peopledb";

 public static String DB_TABLE_NAME="people";

 SQLiteDatabase db;

 DBOpenHelper dbOpenHelper;

 static 

 {

  URI_MATCHER=new UriMatcher(UriMatcher.NO_MATCH);

  URI_MATCHER.addURI(AUTHORITY, PATH_MULTIPLE, PEOPLES);

  URI_MATCHER.addURI(AUTHORITY, PATH_SINGLE, PEOPLE);

  PROJECTION_MAP=new HashMap<String,String>();

  PROJECTION_MAP.put(_ID, "_id");

  PROJECTION_MAP.put(NAME, "name");

  PROJECTION_MAP.put(PHONE, "phone");

  PROJECTION_MAP.put(AGE, "age");

 }

 @Override

 public int delete(Uri uri, String selection, String[] selectionArgs) {

  // TODO Auto-generated method stub

  int count=0;

  switch(URI_MATCHER.match(uri))

  {

  case PEOPLES:

   count=db.delete(DB_TABLE_NAME,  selection, selectionArgs);

   break;

  case PEOPLE:

   String segment =uri.getPathSegments().get(1);

   String where="";

   if(!TextUtils.isEmpty(selection))

   {

    where=" AND ("+selection+")";

   }

   count=db.delete(DB_TABLE_NAME, "_id="+segment+where, selectionArgs);

   break;

  default:

   throw new IllegalArgumentException("Unkonw URI"+uri);

  }

  getContext().getContentResolver().notifyChange(uri, null);//@2

  return count;

 }

 @Override

 public String getType(Uri uri) {

  // TODO Auto-generated method stub

  switch(URI_MATCHER.match(uri))

  {

  case PEOPLES:

   return MIME_TYPE_MULTIPLE;

  case PEOPLE:

   return MIME_TYPE_SINGLE;

  default:

   throw new IllegalArgumentException("Unkown URI "+uri);

  }

 }

 @Override

 public Uri insert(Uri uri, ContentValues values) {

  // TODO Auto-generated method stub

  long rowId=0L;

  if(URI_MATCHER.match(uri)!=PEOPLES)

  {

   throw new IllegalArgumentException("Unkown URI"+uri);

  }

  rowId=db.insert(DB_TABLE_NAME, null, values);

  if(rowId>0)

  {

   Uri result=ContentUris.withAppendedId(content_URI, rowId);

   getContext().getContentResolver().notifyChange(result, null);//@2

   return result;

  }

  else

   throw new SQLException("Failed to insert row into "+uri);

 }

 @Override

 public boolean onCreate() {

  // TODO Auto-generated method stub

  dbOpenHelper=new DBOpenHelper(this.getContext(),DB_NAME,1);

  db=dbOpenHelper.getWritableDatabase();

  return false;

 }

 @Override

 public Cursor query(Uri uri, String[] projection, String selection,

   String[] selectionArgs, String sortOrder) {

  // TODO Auto-generated method stub

  SQLiteQueryBuilder queryBuilder=new SQLiteQueryBuilder();

  queryBuilder.setTables(DBInfo.DB_TABLE_NAME);

  queryBuilder.setProjectionMap(PROJECTION_MAP);

  switch(URI_MATCHER.match(uri))

  {

  case PEOPLES:

   break;

  case  PEOPLE:

   queryBuilder.appendWhere("_id="+uri.getPathSegments().get(1));

   break;

  default:

   throw new IllegalArgumentException("Unkonw URI"+uri);

  }

  String orderBy=null;

  if(TextUtils.isEmpty(sortOrder))

  {

   orderBy=DEFAULT_SORT_ORDER;

  }

  else

   orderBy=sortOrder;

  Cursor c=queryBuilder.query(db, projection, selection, selectionArgs, null, null, orderBy);

  c.setNotificationUri(getContext().getContentResolver(), uri);//@1

  return c;

 }

 @Override

 public int update(Uri uri, ContentValues values, String selection,

   String[] selectionArgs) {

  // TODO Auto-generated method stub

  int count=0;

  switch(URI_MATCHER.match(uri))

  {

  case PEOPLES:

   count=db.update(DB_TABLE_NAME, values, selection, selectionArgs);

   break;

  case PEOPLE:

   String segment =uri.getPathSegments().get(1);

   String where="";

   if(!TextUtils.isEmpty(selection))

   {

    where=" AND ("+selection+")";

   }

   count=db.update(DB_TABLE_NAME, values, "_id="+segment+where, selectionArgs);

   break;

  default:

   throw new IllegalArgumentException("Unkonw URI"+uri);

  }

  getContext().getContentResolver().notifyChange(uri, null);//@2

  return count;

 }

}

class DBOpenHelper extends SQLiteOpenHelper

{

 private static final String DB_CREATE="CREATE TABLE "

  +DBInfo.DB_TABLE_NAME

  +" (_id INTEGER PRIMARY KEY,name TEXT UNIQUE NOT NULL,"

  +"phone TEXT,age INTEGER);";

 final static String tag="hubin";

 public DBOpenHelper(Context context,String dbName,int version)

 {

  super(context,dbName,null,version);

 }

 public void onCreate(SQLiteDatabase db)

 {

  try{

   db.execSQL(DB_CREATE);

  }

  catch(SQLException e )

  {

   Log.e(tag,"",e);

  }

 }

 public void onOpen(SQLiteDatabase db)

 {

  super.onOpen(db);

 }

 public void onUpgrade(SQLiteDatabase db,int oldVersion,int newVersion)

 {

  db.execSQL("DROP TABLE IF EXISTS "+DBInfo.DB_TABLE_NAME);

  this.onCreate(db);

 }

}

class DBInfo

{

 public static String DB_NAME="peopledb";

 public static String DB_TABLE_NAME="people"; 

}

注意1:c.setNotificationUri(getContext().getContentResolver(), uri);

这里是把Cursor C加入到ContentResolver的监督对象组中去。

一旦有与uri相关的变化,ContentResolver就回通知Cursor C.

可能Cursor有个私有的内部类ContentObserver的实现。ContentResolver是通过该类来通知Cursor的。

public abstract void  setNotificationUri  (ContentResolver  cr, Uri  uri)

Register to watch a content URI for changes. This can be the URI of a specific data row (for example, "content://my_provider_type/23"), 

or a a generic URI for a content type.

Parameters

cr  The content resolver from the caller's context. The listener attached to this resolver will be notified.

uri  The content URI to watch. 

注意2: getContext().getContentResolver().notifyChange(uri, null)

通知数据发生了变化。

  public void  notifyChange  (Uri  uri, ContentObserver  observer)

Notify registered observers that a row was updated. To register, call registerContentObserver(). By default, CursorAdapter objects will get this notification.

Parameters

observer  The observer that originated the change, may be null 

这里为null的意思可能就是调用在ContentResolver中注冊的ContentObserver,反之则是调用參数指定的

文件People.java

package com.teleca.provider;

public class People {

 public long id;

 public String name;

 public String phone;

 public int age;

}

文件

Hello.java

package com.teleca.provider;

import java.util.ArrayList;

import java.util.List;

import android.app.Activity;

import android.content.ContentUris;

import android.content.ContentValues;

import android.database.Cursor;

import android.database.SQLException;

import android.net.Uri;

import android.os.Bundle;

import android.os.Handler;

import android.util.Log;

import android.view.View;

import android.view.View.OnClickListener;

import android.widget.Button;

public class Hello extends Activity {

    /** Called when the activity is first created. */

 final static String tag="hubin";

    @Override

    public void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.main);

  Button button = (Button) findViewById(R.id.Button01);

  OnClickListener listener = new OnClickListener() {

   @Override

   public void onClick(View v) {

    cmd = CMD_ADD;

    doAction();

   }

  };

  button.setOnClickListener(listener);

  Button button2 = (Button) findViewById(R.id.Button02);

  OnClickListener listener2 = new OnClickListener() {

   @Override

   public void onClick(View v) {

    cmd = CMD_UPDATE;

    doAction();

   }

  };

  button2.setOnClickListener(listener2);

  Button button3 = (Button) findViewById(R.id.Button03);

  OnClickListener listener3 = new OnClickListener() {

   @Override

   public void onClick(View v) {

    cmd = CMD_QUERY;

    doAction();

   }

  };

  button3.setOnClickListener(listener3);

  Button button4 = (Button) findViewById(R.id.Button04);

  OnClickListener listener4 = new OnClickListener() {

   @Override

   public void onClick(View v) {

    cmd = CMD_QUERY_ALL;

    doAction();

   }

  };

  button4.setOnClickListener(listener4);

  Button button5 = (Button) findViewById(R.id.Button05);

  OnClickListener listener5 = new OnClickListener() {

   @Override

   public void onClick(View v) {

    cmd = CMD_DELETE;

    doAction();

   }

  };

  button5.setOnClickListener(listener5);

  Button button6 = (Button) findViewById(R.id.Button06);

  OnClickListener listener6 = new OnClickListener() {

   @Override

   public void onClick(View v) {

    cmd = CMD_DELETE_ALL;

    doAction();

   }

  };

  button6.setOnClickListener(listener6);

  mHandler = new Handler();

    }

 int cnt = 0;

 private Handler mHandler;

 int cmd = 0;

 final int CMD_ADD = 1;

 final int CMD_UPDATE = 2;

 final int  CMD_QUERY= 3;

 final int CMD_QUERY_ALL = 4;

 final int CMD_DELETE = 5;

 final int CMD_DELETE_ALL = 6;

 People people=new People();

 final static String projection[]=new String[]

                                       {"_id","name","phone","age"};

 class DatabaseThread implements Runnable {

  public void run() {

   if (cmd == CMD_ADD) {

    people.name="robin"+System.currentTimeMillis()%100;

    people.phone=""+System.currentTimeMillis();

    people.age=1;

    ContentValues values=new ContentValues();

    values.put("name", people.name);

    values.put("phone", people.phone);

    values.put("age", people.age);

    Uri uri=getContentResolver().insert(MyProvider.content_URI, values);

    people.id=ContentUris.parseId(uri);

    Log.i("hubin",uri.toString());

   } else if (cmd == CMD_UPDATE) {

    ContentValues values=new ContentValues();

    people.phone=""+System.currentTimeMillis();

    values.put("phone", people.phone);

    Uri uri=ContentUris.withAppendedId(MyProvider.content_URI, people.id);

    getContentResolver().update(uri,values,null,null);

   } else if (cmd == CMD_QUERY) {

    Uri uri=ContentUris.withAppendedId(MyProvider.content_URI, people.id);

    Cursor c=getContentResolver().query(uri, projection, null, null, null);

    People p=get(c);

    printPeople(p);

   } else if (cmd == CMD_QUERY_ALL) {

    Uri uri=MyProvider.content_URI;

    Cursor c=getContentResolver().query(uri, projection, null, null, null);

    List<People> list=getAll(c);

    int total=list.size();

    for(int i=0;i<total;i++)

    {

     printPeople(list.get(i));

    }

   }

   else if (cmd==CMD_DELETE)

   {

    Uri uri=ContentUris.withAppendedId(MyProvider.content_URI, people.id);

    getContentResolver().delete(uri, null, null);

   }

   else if (cmd==CMD_DELETE_ALL)

   {

    Uri uri=MyProvider.content_URI;

    getContentResolver().delete(uri, null, null);

   }

   cnt++;

  }

 }

 void printPeople(People p)

 {

 Log.i(tag, "id:"+p.id);

 Log.i(tag, "name:"+p.name);

 Log.i(tag,"phone:"+p.phone);

 Log.i(tag,"age:"+p.age);

 }

 DatabaseThread dataDealer=new DatabaseThread();

 void doAction() {

  mHandler.post(dataDealer);

 }

 public People get(Cursor c)

 {

  People people=new People();

  try{

   Log.i(tag,"count:"+c.getCount());

   if(c.getCount()>0)

   {

    c.moveToFirst();

    people=new People();

    people.id=c.getLong(0);

    people.name=c.getString(1);

    people.phone=c.getString(2);

    people.age=c.getInt(3);

   }

  }catch(SQLException e)

  {

   Log.i(tag,"",e);

  }

  finally

  {

   if(c!=null&&!c.isClosed())

   {

    c.close();

   }

  }

  return people;

 }

 public List<People> getAll(Cursor c)

 {

  ArrayList<People> ret=new ArrayList<People>();

  try

  {

   int count=c.getCount();

   c.moveToFirst();

   People people;

   for(int i=0;i<count;i++)

   {

    people=new People();

    people.id=c.getLong(0);

    people.name=c.getString(1);

    people.phone=c.getString(2);

    people.age=c.getInt(3);

    ret.add(people);

    c.moveToNext();

   }

   

  }catch(SQLException e)

  {

   Log.i(tag,"",e);

  }

  finally

  {

   if(c!=null&&!c.isClosed())

   {

    c.close();

   }

  }

  return ret;

 }

}

注意:Cursor c不用时要关掉。

文件AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>

<manifest xmlns:android="http://schemas.android.com/apk/res/android"

      package="com.teleca.provider"

      android:versionCode="1"

      android:versionName="1.0">

    <application android:icon="@drawable/icon" android:label="@string/app_name">

        <activity android:name=".Hello"

                  android:label="@string/app_name">

            <intent-filter>

                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />

            </intent-filter>

        </activity>

    <provider android:syncable="true" android:name="MyProvider" android:authorities="com.teleca.PeopleProvider"></provider>

</application>

    <uses-sdk android:minSdkVersion="7" />

</manifest>

list_row.xml文件 
<?xml version="1.0" encoding="utf-8"?> 
<TextView xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="@string/hello" android:id="@+id/list_row"
/>
main.xml文件
 <?xml version="1.0" encoding="utf-8"?>
 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > 
<ListView android:id="@id/android:list" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#00FF00" android:layout_weight="1" android:drawSelectorOnTop="false"/> 
<Button android:text="@+string/Add" android:id="@+id/Button01" android:layout_width="wrap_content" android:layout_height="wrap_content"></Button>
<Button android:text="@+string/DeleteAll" android:id="@+id/Button02" android:layout_width="wrap_content"></Button>
</LinearLayout>
须要的权限:
<uses-permission android:name="android.permission.READ_CONTACTS" />
 <uses-permission android:name="android.permission.WRITE_CONTACTS" />
<uses-permission android:name="android.permission.READ_CONTACTS"></uses-permission>

真的须要这些权限?为什么须要呢?或许是我曾经写错了吧

ContentProvider简要的更多相关文章

  1. 【理论基础】ContentProvider的简要概述

    对于Android应用而言,他们必须相互独立,各自运行在自己的Dalvik虚拟机实例中,如果这些Android应用之间需要实现实时的数据交换——例如我们开发了一个发送短信的程序,当发送短信时需要从联系 ...

  2. ContentResolver + SqliteOpenHelper + ContentProvider 理解

    惭愧,现在才接触到ContentResolver的用法 这个类主要是Android用来实现应用程序之间数据共享的 一个应用程序可以将自己的数据完全暴露出去,外界更本看不到,也不用看到这个应用程序暴露的 ...

  3. Android中使用ContentProvider进行跨进程方法调用

    原文同一时候发表在我的博客 点我进入还能看到很多其它 需求背景 近期接到这样一个需求,须要和别的 App 进行联动交互,比方下载器 App 和桌面 App 进行联动.桌面的 App 能直接显示下载器 ...

  4. ContentResolver,ContentProvider,ContentObserver使用记录

    版权声明:本文出自汪磊的博客,转载请务必注明出处. 本篇博客只是记录一下ContentProvider的使用(这部分工作中用的比较少总是忘记),没有太深入研究.已经熟练掌握使用方式,想深入了解内部机制 ...

  5. android 多应用程序数据共享 ContentProvider和ContentResolver

      android 没有一个可以将所有应用程序数据统一放置的地方,即两个应用程序间的数据不能共享.但ContentProvider与ContentResolver可以解决多应用程序数据共享. 我们都知 ...

  6. Android之ContentProvider数据存储

    一.ContentProvider保存数据介绍 一个程序可以通过实现一个ContentProvider的抽象接口将自己的数据完全暴露出去,而且ContentProvider是以类似数据库中表的方式将数 ...

  7. Xamarin.Android之ContentProvider

    一.前言 掌握了如何使用SQLiteOpenHelper之后,我们就可以进行下一步的学习.本章我们将会学习如何使用ContentProvider来将数据库方面的操作封装起来,同时它还可以供其他应用访问 ...

  8. ContentProvider域名替换小工具

    开发项目域名想怎么换就怎么换,就是这么任性! 这是一个很有意思的小工具! 这是一个方便开发人员和测试人员的小工具!! 吐槽: 一直在做Android开发,一直总有一个问题存在:做自己公司的apk开发时 ...

  9. JavaScript权威设计--JavaScript函数(简要学习笔记十一)

    1.函数调用的四种方式 第三种:构造函数调用 如果构造函数调用在圆括号内包含一组实参列表,先计算这些实参表达式,然后传入函数内.这和函数调用和方法调用是一致的.但如果构造函数没有形参,JavaScri ...

随机推荐

  1. [置顶] 63行代码完美实现html5 贪吃蛇游戏

    以前也很少关注html5,感觉选择html已经慢慢成为趋势,想了解下.就找了个游戏学习了,写完这个游戏感觉html5和js结合很紧密,如果js不是特别好.估计需要先补习下js,这个只是个人的建议,不一 ...

  2. SDUTOJ 2128 树结构练习——排序二叉树的中序遍历

    watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvUl9NaXNheWE=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA ...

  3. SQL Server 基础 01 数据库、表操作

    对着书慢慢学习,一天一点点! 数据库操作 (create.alter.drop)  --3-3-1 /create database 语句创建数据库 create database testSQL - ...

  4. Jsp的include指令静态导入和动态导入的区别

    1.什么是静态导入? 静态导入指的是,将一个外部文件嵌入到当前JSP文件中,同时解析这个页面的JSP语句,它会把目标页面的其他编译指令也包含进来. include的静态导入指令使用语法: <%@ ...

  5. cocos2dx-lua捕获用户touch事件的几种方式

    这里仅仅针对lua 1.为每一个关心的事件注冊回调函数 详细分为下面几种 1>单点触摸 注冊函数为 cc.Handler.EVENT_TOUCH_BEGAN      = 40 cc.Handl ...

  6. Nodejs 项目开发

    最近这几个月都在学习nodejs. 国内nodejs的资料相对较少,就我所搜索到的,CSDN目前的代码托管平台有不少从github弄过来的开源镜像,其它的不错的社区有cnodejs,byvoid的个人 ...

  7. CSDN-Code平台使用过程中的5点经验教训

    昨天又创建了一个项目,fucms,可是本地一直没有权限提交,搞了非常久,试了几十次,都不行,我是非常的灰心和郁闷.  刚刚,和CSDN-Code的官方客服咨询了非常久非常久,最终摸索出来了一些心得体会 ...

  8. Java byte数据类型详解

    public static String bytes2HexString(byte[] b) { String ret = ""; for (int i = 0; i < b ...

  9. 重操JS旧业第八弹:面向对象与继承

    js里面没有语言语法层面的继承机制,但这并不意味着js就不能实现继承,利用js属性和方法动态性来模拟实现继承,通过总结大概有如下方法实现: 1 原型链继承 我们知道原型在对象中扮演着重要的角色,函数本 ...

  10. JSP 的9个内置对象

    JSP内置对象:我们在使用JSP进行页面编程时可以直接使用而不需自己创建的一些Web容器已为用户创建好的JSP内置对象.如request,session,response,out等. 下面就JSP2. ...