Android之ContentProvider数据存储
一、ContentProvider保存数据介绍
一个程序可以通过实现一个ContentProvider的抽象接口将自己的数据完全暴露出去,而且ContentProvider是以类似数据库中表的方式将数据暴露的。那么外界获取其提供的数据,也就应该与从数据库中获取数据的操作基本一样,只不过是采用URL来表示外界需要访问的“数据库”。
ContentProvider提供了一种多应用间数据共享的方式。
ContentProvider是个实现了一组用于提供其他应用程序存取数据的标准方法的类。应用程序可以在ContentProvider中执行如下操作:查询数据、修改数据、添加数据、删除数据。
标准的ContentProvider:Android提供了一些已经在系统中实现的标准ContentProvider,比如联系人信息,图片库等等,可以用这些ContentProvider来访问设备上存储的联系人信息、图片等等。
在ContentProvider中使用的查询字符串有别于标准的SQL查询,很多诸如select、add、delete、modify等操作都使用一种特殊的URL进行,这种URL由3部分组成,“content://”,代表数据的路径和一个可选的表示数据的ID。
content://media/internal/images 这个URL将返回设备上存储的所有图片
content://contacts/people/ 这个URL将返回设备上的所有联系人信息
content://contacts/people/45 这个URL返回单个结果(联系人信息中ID为45的联系人记录)
如果想要存储字节型数据,比如位图文件等,那保存该数据的数据列其实是一个表示实际保存保存文件的URL字符串,客户端通过它来读取对应的文件数据,处理这种数据类型的ContentProvider需要实现一个名为_data的字段,_data字段列出了该文件在Android文件系统上的精确路径。这个字段不仅是供客户端使用,而且也可以供ContentResolver使用。客户端可以调用ContentResolver.openOutputStream()方法来处理该URL指向的文件资源,如果是ContentResolver本身的话,由于其持有的权限比客户端要高,所以它能直接访问该数据文件。
二、使用方法
大多数ContentProvider使用Android文件系统或者SQLite数据库来保持数据,但是也可以以任何方式来存储。本例用SQLite数据库来保持数据。
1.创建一个接口,定义了一个名为CONTENT_URL,并且是public static final的Uri类型的类变量,必须为其指定一个唯一的字符串值,最好的方案是类的全称,和数据列的名称。
- public interface IProivderMetaData {
- public static final String AUTHORITY = "com.zhangmiao.datastoragedemo";
- public static final String DB_NAME = "book.db";
- public static final int VERSION = 1;
- public interface BookTableMetaData extends BaseColumns {
- public static final String TABLE_NAME = "book";
- public static final Uri CONTENT_URI = Uri.parse("content://"
- + AUTHORITY + "/" + TABLE_NAME);
- public static final String BOOK_ID = "_id";
- public static final String BOOK_NAME = "name";
- public static final String BOOK_PUBLISHER = "publisher";
- public static final String SORT_ORDER = "_id desc";
- public static final String CONTENT_LIST = "vnd.android.cursor.dir/vnd.bookprovider.book";
- public static final String CONTENT_ITEM = "vnd.android.cursor.item/vnd.bookprovider.book";
- }
- }
2.实现SQLiteOpenHelper
- public class ContentProviderDBHelper extends SQLiteOpenHelper implements IProivderMetaData {
- private static final String TAG = "ContentProviderDBHelper";
- public ContentProviderDBHelper(Context context) {
- super(context, DB_NAME, null, VERSION);
- }
- @Override
- public void onCreate(SQLiteDatabase db) {
- ...
- }
- @Override
- public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
- ...
- }
- }
3.创建一个继承了ContentProvider父类的类
- public class ContentProviderDBHelper extends SQLiteOpenHelper implements IProivderMetaData {
- public ContentProviderDBHelper(Context context) {
- super(context, DB_NAME, null, VERSION);
- }
- @Override
- public void onCreate(SQLiteDatabase db) {
- ...
- }
- @Override
- public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
- ...
- }
- }
4.在AndroidManifest.xml中使用标签来设置调用ContentProvider。
- <provider
- android:authorities="com.zhangmiao.datastoragedemo"
- android:name=".BookContentProvider"/>
5.增加数据
- mContentResolver = getContentResolver();
- String[] bookNames = new String[]{"Chinese", "Math", "English", "Sports"};
- String[] bookPublishers = new String[]{"XinHua", "GongXin", "DianZi", "YouDian"};
- for (int i = 0; i < bookNames.length; i++) {
- ContentValues values = new ContentValues();
- values.put(IProivderMetaData.BookTableMetaData.BOOK_NAME, bookNames[i]);
- values.put(IProivderMetaData.BookTableMetaData.BOOK_PUBLISHER, bookPublishers[i]);
- mContentResolver.insert(IProivderMetaData.BookTableMetaData.CONTENT_URI, values);
- }
6.删除数据
- String bookId = "1";
- if (!"".equals(bookId)) {
- ContentValues values1 = new ContentValues();
- values1.put(IProivderMetaData.BookTableMetaData.BOOK_ID,bookId);
- mContentResolver.delete(Uri.withAppendedPath(
IProivderMetaData.BookTableMetaData.CONTENT_URI,- bookId),
"_id = ?",- new String[]{bookId}
);- } else {
- mContentResolver.delete(
- IProivderMetaData.BookTableMetaData.CONTENT_URI,
- null,
- null
);- }
7.查询数据
- Cursor cursor = mContentResolver.query(IProivderMetaData.BookTableMetaData.CONTENT_URI,
null, null, null, null);- String text = "";
- if (cursor != null) {
- while (cursor.moveToNext()) {
- String bookIdText =
cursor.getString(cursor.getColumnIndex(IProivderMetaData.BookTableMetaData.BOOK_ID));- String bookNameText =
cursor.getString(cursor.getColumnIndex(IProivderMetaData.BookTableMetaData.BOOK_NAME));- String bookPublisherText =
- cursor.getString(cursor.getColumnIndex(IProivderMetaData.BookTableMetaData.BOOK_PUBLISHER));
- text += "id = " + bookIdText + ",name = " + bookNameText +
",publisher = " + bookPublisherText + "\n";- }
- cursor.close();
- mTableInfo.setText(text);
- }
8.更新数据
- String bookId1 = "2";
- String bookName = "Art";
- String bookPublisher = "TieDao";
- ContentValues values2 = new ContentValues();
- values2.put(IProivderMetaData.BookTableMetaData.BOOK_NAME,bookName);
- values2.put(IProivderMetaData.BookTableMetaData.BOOK_PUBLISHER,bookPublisher);
- if ("".equals(bookId1)) {
- mContentResolver.update(IProivderMetaData.BookTableMetaData.CONTENT_URI,
- values2, null, null);
- } else {
- mContentResolver.update(Uri.withAppendedPath(IProivderMetaData.BookTableMetaData.CONTENT_URI, bookId1),
- values2, "_id = ? ", new String[]{bookId1}
);- }
三、小案例
1.添加strings.xml文件
- <string name="content_provider">ContentProvider</string>
- <string name="add_data">增加数据</string>
- <string name="delete_data">删除数据</string>
- <string name="update_data">更改数据</string>
- <string name="query_data">查询数据</string>
2.修改activity_main.xml文件
- <?xml version="1.0" encoding="utf-8"?>
- <android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:fitsSystemWindows="true"
- tools:context="com.zhangmiao.datastoragedemo.MainActivity">
- <LinearLayout
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:orientation="vertical">
- <TextView
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_gravity="center_horizontal"
- android:text="@string/content_provider" />
- <LinearLayout
- android:layout_width="match_parent"
- android:layout_height="wrap_content"
- android:layout_marginBottom="@dimen/fab_margin"
- android:layout_marginTop="@dimen/fab_margin"
- android:orientation="horizontal">
- <Button
- android:id="@+id/provider_add"
- android:layout_width="0dp"
- android:layout_height="wrap_content"
- android:layout_weight="1"
- android:text="@string/add_data" />
- <Button
- android:id="@+id/provider_delete"
- android:layout_width="0dp"
- android:layout_height="wrap_content"
- android:layout_weight="1"
- android:text="@string/delete_data" />
- <Button
- android:id="@+id/provider_update"
- android:layout_width="0dp"
- android:layout_height="wrap_content"
- android:layout_weight="1"
- android:text="@string/update_data" />
- <Button
- android:id="@+id/provider_query"
- android:layout_width="0dp"
- android:layout_height="wrap_content"
- android:layout_weight="1"
- android:text="@string/query_data" />
- </LinearLayout>
- <TextView
- android:id="@+id/table_info"
- android:layout_width="match_parent"
- android:layout_height="wrap_content"
- android:text="@string/app_name" />
- </LinearLayout>
- </android.support.design.widget.CoordinatorLayout>
3.添加IProviderMetaData接口
- package com.zhangmiao.datastoragedemo;
- import android.net.Uri;
- import android.provider.BaseColumns;
- /**
- * Created by zhangmiao on 2016/12/20.
- */
- public interface IProviderMetaData {
- public static final String AUTHORITY = "com.zhangmiao.datastoragedemo";
- public static final String DB_NAME = "book.db";
- public static final int VERSION = 1;
- public interface BookTableMetaData extends BaseColumns {
- public static final String TABLE_NAME = "book";
- public static final Uri CONTENT_URI = Uri.parse("content://"
- + AUTHORITY + "/" + TABLE_NAME);
- public static final String BOOK_ID = "_id";
- public static final String BOOK_NAME = "name";
- public static final String BOOK_PUBLISHER = "publisher";
- public static final String SORT_ORDER = "_id desc";
- public static final String CONTENT_LIST = "vnd.android.cursor.dir/vnd.bookprovider.book";
- public static final String CONTENT_ITEM = "vnd.android.cursor.item/vnd.bookprovider.book";
- }
- }
4.添加ContentProviderDBHelper类
- package com.zhangmiao.datastoragedemo;
- import android.content.Context;
- import android.database.sqlite.SQLiteDatabase;
- import android.database.sqlite.SQLiteOpenHelper;
- import android.util.Log;
- /**
- * Created by zhangmiao on 2016/12/20.
- */
- public class ContentProviderDBHelper extends SQLiteOpenHelper implements IProviderMetaData {
- private static final String TAG = "ContentProviderDBHelper";
- public ContentProviderDBHelper(Context context) {
- super(context, DB_NAME, null, VERSION);
- }
- @Override
- public void onCreate(SQLiteDatabase db) {
- String TABLESQL = "CREATE TABLE IF NOT EXISTS "
- + BookTableMetaData.TABLE_NAME + " ("
- + BookTableMetaData.BOOK_ID + " INTEGER PRIMARY KEY AUTOINCREMENT,"
- + BookTableMetaData.BOOK_NAME + " VARCHAR,"
- + BookTableMetaData.BOOK_PUBLISHER + " VARCHAR)";
- db.execSQL(TABLESQL);
- }
- @Override
- public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
- Log.w(TAG, "Upgrading database from version " + oldVersion + "to"
- + newVersion + ", which will destroy all old data");
- db.execSQL("DROP TABLE IF EXISTS " + DB_NAME);
- onCreate(db);
- }
- }
5.添加BookContentProvider类
- package com.zhangmiao.datastoragedemo;
- import android.content.ContentProvider;
- import android.content.ContentUris;
- import android.content.ContentValues;
- import android.content.UriMatcher;
- import android.database.Cursor;
- import android.database.sqlite.SQLiteDatabase;
- import android.net.Uri;
- import android.support.annotation.Nullable;
- import android.util.Log;
- /**
- * Created by zhangmiao on 2016/12/21.
- */
- public class BookContentProvider extends ContentProvider {
- private static final String TAG = "BookContentProvider";
- private static UriMatcher uriMatcher = null;
- private static final int BOOKS = 1;
- private static final int BOOK = 2;
- private ContentProviderDBHelper dbHelper;
- private SQLiteDatabase db;
- static {
- uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
- uriMatcher.addURI(IProviderMetaData.AUTHORITY,
- IProviderMetaData.BookTableMetaData.TABLE_NAME, BOOKS);
- uriMatcher.addURI(IProviderMetaData.AUTHORITY,
- IProviderMetaData.BookTableMetaData.TABLE_NAME + "/#",
- BOOK);
- }
- @Override
- public boolean onCreate() {
- dbHelper = new ContentProviderDBHelper(getContext());
- return (dbHelper == null) ? false : true;
- }
- @Nullable
- @Override
- public String getType(Uri uri) {
- switch (uriMatcher.match(uri)) {
- case BOOKS:
- return IProviderMetaData.BookTableMetaData.CONTENT_LIST;
- case BOOK:
- return IProviderMetaData.BookTableMetaData.CONTENT_ITEM;
- default:
- throw new IllegalArgumentException("This is a unKnow Uri"
- + uri.toString());
- }
- }
- @Nullable
- @Override
- public Uri insert(Uri uri, ContentValues values) {
- switch (uriMatcher.match(uri)) {
- case BOOKS:
- db = dbHelper.getWritableDatabase();
- long rowId = db.insert(
- IProviderMetaData.BookTableMetaData.TABLE_NAME,
- IProviderMetaData.BookTableMetaData.BOOK_ID,
- values);
- Uri insertUri = Uri.withAppendedPath(uri, "/" + rowId);
- Log.i(TAG, "insertUri:" + insertUri.toString());
- getContext().getContentResolver().notifyChange(uri, null);
- return insertUri;
- case BOOK:
- default:
- throw new IllegalArgumentException("This is a unKnow Uri"
- + uri.toString());
- }
- }
- @Nullable
- @Override
- public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
- db = dbHelper.getReadableDatabase();
- switch (uriMatcher.match(uri)) {
- case BOOKS:
- return db.query(IProviderMetaData.BookTableMetaData.TABLE_NAME,
- projection, selection, selectionArgs, null, null,
- sortOrder);
- case BOOK:
- long id = ContentUris.parseId(uri);
- String where = "_id=" + id;
- if (selection != null && !"".equals(selection)) {
- where = selection + " and " + where;
- }
- return db.query(IProviderMetaData.BookTableMetaData.TABLE_NAME,
- projection, where, selectionArgs, null, null, sortOrder);
- default:
- throw new IllegalArgumentException("This is a unKnow Uri"
- + uri.toString());
- }
- }
- @Override
- public int delete(Uri uri, String selection, String[] selectionArgs) {
- db = dbHelper.getWritableDatabase();
- switch (uriMatcher.match(uri)) {
- case BOOKS:
- return db.delete(IProviderMetaData.BookTableMetaData.TABLE_NAME,
- selection, selectionArgs);
- case BOOK:
- long id = ContentUris.parseId(uri);
- String where = "_id=" + id;
- if (selection != null && !"".equals(selection)) {
- where = selection + " and " + where;
- }
- return db.delete(IProviderMetaData.BookTableMetaData.TABLE_NAME,
- selection, selectionArgs);
- default:
- throw new IllegalArgumentException("This is a unKnow Uri"
- + uri.toString());
- }
- }
- @Override
- public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
- db = dbHelper.getWritableDatabase();
- switch (uriMatcher.match(uri)) {
- case BOOKS:
- return db.update(IProviderMetaData.BookTableMetaData.TABLE_NAME,
- values, null, null);
- case BOOK:
- long id = ContentUris.parseId(uri);
- String where = "_id=" + id;
- if (selection != null && !"".equals(selection)) {
- where = selection + " and " + where;
- }
- return db.update(IProviderMetaData.BookTableMetaData.TABLE_NAME,
- values, selection, selectionArgs);
- default:
- throw new IllegalArgumentException("This is a unKnow Uri"
- + uri.toString());
- }
- }
- }
6.修改AndroidManifest.xml文件
- <provider
- android:authorities="com.zhangmiao.datastoragedemo"
- android:name=".BookContentProvider"/>
7.修改MainActivity
- package com.zhangmiao.datastoragedemo;
- import android.content.ContentResolver;
- import android.content.ContentValues;
- import android.database.Cursor;
- import android.net.*;
- import android.os.Bundle;
- import android.support.v7.app.AppCompatActivity;
- import android.util.Log;
- import android.view.View;
- import android.widget.Button;
- import android.widget.TextView;import java.util.ArrayList;
- import java.util.List;
- public class MainActivity extends AppCompatActivity implements View.OnClickListener {private ContentResolver mContentResolver;
- private BookContentProvider mBookContentProvider;private TextView mTableInfo;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- Log.v("MainActivity", "onCreate");
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- Button cpAdd = (Button) findViewById(R.id.provider_add);
- Button cpDelete = (Button) findViewById(R.id.provider_delete);
- Button cpUpdate = (Button) findViewById(R.id.provider_update);
- Button cpQuery = (Button) findViewById(R.id.provider_query);
- mTableInfo = (TextView) findViewById(R.id.table_info);
- cpAdd.setOnClickListener(this);
- cpDelete.setOnClickListener(this);
- cpQuery.setOnClickListener(this);
- cpUpdate.setOnClickListener(this);
- }
- @Override
- public void onClick(View v) {
- switch (v.getId()) {case R.id.provider_add:
- mContentResolver = getContentResolver();
- String[] bookNames = new String[]{"Chinese", "Math", "English", "Sports"};
- String[] bookPublishers = new String[]{"XinHua", "GongXin", "DianZi", "YouDian"};
- for (int i = 0; i < bookNames.length; i++) {
- ContentValues values = new ContentValues();
- values.put(IProviderMetaData.BookTableMetaData.BOOK_NAME, bookNames[i]);
- values.put(IProviderMetaData.BookTableMetaData.BOOK_PUBLISHER, bookPublishers[i]);
- mContentResolver.insert(IProviderMetaData.BookTableMetaData.CONTENT_URI, values);
- }
- break;
- case R.id.provider_delete:
- String bookId = "1";
- if (!"".equals(bookId)) {
- ContentValues values1 = new ContentValues();
- values1.put(IProviderMetaData.BookTableMetaData.BOOK_ID,
- bookId);
- mContentResolver.delete(
- Uri.withAppendedPath(
- IProviderMetaData.BookTableMetaData.CONTENT_URI,
- bookId
- ), "_id = ?",
- new String[]{bookId}
- );
- } else {
- mContentResolver.delete(
- IProviderMetaData.BookTableMetaData.CONTENT_URI,
- null,
- null
- );
- }
- break;
- case R.id.provider_query:
- Cursor cursor = mContentResolver.query(IProviderMetaData.BookTableMetaData.CONTENT_URI, null, null, null, null);
- String text = "";
- if (cursor != null) {
- while (cursor.moveToNext()) {
- String bookIdText =
- cursor.getString(cursor.getColumnIndex(IProviderMetaData.BookTableMetaData.BOOK_ID));
- String bookNameText =
- cursor.getString(cursor.getColumnIndex(IProviderMetaData.BookTableMetaData.BOOK_NAME));
- String bookPublisherText =
- cursor.getString(cursor.getColumnIndex(IProviderMetaData.BookTableMetaData.BOOK_PUBLISHER));
- text += "id = " + bookIdText + ",name = " + bookNameText + ",publisher = " + bookPublisherText + "\n";
- }
- cursor.close();
- mTableInfo.setText(text);
- }
- break;
- case R.id.provider_update:
- String bookId1 = "2";
- String bookName = "Art";
- String bookPublisher = "TieDao";
- ContentValues values2 = new ContentValues();
- values2.put(IProviderMetaData.BookTableMetaData.BOOK_NAME,
- bookName);
- values2.put(IProviderMetaData.BookTableMetaData.BOOK_PUBLISHER,
- bookPublisher);
- if ("".equals(bookId1)) {
- mContentResolver.update(
- IProviderMetaData.BookTableMetaData.CONTENT_URI,
- values2, null, null);
- } else {
- mContentResolver.update(
- Uri.withAppendedPath(
- IProviderMetaData.BookTableMetaData.CONTENT_URI, bookId1),
- values2, "_id = ? ", new String[]{bookId1});
- }
- break;default:
- Log.v("MainActivity", "default");
- break;
- }
- }
- }
代码下载地址:https://github.com/ZhangMiao147/DataStorageDemo
参考文章:https://liuzhichao.com/p/562.html
Android之ContentProvider数据存储的更多相关文章
- Android五种数据存储方式
android 五种数据存储 :SharePreferences.SQLite.Contert Provider.File.网络存储 Android系统提供了四种存储数据方式.分别为:SharePre ...
- Android——几种数据存储应用浅谈
(1)android中的数据存储主要有五种方式: 第一种.sharedPreferences存储数据, 适用范围:保存少量的数据,且这些数据的格式非常简单:字符串型.基本类型的值.比如应用程序的各种配 ...
- 67.Android中的数据存储总结
转载:http://mp.weixin.qq.com/s?__biz=MzIzMjE1Njg4Mw==&mid=2650117688&idx=1&sn=d6c73f9f04d0 ...
- Android Learning:数据存储方案归纳与总结
前言 最近在学习<第一行android代码>和<疯狂android讲义>,我的感触是Android应用的本质其实就是数据的处理,包括数据的接收,存储,处理以及显示,我想针对这几 ...
- Android中的数据存储
Android中的数据存储主要分为三种基本方法: 1.利用shared preferences存储一些轻量级的键值对数据. 2.传统文件系统. 3.利用SQLite的数据库管理系统. 对SharedP ...
- Android中的数据存储(二):文件存储 2017-05-25 08:16 35人阅读 评论(0) 收藏
文件存储 这是本人(菜鸟)学习android数据存储时接触的有关文件存储的知识以及本人自己写的简单地demo,为初学者学习和使用文件存储提供一些帮助.. 如果有需要查看SharedPreference ...
- Android下的数据存储与訪问 --- 以文件的形式
Android下的数据存储与訪问 --- 以文件的形式 1.1 储存文件存放在手机内存中: // *** 储存数据到 /data/data/包名/files/jxn.txt文件里 String dat ...
- Android开发8:数据存储(二)——SQLite数据库和ContentProvider的使用
前言 啦啦啦各位小伙伴们许久不见了~学期末和过年期间自己忙着做其他事没能及时更新Android开发系列课程的博客,实在是罪过罪过~ 好啦~废话不多说,进入我们今天的主题.今天我们将和大家学习其他的数据 ...
- Android之网络数据存储
一.网络保存数据介绍 可以使用网络来保存数据,在需要的时候从网络上获取数据,进而显示在App中. 用网络保存数据的方法有很多种,对于不同的网络数据采用不同的上传与获取方法. 本文利用LeanCloud ...
随机推荐
- JAVA语言中的修饰符
JAVA语言中的修饰符 -----------------------------------------------01--------------------------------------- ...
- 07. Web大前端时代之:HTML5+CSS3入门系列~H5 地理位置
Web大前端时代之:HTML5+CSS3入门系列:http://www.cnblogs.com/dunitian/p/5121725.html 源码:https://github.com/duniti ...
- 水印第三版 ~ 变态水印(这次用Magick.NET来实现,附需求分析和源码)
技能 汇总:http://www.cnblogs.com/dunitian/p/4822808.html#skill 以前的水印,只是简单走起,用的是原生态的方法.现在各种变态水印,于是就不再用原生态 ...
- 23种设计模式--观察者模式-Observer Pattern
一.观察者模式的介绍 观察者模式从字面的意思上理解,肯定有两个对象一个是观察者,另外一个是被观察者,观察者模式就是当被观察者发生改变得时候发送通知给观察者,当然这个观察者可以是多个对象,在项 ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(81)-数据筛选(万能查询)
系列目录 前言 听标题的名字似乎是一个非常牛X复杂的功能,但是实际上它确实是非常复杂的,我们本节将演示如何实现对数据,进行组合查询(数据筛选) 我们都知道Excel中是如何筛选数据的.就像下面一样 他 ...
- scala练习题1 基础知识
1, 在scala REPL中输入3. 然后按下tab键,有哪些方法可以被调用? 24个方法可以被调用, 8个基本类型: 基本的操作符, 等: 2,在scala REPL中,计算3的平方根,然 ...
- Maven多模块,Dubbo分布式服务框架,SpringMVC,前后端分离项目,基础搭建,搭建过程出现的问题
现互联网公司后端架构常用到Spring+SpringMVC+MyBatis,通过Maven来构建.通过学习,我已经掌握了基本的搭建过程,写下基础文章为而后的深入学习奠定基础. 首先说一下这篇文章的主要 ...
- 【干货分享】流程DEMO-补打卡
流程名: 补打卡申请 业务描述: 当员工在该出勤的工作日出勤但漏打卡时,于一周内填写补打卡申请. 流程相关文件: 流程包.xml 流程说明: 直接导入流程包文件,即可使用本流程 表单: 流程: 图片 ...
- NV显卡Ubuntu14.04更新软件导致登录死循环,不过可以进入tty模式
注意:此方法只适用于nv显卡的电脑! 在网上寻找各种方法无果的情况下,选择重新安装显卡驱动,成功登录进入图形界面. 一.首先需要在另外一台电脑(windows系统也可以)上下载NVIDIA相应显卡驱动 ...
- Jquery EasyUI 开发实录
有好几年没有用过EasyUI了,最近在外包做的一个项目中新增功能时,又用到了,本以为和按照以前那样用就可以了,可当我真正用的时候,发现许多地方不一样了,就连官网的文档都更新了,最突出的就是不知道什么时 ...